diff --git a/src/java.base/share/classes/java/nio/file/Files.java b/src/java.base/share/classes/java/nio/file/Files.java index 0d9bba9cbca..6e00c7e484b 100644 --- a/src/java.base/share/classes/java/nio/file/Files.java +++ b/src/java.base/share/classes/java/nio/file/Files.java @@ -80,7 +80,6 @@ import java.util.stream.StreamSupport; import jdk.internal.util.ArraysSupport; import sun.nio.ch.FileChannelImpl; import sun.nio.cs.UTF_8; -import sun.nio.fs.AbstractFileSystemProvider; /** * This class consists exclusively of static methods that operate on files, @@ -1596,7 +1595,7 @@ public final class Files { byte[] buffer1 = new byte[BUFFER_SIZE]; byte[] buffer2 = new byte[BUFFER_SIZE]; try (InputStream in1 = Files.newInputStream(path); - InputStream in2 = Files.newInputStream(path2);) { + InputStream in2 = Files.newInputStream(path2)) { long totalRead = 0; while (true) { int nRead1 = in1.readNBytes(buffer1, 0, BUFFER_SIZE); @@ -2312,14 +2311,10 @@ public final class Files { * method denies read access to the file. */ public static boolean isDirectory(Path path, LinkOption... options) { - if (options.length == 0) { - FileSystemProvider provider = provider(path); - if (provider instanceof AbstractFileSystemProvider) - return ((AbstractFileSystemProvider)provider).isDirectory(path); - } - try { - return readAttributes(path, BasicFileAttributes.class, options).isDirectory(); + var attrs = provider(path) + .readAttributesIfExists(path, BasicFileAttributes.class, options); + return (attrs != null) && attrs.isDirectory(); } catch (IOException ioe) { return false; } @@ -2355,14 +2350,10 @@ public final class Files { * method denies read access to the file. */ public static boolean isRegularFile(Path path, LinkOption... options) { - if (options.length == 0) { - FileSystemProvider provider = provider(path); - if (provider instanceof AbstractFileSystemProvider) - return ((AbstractFileSystemProvider)provider).isRegularFile(path); - } - try { - return readAttributes(path, BasicFileAttributes.class, options).isRegularFile(); + var attrs = provider(path) + .readAttributesIfExists(path, BasicFileAttributes.class, options); + return (attrs != null) && attrs.isRegularFile(); } catch (IOException ioe) { return false; } @@ -2504,7 +2495,7 @@ public final class Files { * the path to the file to test * @param options * options indicating how symbolic links are handled - * . + * * @return {@code true} if the file exists; {@code false} if the file does * not exist or its existence cannot be determined. * @@ -2517,27 +2508,7 @@ public final class Files { * @see FileSystemProvider#checkAccess */ public static boolean exists(Path path, LinkOption... options) { - if (options.length == 0) { - FileSystemProvider provider = provider(path); - if (provider instanceof AbstractFileSystemProvider) - return ((AbstractFileSystemProvider)provider).exists(path); - } - - try { - if (followLinks(options)) { - provider(path).checkAccess(path); - } else { - // attempt to read attributes without following links - readAttributes(path, BasicFileAttributes.class, - LinkOption.NOFOLLOW_LINKS); - } - // file exists - return true; - } catch (IOException x) { - // does not exist or unable to determine if file exists - return false; - } - + return provider(path).exists(path, options); } /** diff --git a/src/java.base/share/classes/java/nio/file/spi/FileSystemProvider.java b/src/java.base/share/classes/java/nio/file/spi/FileSystemProvider.java index 965ab557253..73575829194 100644 --- a/src/java.base/share/classes/java/nio/file/spi/FileSystemProvider.java +++ b/src/java.base/share/classes/java/nio/file/spi/FileSystemProvider.java @@ -1170,4 +1170,122 @@ public abstract class FileSystemProvider { public abstract void setAttribute(Path path, String attribute, Object value, LinkOption... options) throws IOException; + + /** + * Tests whether a file exists. This method works in exactly the + * manner specified by the {@link Files#exists(Path, LinkOption...)} method. + * + * @implSpec + * The default implementation of this method invokes the + * {@link #checkAccess(Path, AccessMode...)} method when symbolic links + * are followed. If the option {@link LinkOption#NOFOLLOW_LINKS NOFOLLOW_LINKS} + * is present then symbolic links are not followed and the method + * {@link #readAttributes(Path, Class, LinkOption...)} is called + * to determine whether a file exists. + * + * @param path + * the path to the file to test + * @param options + * options indicating how symbolic links are handled + * + * @return {@code true} if the file exists; {@code false} if the file does + * not exist or its existence cannot be determined. + * + * @throws SecurityException + * In the case of the default provider, the {@link + * SecurityManager#checkRead(String)} is invoked to check + * read access to the file. + * + * @since 20 + */ + public boolean exists(Path path, LinkOption... options) { + try { + if (followLinks(options)) { + this.checkAccess(path); + } else { + // attempt to read attributes without following links + readAttributes(path, BasicFileAttributes.class, LinkOption.NOFOLLOW_LINKS); + } + // file exists + return true; + } catch (IOException x) { + // does not exist or unable to determine if file exists + return false; + } + } + + /** + * Reads a file's attributes as a bulk operation if it exists. + * + *

The {@code type} parameter is the type of the attributes required + * and this method returns an instance of that type if supported. All + * implementations support a basic set of file attributes and so invoking + * this method with a {@code type} parameter of {@code + * BasicFileAttributes.class} will not throw {@code + * UnsupportedOperationException}. + * + *

The {@code options} array may be used to indicate how symbolic links + * are handled for the case that the file is a symbolic link. By default, + * symbolic links are followed and the file attribute of the final target + * of the link is read. If the option {@link LinkOption#NOFOLLOW_LINKS + * NOFOLLOW_LINKS} is present then symbolic links are not followed. + * + *

It is implementation specific if all file attributes are read as an + * atomic operation with respect to other file system operations. + * + * @implSpec + * The default implementation of this method invokes the + * {@link #readAttributes(Path, Class, LinkOption...)} method + * to read the file's attributes. + * + * @param + * The {@code BasicFileAttributes} type + * @param path + * the path to the file + * @param type + * the {@code Class} of the file attributes required + * to read + * @param options + * options indicating how symbolic links are handled + * + * @return the file attributes or null if the file does not exist + * + * @throws UnsupportedOperationException + * if an attributes of the given type are not supported + * @throws IOException + * if an I/O error occurs + * @throws SecurityException + * In the case of the default provider, a security manager is + * installed, its {@link SecurityManager#checkRead(String) checkRead} + * method is invoked to check read access to the file. If this + * method is invoked to read security sensitive attributes then the + * security manager may be invoked to check for additional permissions. + * + * @since 20 + */ + public A readAttributesIfExists(Path path, + Class type, + LinkOption... options) + throws IOException + { + try { + return readAttributes(path, type, options); + } catch (NoSuchFileException ignore) { + return null; + } + } + + private static boolean followLinks(LinkOption... options) { + boolean followLinks = true; + for (LinkOption opt: options) { + if (opt == LinkOption.NOFOLLOW_LINKS) { + followLinks = false; + continue; + } + if (opt == null) + throw new NullPointerException(); + throw new AssertionError("Should not get here"); + } + return followLinks; + } } diff --git a/src/java.base/share/classes/sun/nio/fs/AbstractFileSystemProvider.java b/src/java.base/share/classes/sun/nio/fs/AbstractFileSystemProvider.java index a9007abd499..3d266e6d5b9 100644 --- a/src/java.base/share/classes/sun/nio/fs/AbstractFileSystemProvider.java +++ b/src/java.base/share/classes/sun/nio/fs/AbstractFileSystemProvider.java @@ -27,7 +27,6 @@ package sun.nio.fs; import java.nio.file.Path; import java.nio.file.LinkOption; -import java.nio.file.attribute.BasicFileAttributes; import java.nio.file.spi.FileSystemProvider; import java.io.IOException; import java.util.Map; @@ -110,51 +109,6 @@ public abstract class AbstractFileSystemProvider extends FileSystemProvider { return implDelete(file, false); } - /** - * Tests whether a file is a directory. - * - * @return {@code true} if the file is a directory; {@code false} if - * the file does not exist, is not a directory, or it cannot - * be determined if the file is a directory or not. - */ - public boolean isDirectory(Path file) { - try { - return readAttributes(file, BasicFileAttributes.class).isDirectory(); - } catch (IOException ioe) { - return false; - } - } - - /** - * Tests whether a file is a regular file with opaque content. - * - * @return {@code true} if the file is a regular file; {@code false} if - * the file does not exist, is not a regular file, or it - * cannot be determined if the file is a regular file or not. - */ - public boolean isRegularFile(Path file) { - try { - return readAttributes(file, BasicFileAttributes.class).isRegularFile(); - } catch (IOException ioe) { - return false; - } - } - - /** - * Checks the existence of a file. - * - * @return {@code true} if the file exists; {@code false} if the file does - * not exist or its existence cannot be determined. - */ - public boolean exists(Path file) { - try { - checkAccess(file); - return true; - } catch (IOException ioe) { - return false; - } - } - /** * Returns a path name as bytes for a Unix domain socket. * Different encodings may be used for these names on some platforms. diff --git a/src/java.base/unix/classes/sun/nio/fs/UnixFileAttributes.java b/src/java.base/unix/classes/sun/nio/fs/UnixFileAttributes.java index 4f0b159985b..3257b5bf32e 100644 --- a/src/java.base/unix/classes/sun/nio/fs/UnixFileAttributes.java +++ b/src/java.base/unix/classes/sun/nio/fs/UnixFileAttributes.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2008, 2019, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2008, 2022, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it @@ -74,6 +74,19 @@ class UnixFileAttributes return attrs; } + // get the UnixFileAttributes for a given file. Returns null if the file does not exist. + static UnixFileAttributes getIfExists(UnixPath path) throws UnixException { + UnixFileAttributes attrs = new UnixFileAttributes(); + int errno = UnixNativeDispatcher.stat2(path, attrs); + if (errno == 0) { + return attrs; + } else if (errno == UnixConstants.ENOENT) { + return null; + } else { + throw new UnixException(errno); + } + } + // get the UnixFileAttributes for an open file static UnixFileAttributes get(int fd) throws UnixException { UnixFileAttributes attrs = new UnixFileAttributes(); @@ -251,16 +264,6 @@ class UnixFileAttributes return UnixAsBasicFileAttributes.wrap(this); } - // unwrap BasicFileAttributes to get the underlying UnixFileAttributes - // object. Returns null is not wrapped. - static UnixFileAttributes toUnixFileAttributes(BasicFileAttributes attrs) { - if (attrs instanceof UnixFileAttributes) - return (UnixFileAttributes)attrs; - if (attrs instanceof UnixAsBasicFileAttributes) { - return ((UnixAsBasicFileAttributes)attrs).unwrap(); - } - return null; - } // wrap a UnixFileAttributes object as a BasicFileAttributes private static class UnixAsBasicFileAttributes implements BasicFileAttributes { @@ -274,9 +277,6 @@ class UnixFileAttributes return new UnixAsBasicFileAttributes(attrs); } - UnixFileAttributes unwrap() { - return attrs; - } @Override public FileTime lastModifiedTime() { diff --git a/src/java.base/unix/classes/sun/nio/fs/UnixFileSystemProvider.java b/src/java.base/unix/classes/sun/nio/fs/UnixFileSystemProvider.java index 99703d94755..f878938821a 100644 --- a/src/java.base/unix/classes/sun/nio/fs/UnixFileSystemProvider.java +++ b/src/java.base/unix/classes/sun/nio/fs/UnixFileSystemProvider.java @@ -126,7 +126,7 @@ public abstract class UnixFileSystemProvider return (V) UnixFileAttributeViews.createOwnerView(file, followLinks); if (type == null) throw new NullPointerException(); - return (V) null; + return null; } @Override @@ -148,6 +148,25 @@ public abstract class UnixFileSystemProvider return (A) getFileAttributeView(file, view, options).readAttributes(); } + @Override + public A readAttributesIfExists(Path path, + Class type, + LinkOption... options) + throws IOException + { + if (type == BasicFileAttributes.class && Util.followLinks(options)) { + UnixPath file = UnixPath.toUnixPath(path); + try { + @SuppressWarnings("unchecked") + A attrs = (A) UnixFileAttributes.getIfExists(file); + return attrs; + } catch (UnixException e) { + e.rethrowAsIOException(file); + } + } + return super.readAttributesIfExists(path, type, options); + } + @Override protected DynamicFileAttributeView getFileAttributeView(Path obj, String name, @@ -281,10 +300,10 @@ public abstract class UnixFileSystemProvider } else { for (AccessMode mode: modes) { switch (mode) { - case READ : r = true; break; - case WRITE : w = true; break; - case EXECUTE : x = true; break; - default: throw new AssertionError("Should not get here"); + case READ -> r = true; + case WRITE -> w = true; + case EXECUTE -> x = true; + default -> throw new AssertionError("Should not get here"); } } } @@ -321,9 +340,8 @@ public abstract class UnixFileSystemProvider return true; if (obj2 == null) throw new NullPointerException(); - if (!(obj2 instanceof UnixPath)) + if (!(obj2 instanceof UnixPath file2)) return false; - UnixPath file2 = (UnixPath)obj2; // check security manager access to both files file1.checkRead(); @@ -516,28 +534,16 @@ public abstract class UnixFileSystemProvider } @Override - public final boolean isDirectory(Path obj) { - UnixPath file = UnixPath.toUnixPath(obj); - file.checkRead(); - int mode = UnixNativeDispatcher.stat(file); - return ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR); - } - - @Override - public final boolean isRegularFile(Path obj) { - UnixPath file = UnixPath.toUnixPath(obj); - file.checkRead(); - int mode = UnixNativeDispatcher.stat(file); - return ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFREG); - } + public boolean exists(Path path, LinkOption... options) { + if (Util.followLinks(options)) { + UnixPath file = UnixPath.toUnixPath(path); + file.checkRead(); + return UnixNativeDispatcher.exists(file); + } else { + return super.exists(path, options); + } - @Override - public final boolean exists(Path obj) { - UnixPath file = UnixPath.toUnixPath(obj); - file.checkRead(); - return UnixNativeDispatcher.exists(file); } - /** * Returns a {@code FileTypeDetector} for this platform. */ diff --git a/src/java.base/unix/classes/sun/nio/fs/UnixNativeDispatcher.java b/src/java.base/unix/classes/sun/nio/fs/UnixNativeDispatcher.java index 4f4afaedcd6..b3bc08e76aa 100644 --- a/src/java.base/unix/classes/sun/nio/fs/UnixNativeDispatcher.java +++ b/src/java.base/unix/classes/sun/nio/fs/UnixNativeDispatcher.java @@ -318,33 +318,28 @@ class UnixNativeDispatcher { try (NativeBuffer buffer = copyToNativeBuffer(path)) { long comp = Blocker.begin(); try { - stat0(buffer.address(), attrs); + int errno = stat0(buffer.address(), attrs); + if (errno != 0) { + throw new UnixException(errno); + } } finally { Blocker.end(comp); } } } - private static native void stat0(long pathAddress, UnixFileAttributes attrs) - throws UnixException; - - /** - * stat(const char* path, struct stat* buf) - * - * @return st_mode (file type and mode) or 0 if an error occurs. - */ - static int stat(UnixPath path) { + static int stat2(UnixPath path, UnixFileAttributes attrs) { try (NativeBuffer buffer = copyToNativeBuffer(path)) { long comp = Blocker.begin(); try { - return stat1(buffer.address()); + return stat0(buffer.address(), attrs); } finally { Blocker.end(comp); } } } - private static native int stat1(long pathAddress); + private static native int stat0(long pathAddress, UnixFileAttributes attrs); /** * lstat(const char* path, struct stat* buf) diff --git a/src/java.base/unix/classes/sun/nio/fs/UnixUriUtils.java b/src/java.base/unix/classes/sun/nio/fs/UnixUriUtils.java index 133e123b636..11684a93bcb 100644 --- a/src/java.base/unix/classes/sun/nio/fs/UnixUriUtils.java +++ b/src/java.base/unix/classes/sun/nio/fs/UnixUriUtils.java @@ -118,10 +118,11 @@ class UnixUriUtils { if (sb.charAt(sb.length()-1) != '/') { try { up.checkRead(); - int mode = UnixNativeDispatcher.stat(up); - if ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR) + UnixFileAttributes attrs = UnixFileAttributes.getIfExists(up); + if (attrs != null + && ((attrs.mode() & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR)) sb.append('/'); - } catch (SecurityException ignore) { } + } catch (UnixException | SecurityException ignore) { } } try { diff --git a/src/java.base/unix/native/libnio/fs/UnixNativeDispatcher.c b/src/java.base/unix/native/libnio/fs/UnixNativeDispatcher.c index f115b7b056f..9371bc00181 100644 --- a/src/java.base/unix/native/libnio/fs/UnixNativeDispatcher.c +++ b/src/java.base/unix/native/libnio/fs/UnixNativeDispatcher.c @@ -521,7 +521,7 @@ static void prepAttributes(JNIEnv* env, struct stat64* buf, jobject attrs) { #endif } -JNIEXPORT void JNICALL +JNIEXPORT jint JNICALL Java_sun_nio_fs_UnixNativeDispatcher_stat0(JNIEnv* env, jclass this, jlong pathAddress, jobject attrs) { @@ -530,24 +530,11 @@ Java_sun_nio_fs_UnixNativeDispatcher_stat0(JNIEnv* env, jclass this, const char* path = (const char*)jlong_to_ptr(pathAddress); RESTARTABLE(stat64(path, &buf), err); - if (err == -1) { - throwUnixException(env, errno); - } else { + if (err == 0) { prepAttributes(env, &buf, attrs); - } -} - -JNIEXPORT jint JNICALL -Java_sun_nio_fs_UnixNativeDispatcher_stat1(JNIEnv* env, jclass this, jlong pathAddress) { - int err; - struct stat64 buf; - const char* path = (const char*)jlong_to_ptr(pathAddress); - - RESTARTABLE(stat64(path, &buf), err); - if (err == -1) { return 0; } else { - return (jint)buf.st_mode; + return errno; } } diff --git a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystemProvider.java b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystemProvider.java index d95f2944fae..4c861c48f1b 100644 --- a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystemProvider.java +++ b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystemProvider.java @@ -191,6 +191,15 @@ public class ZipFileSystemProvider extends FileSystemProvider { toZipPath(path).delete(); } + @Override + public boolean exists(Path path, LinkOption... options) { + if (options.length == 0) { + return toZipPath(path).exists(); + } else { + return super.exists(path, options); + } + } + @Override public V getFileAttributeView(Path path, Class type, LinkOption... options) @@ -284,6 +293,15 @@ public class ZipFileSystemProvider extends FileSystemProvider { return toZipPath(path).readAttributes(attributes, options); } + @Override + public A readAttributesIfExists(Path path, + Class type, + LinkOption... options) + throws IOException + { + return exists(path) ? readAttributes(path, type, options) : null; + } + @Override public Path readSymbolicLink(Path link) { throw new UnsupportedOperationException("Not supported."); diff --git a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipPath.java b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipPath.java index 7f4257a89fc..3f2ac98ff4e 100644 --- a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipPath.java +++ b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipPath.java @@ -329,9 +329,8 @@ final class ZipPath implements Path { @Override public boolean startsWith(Path other) { Objects.requireNonNull(other, "other"); - if (!(other instanceof ZipPath)) + if (!(other instanceof final ZipPath o)) return false; - final ZipPath o = (ZipPath)other; if (o.isAbsolute() != this.isAbsolute() || o.path.length > this.path.length) return false; @@ -349,9 +348,8 @@ final class ZipPath implements Path { @Override public boolean endsWith(Path other) { Objects.requireNonNull(other, "other"); - if (!(other instanceof ZipPath)) + if (!(other instanceof final ZipPath o)) return false; - final ZipPath o = (ZipPath)other; int olast = o.path.length - 1; if (olast > 0 && o.path[olast] == '/') olast--; @@ -382,17 +380,17 @@ final class ZipPath implements Path { } @Override - public final Path resolveSibling(String other) { + public Path resolveSibling(String other) { return resolveSibling(zfs.getPath(other)); } @Override - public final boolean startsWith(String other) { + public boolean startsWith(String other) { return startsWith(zfs.getPath(other)); } @Override - public final boolean endsWith(String other) { + public boolean endsWith(String other) { return endsWith(zfs.getPath(other)); } @@ -671,7 +669,7 @@ final class ZipPath implements Path { } @Override - public final File toFile() { + public File toFile() { throw new UnsupportedOperationException(); } @@ -901,7 +899,7 @@ final class ZipPath implements Path { } } - private boolean exists() { + boolean exists() { return zfs.exists(getResolvedPath()); }