Skip to content

Commit d48694d

Browse files
author
Lance Andersen
committedJul 5, 2022
8283335: Add exists and readAttributesIfExists methods to FileSystemProvider
Reviewed-by: alanb
1 parent c45d613 commit d48694d

File tree

13 files changed

+577
-175
lines changed

13 files changed

+577
-175
lines changed
 

‎src/java.base/share/classes/java/nio/file/Files.java

+9-38
Original file line numberDiff line numberDiff line change
@@ -80,7 +80,6 @@
8080
import jdk.internal.util.ArraysSupport;
8181
import sun.nio.ch.FileChannelImpl;
8282
import sun.nio.cs.UTF_8;
83-
import sun.nio.fs.AbstractFileSystemProvider;
8483

8584
/**
8685
* This class consists exclusively of static methods that operate on files,
@@ -1594,7 +1593,7 @@ public static long mismatch(Path path, Path path2) throws IOException {
15941593
byte[] buffer1 = new byte[BUFFER_SIZE];
15951594
byte[] buffer2 = new byte[BUFFER_SIZE];
15961595
try (InputStream in1 = Files.newInputStream(path);
1597-
InputStream in2 = Files.newInputStream(path2);) {
1596+
InputStream in2 = Files.newInputStream(path2)) {
15981597
long totalRead = 0;
15991598
while (true) {
16001599
int nRead1 = in1.readNBytes(buffer1, 0, BUFFER_SIZE);
@@ -2310,14 +2309,10 @@ public static boolean isSymbolicLink(Path path) {
23102309
* method denies read access to the file.
23112310
*/
23122311
public static boolean isDirectory(Path path, LinkOption... options) {
2313-
if (options.length == 0) {
2314-
FileSystemProvider provider = provider(path);
2315-
if (provider instanceof AbstractFileSystemProvider)
2316-
return ((AbstractFileSystemProvider)provider).isDirectory(path);
2317-
}
2318-
23192312
try {
2320-
return readAttributes(path, BasicFileAttributes.class, options).isDirectory();
2313+
var attrs = provider(path)
2314+
.readAttributesIfExists(path, BasicFileAttributes.class, options);
2315+
return (attrs != null) && attrs.isDirectory();
23212316
} catch (IOException ioe) {
23222317
return false;
23232318
}
@@ -2353,14 +2348,10 @@ public static boolean isDirectory(Path path, LinkOption... options) {
23532348
* method denies read access to the file.
23542349
*/
23552350
public static boolean isRegularFile(Path path, LinkOption... options) {
2356-
if (options.length == 0) {
2357-
FileSystemProvider provider = provider(path);
2358-
if (provider instanceof AbstractFileSystemProvider)
2359-
return ((AbstractFileSystemProvider)provider).isRegularFile(path);
2360-
}
2361-
23622351
try {
2363-
return readAttributes(path, BasicFileAttributes.class, options).isRegularFile();
2352+
var attrs = provider(path)
2353+
.readAttributesIfExists(path, BasicFileAttributes.class, options);
2354+
return (attrs != null) && attrs.isRegularFile();
23642355
} catch (IOException ioe) {
23652356
return false;
23662357
}
@@ -2502,7 +2493,7 @@ private static boolean followLinks(LinkOption... options) {
25022493
* the path to the file to test
25032494
* @param options
25042495
* options indicating how symbolic links are handled
2505-
* .
2496+
*
25062497
* @return {@code true} if the file exists; {@code false} if the file does
25072498
* not exist or its existence cannot be determined.
25082499
*
@@ -2515,27 +2506,7 @@ private static boolean followLinks(LinkOption... options) {
25152506
* @see FileSystemProvider#checkAccess
25162507
*/
25172508
public static boolean exists(Path path, LinkOption... options) {
2518-
if (options.length == 0) {
2519-
FileSystemProvider provider = provider(path);
2520-
if (provider instanceof AbstractFileSystemProvider)
2521-
return ((AbstractFileSystemProvider)provider).exists(path);
2522-
}
2523-
2524-
try {
2525-
if (followLinks(options)) {
2526-
provider(path).checkAccess(path);
2527-
} else {
2528-
// attempt to read attributes without following links
2529-
readAttributes(path, BasicFileAttributes.class,
2530-
LinkOption.NOFOLLOW_LINKS);
2531-
}
2532-
// file exists
2533-
return true;
2534-
} catch (IOException x) {
2535-
// does not exist or unable to determine if file exists
2536-
return false;
2537-
}
2538-
2509+
return provider(path).exists(path, options);
25392510
}
25402511

25412512
/**

‎src/java.base/share/classes/java/nio/file/spi/FileSystemProvider.java

+119-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2007, 2021, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2007, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -1170,4 +1170,122 @@ public abstract Map<String,Object> readAttributes(Path path, String attributes,
11701170
public abstract void setAttribute(Path path, String attribute,
11711171
Object value, LinkOption... options)
11721172
throws IOException;
1173+
1174+
/**
1175+
* Tests whether a file exists. This method works in exactly the
1176+
* manner specified by the {@link Files#exists(Path, LinkOption...)} method.
1177+
*
1178+
* @implSpec
1179+
* The default implementation of this method invokes the
1180+
* {@link #checkAccess(Path, AccessMode...)} method when symbolic links
1181+
* are followed. If the option {@link LinkOption#NOFOLLOW_LINKS NOFOLLOW_LINKS}
1182+
* is present then symbolic links are not followed and the method
1183+
* {@link #readAttributes(Path, Class, LinkOption...)} is called
1184+
* to determine whether a file exists.
1185+
*
1186+
* @param path
1187+
* the path to the file to test
1188+
* @param options
1189+
* options indicating how symbolic links are handled
1190+
*
1191+
* @return {@code true} if the file exists; {@code false} if the file does
1192+
* not exist or its existence cannot be determined.
1193+
*
1194+
* @throws SecurityException
1195+
* In the case of the default provider, the {@link
1196+
* SecurityManager#checkRead(String)} is invoked to check
1197+
* read access to the file.
1198+
*
1199+
* @since 20
1200+
*/
1201+
public boolean exists(Path path, LinkOption... options) {
1202+
try {
1203+
if (followLinks(options)) {
1204+
this.checkAccess(path);
1205+
} else {
1206+
// attempt to read attributes without following links
1207+
readAttributes(path, BasicFileAttributes.class, LinkOption.NOFOLLOW_LINKS);
1208+
}
1209+
// file exists
1210+
return true;
1211+
} catch (IOException x) {
1212+
// does not exist or unable to determine if file exists
1213+
return false;
1214+
}
1215+
}
1216+
1217+
/**
1218+
* Reads a file's attributes as a bulk operation if it exists.
1219+
*
1220+
* <p> The {@code type} parameter is the type of the attributes required
1221+
* and this method returns an instance of that type if supported. All
1222+
* implementations support a basic set of file attributes and so invoking
1223+
* this method with a {@code type} parameter of {@code
1224+
* BasicFileAttributes.class} will not throw {@code
1225+
* UnsupportedOperationException}.
1226+
*
1227+
* <p> The {@code options} array may be used to indicate how symbolic links
1228+
* are handled for the case that the file is a symbolic link. By default,
1229+
* symbolic links are followed and the file attribute of the final target
1230+
* of the link is read. If the option {@link LinkOption#NOFOLLOW_LINKS
1231+
* NOFOLLOW_LINKS} is present then symbolic links are not followed.
1232+
*
1233+
* <p> It is implementation specific if all file attributes are read as an
1234+
* atomic operation with respect to other file system operations.
1235+
*
1236+
* @implSpec
1237+
* The default implementation of this method invokes the
1238+
* {@link #readAttributes(Path, Class, LinkOption...)} method
1239+
* to read the file's attributes.
1240+
*
1241+
* @param <A>
1242+
* The {@code BasicFileAttributes} type
1243+
* @param path
1244+
* the path to the file
1245+
* @param type
1246+
* the {@code Class} of the file attributes required
1247+
* to read
1248+
* @param options
1249+
* options indicating how symbolic links are handled
1250+
*
1251+
* @return the file attributes or null if the file does not exist
1252+
*
1253+
* @throws UnsupportedOperationException
1254+
* if an attributes of the given type are not supported
1255+
* @throws IOException
1256+
* if an I/O error occurs
1257+
* @throws SecurityException
1258+
* In the case of the default provider, a security manager is
1259+
* installed, its {@link SecurityManager#checkRead(String) checkRead}
1260+
* method is invoked to check read access to the file. If this
1261+
* method is invoked to read security sensitive attributes then the
1262+
* security manager may be invoked to check for additional permissions.
1263+
*
1264+
* @since 20
1265+
*/
1266+
public <A extends BasicFileAttributes> A readAttributesIfExists(Path path,
1267+
Class<A> type,
1268+
LinkOption... options)
1269+
throws IOException
1270+
{
1271+
try {
1272+
return readAttributes(path, type, options);
1273+
} catch (NoSuchFileException ignore) {
1274+
return null;
1275+
}
1276+
}
1277+
1278+
private static boolean followLinks(LinkOption... options) {
1279+
boolean followLinks = true;
1280+
for (LinkOption opt: options) {
1281+
if (opt == LinkOption.NOFOLLOW_LINKS) {
1282+
followLinks = false;
1283+
continue;
1284+
}
1285+
if (opt == null)
1286+
throw new NullPointerException();
1287+
throw new AssertionError("Should not get here");
1288+
}
1289+
return followLinks;
1290+
}
11731291
}

‎src/java.base/share/classes/sun/nio/fs/AbstractFileSystemProvider.java

+1-47
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2011, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2011, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -27,7 +27,6 @@
2727

2828
import java.nio.file.Path;
2929
import java.nio.file.LinkOption;
30-
import java.nio.file.attribute.BasicFileAttributes;
3130
import java.nio.file.spi.FileSystemProvider;
3231
import java.io.IOException;
3332
import java.util.Map;
@@ -110,51 +109,6 @@ public final boolean deleteIfExists(Path file) throws IOException {
110109
return implDelete(file, false);
111110
}
112111

113-
/**
114-
* Tests whether a file is a directory.
115-
*
116-
* @return {@code true} if the file is a directory; {@code false} if
117-
* the file does not exist, is not a directory, or it cannot
118-
* be determined if the file is a directory or not.
119-
*/
120-
public boolean isDirectory(Path file) {
121-
try {
122-
return readAttributes(file, BasicFileAttributes.class).isDirectory();
123-
} catch (IOException ioe) {
124-
return false;
125-
}
126-
}
127-
128-
/**
129-
* Tests whether a file is a regular file with opaque content.
130-
*
131-
* @return {@code true} if the file is a regular file; {@code false} if
132-
* the file does not exist, is not a regular file, or it
133-
* cannot be determined if the file is a regular file or not.
134-
*/
135-
public boolean isRegularFile(Path file) {
136-
try {
137-
return readAttributes(file, BasicFileAttributes.class).isRegularFile();
138-
} catch (IOException ioe) {
139-
return false;
140-
}
141-
}
142-
143-
/**
144-
* Checks the existence of a file.
145-
*
146-
* @return {@code true} if the file exists; {@code false} if the file does
147-
* not exist or its existence cannot be determined.
148-
*/
149-
public boolean exists(Path file) {
150-
try {
151-
checkAccess(file);
152-
return true;
153-
} catch (IOException ioe) {
154-
return false;
155-
}
156-
}
157-
158112
/**
159113
* Returns a path name as bytes for a Unix domain socket.
160114
* Different encodings may be used for these names on some platforms.

‎src/java.base/unix/classes/sun/nio/fs/UnixFileAttributes.java

+14-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2008, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2008, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -74,6 +74,19 @@ static UnixFileAttributes get(UnixPath path, boolean followLinks)
7474
return attrs;
7575
}
7676

77+
// get the UnixFileAttributes for a given file. Returns null if the file does not exist.
78+
static UnixFileAttributes getIfExists(UnixPath path) throws UnixException {
79+
UnixFileAttributes attrs = new UnixFileAttributes();
80+
int errno = UnixNativeDispatcher.stat2(path, attrs);
81+
if (errno == 0) {
82+
return attrs;
83+
} else if (errno == UnixConstants.ENOENT) {
84+
return null;
85+
} else {
86+
throw new UnixException(errno);
87+
}
88+
}
89+
7790
// get the UnixFileAttributes for an open file
7891
static UnixFileAttributes get(int fd) throws UnixException {
7992
UnixFileAttributes attrs = new UnixFileAttributes();
@@ -251,16 +264,6 @@ BasicFileAttributes asBasicFileAttributes() {
251264
return UnixAsBasicFileAttributes.wrap(this);
252265
}
253266

254-
// unwrap BasicFileAttributes to get the underlying UnixFileAttributes
255-
// object. Returns null is not wrapped.
256-
static UnixFileAttributes toUnixFileAttributes(BasicFileAttributes attrs) {
257-
if (attrs instanceof UnixFileAttributes)
258-
return (UnixFileAttributes)attrs;
259-
if (attrs instanceof UnixAsBasicFileAttributes) {
260-
return ((UnixAsBasicFileAttributes)attrs).unwrap();
261-
}
262-
return null;
263-
}
264267

265268
// wrap a UnixFileAttributes object as a BasicFileAttributes
266269
private static class UnixAsBasicFileAttributes implements BasicFileAttributes {
@@ -274,9 +277,6 @@ static UnixAsBasicFileAttributes wrap(UnixFileAttributes attrs) {
274277
return new UnixAsBasicFileAttributes(attrs);
275278
}
276279

277-
UnixFileAttributes unwrap() {
278-
return attrs;
279-
}
280280

281281
@Override
282282
public FileTime lastModifiedTime() {

‎src/java.base/unix/classes/sun/nio/fs/UnixFileSystemProvider.java

+32-27
Original file line numberDiff line numberDiff line change
@@ -126,7 +126,7 @@ public <V extends FileAttributeView> V getFileAttributeView(Path obj,
126126
return (V) UnixFileAttributeViews.createOwnerView(file, followLinks);
127127
if (type == null)
128128
throw new NullPointerException();
129-
return (V) null;
129+
return null;
130130
}
131131

132132
@Override
@@ -148,6 +148,25 @@ else if (type == null)
148148
return (A) getFileAttributeView(file, view, options).readAttributes();
149149
}
150150

151+
@Override
152+
public <A extends BasicFileAttributes> A readAttributesIfExists(Path path,
153+
Class<A> type,
154+
LinkOption... options)
155+
throws IOException
156+
{
157+
if (type == BasicFileAttributes.class && Util.followLinks(options)) {
158+
UnixPath file = UnixPath.toUnixPath(path);
159+
try {
160+
@SuppressWarnings("unchecked")
161+
A attrs = (A) UnixFileAttributes.getIfExists(file);
162+
return attrs;
163+
} catch (UnixException e) {
164+
e.rethrowAsIOException(file);
165+
}
166+
}
167+
return super.readAttributesIfExists(path, type, options);
168+
}
169+
151170
@Override
152171
protected DynamicFileAttributeView getFileAttributeView(Path obj,
153172
String name,
@@ -281,10 +300,9 @@ public void checkAccess(Path obj, AccessMode... modes) throws IOException {
281300
} else {
282301
for (AccessMode mode: modes) {
283302
switch (mode) {
284-
case READ : r = true; break;
285-
case WRITE : w = true; break;
286-
case EXECUTE : x = true; break;
287-
default: throw new AssertionError("Should not get here");
303+
case READ -> r = true;
304+
case WRITE -> w = true;
305+
case EXECUTE -> x = true;
288306
}
289307
}
290308
}
@@ -321,9 +339,8 @@ public boolean isSameFile(Path obj1, Path obj2) throws IOException {
321339
return true;
322340
if (obj2 == null)
323341
throw new NullPointerException();
324-
if (!(obj2 instanceof UnixPath))
342+
if (!(obj2 instanceof UnixPath file2))
325343
return false;
326-
UnixPath file2 = (UnixPath)obj2;
327344

328345
// check security manager access to both files
329346
file1.checkRead();
@@ -516,28 +533,16 @@ public Path readSymbolicLink(Path obj1) throws IOException {
516533
}
517534

518535
@Override
519-
public final boolean isDirectory(Path obj) {
520-
UnixPath file = UnixPath.toUnixPath(obj);
521-
file.checkRead();
522-
int mode = UnixNativeDispatcher.stat(file);
523-
return ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR);
524-
}
525-
526-
@Override
527-
public final boolean isRegularFile(Path obj) {
528-
UnixPath file = UnixPath.toUnixPath(obj);
529-
file.checkRead();
530-
int mode = UnixNativeDispatcher.stat(file);
531-
return ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFREG);
532-
}
536+
public boolean exists(Path path, LinkOption... options) {
537+
if (Util.followLinks(options)) {
538+
UnixPath file = UnixPath.toUnixPath(path);
539+
file.checkRead();
540+
return UnixNativeDispatcher.exists(file);
541+
} else {
542+
return super.exists(path, options);
543+
}
533544

534-
@Override
535-
public final boolean exists(Path obj) {
536-
UnixPath file = UnixPath.toUnixPath(obj);
537-
file.checkRead();
538-
return UnixNativeDispatcher.exists(file);
539545
}
540-
541546
/**
542547
* Returns a {@code FileTypeDetector} for this platform.
543548
*/

‎src/java.base/unix/classes/sun/nio/fs/UnixNativeDispatcher.java

+7-12
Original file line numberDiff line numberDiff line change
@@ -318,33 +318,28 @@ static void stat(UnixPath path, UnixFileAttributes attrs) throws UnixException {
318318
try (NativeBuffer buffer = copyToNativeBuffer(path)) {
319319
long comp = Blocker.begin();
320320
try {
321-
stat0(buffer.address(), attrs);
321+
int errno = stat0(buffer.address(), attrs);
322+
if (errno != 0) {
323+
throw new UnixException(errno);
324+
}
322325
} finally {
323326
Blocker.end(comp);
324327
}
325328
}
326329
}
327-
private static native void stat0(long pathAddress, UnixFileAttributes attrs)
328-
throws UnixException;
329-
330330

331-
/**
332-
* stat(const char* path, struct stat* buf)
333-
*
334-
* @return st_mode (file type and mode) or 0 if an error occurs.
335-
*/
336-
static int stat(UnixPath path) {
331+
static int stat2(UnixPath path, UnixFileAttributes attrs) {
337332
try (NativeBuffer buffer = copyToNativeBuffer(path)) {
338333
long comp = Blocker.begin();
339334
try {
340-
return stat1(buffer.address());
335+
return stat0(buffer.address(), attrs);
341336
} finally {
342337
Blocker.end(comp);
343338
}
344339
}
345340
}
346-
private static native int stat1(long pathAddress);
347341

342+
private static native int stat0(long pathAddress, UnixFileAttributes attrs);
348343

349344
/**
350345
* lstat(const char* path, struct stat* buf)

‎src/java.base/unix/classes/sun/nio/fs/UnixUriUtils.java

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2008, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2008, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -118,10 +118,11 @@ static URI toUri(UnixPath up) {
118118
if (sb.charAt(sb.length()-1) != '/') {
119119
try {
120120
up.checkRead();
121-
int mode = UnixNativeDispatcher.stat(up);
122-
if ((mode & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR)
121+
UnixFileAttributes attrs = UnixFileAttributes.getIfExists(up);
122+
if (attrs != null
123+
&& ((attrs.mode() & UnixConstants.S_IFMT) == UnixConstants.S_IFDIR))
123124
sb.append('/');
124-
} catch (SecurityException ignore) { }
125+
} catch (UnixException | SecurityException ignore) { }
125126
}
126127

127128
try {

‎src/java.base/unix/native/libnio/fs/UnixNativeDispatcher.c

+3-16
Original file line numberDiff line numberDiff line change
@@ -521,7 +521,7 @@ static void prepAttributes(JNIEnv* env, struct stat64* buf, jobject attrs) {
521521
#endif
522522
}
523523

524-
JNIEXPORT void JNICALL
524+
JNIEXPORT jint JNICALL
525525
Java_sun_nio_fs_UnixNativeDispatcher_stat0(JNIEnv* env, jclass this,
526526
jlong pathAddress, jobject attrs)
527527
{
@@ -530,24 +530,11 @@ Java_sun_nio_fs_UnixNativeDispatcher_stat0(JNIEnv* env, jclass this,
530530
const char* path = (const char*)jlong_to_ptr(pathAddress);
531531

532532
RESTARTABLE(stat64(path, &buf), err);
533-
if (err == -1) {
534-
throwUnixException(env, errno);
535-
} else {
533+
if (err == 0) {
536534
prepAttributes(env, &buf, attrs);
537-
}
538-
}
539-
540-
JNIEXPORT jint JNICALL
541-
Java_sun_nio_fs_UnixNativeDispatcher_stat1(JNIEnv* env, jclass this, jlong pathAddress) {
542-
int err;
543-
struct stat64 buf;
544-
const char* path = (const char*)jlong_to_ptr(pathAddress);
545-
546-
RESTARTABLE(stat64(path, &buf), err);
547-
if (err == -1) {
548535
return 0;
549536
} else {
550-
return (jint)buf.st_mode;
537+
return errno;
551538
}
552539
}
553540

‎src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystemProvider.java

+19
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,15 @@ public final void delete(Path path) throws IOException {
191191
toZipPath(path).delete();
192192
}
193193

194+
@Override
195+
public boolean exists(Path path, LinkOption... options) {
196+
if (options.length == 0) {
197+
return toZipPath(path).exists();
198+
} else {
199+
return super.exists(path, options);
200+
}
201+
}
202+
194203
@Override
195204
public <V extends FileAttributeView> V
196205
getFileAttributeView(Path path, Class<V> type, LinkOption... options)
@@ -284,6 +293,16 @@ public OutputStream newOutputStream(Path path, OpenOption... options)
284293
return toZipPath(path).readAttributes(attributes, options);
285294
}
286295

296+
@Override
297+
@SuppressWarnings("unchecked") // Cast to A
298+
public <A extends BasicFileAttributes> A readAttributesIfExists(Path path,
299+
Class<A> type,
300+
LinkOption... options)
301+
throws IOException
302+
{
303+
return (A) toZipPath(path).readAttributesIfExists();
304+
}
305+
287306
@Override
288307
public Path readSymbolicLink(Path link) {
289308
throw new UnsupportedOperationException("Not supported.");

‎src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipPath.java

+11-9
Original file line numberDiff line numberDiff line change
@@ -329,9 +329,8 @@ public Path resolveSibling(Path other) {
329329
@Override
330330
public boolean startsWith(Path other) {
331331
Objects.requireNonNull(other, "other");
332-
if (!(other instanceof ZipPath))
332+
if (!(other instanceof final ZipPath o))
333333
return false;
334-
final ZipPath o = (ZipPath)other;
335334
if (o.isAbsolute() != this.isAbsolute() ||
336335
o.path.length > this.path.length)
337336
return false;
@@ -349,9 +348,8 @@ public boolean startsWith(Path other) {
349348
@Override
350349
public boolean endsWith(Path other) {
351350
Objects.requireNonNull(other, "other");
352-
if (!(other instanceof ZipPath))
351+
if (!(other instanceof final ZipPath o))
353352
return false;
354-
final ZipPath o = (ZipPath)other;
355353
int olast = o.path.length - 1;
356354
if (olast > 0 && o.path[olast] == '/')
357355
olast--;
@@ -382,17 +380,17 @@ public ZipPath resolve(String other) {
382380
}
383381

384382
@Override
385-
public final Path resolveSibling(String other) {
383+
public Path resolveSibling(String other) {
386384
return resolveSibling(zfs.getPath(other));
387385
}
388386

389387
@Override
390-
public final boolean startsWith(String other) {
388+
public boolean startsWith(String other) {
391389
return startsWith(zfs.getPath(other));
392390
}
393391

394392
@Override
395-
public final boolean endsWith(String other) {
393+
public boolean endsWith(String other) {
396394
return endsWith(zfs.getPath(other));
397395
}
398396

@@ -671,7 +669,7 @@ public WatchKey register(WatchService watcher, WatchEvent.Kind<?>... events) {
671669
}
672670

673671
@Override
674-
public final File toFile() {
672+
public File toFile() {
675673
throw new UnsupportedOperationException();
676674
}
677675

@@ -844,6 +842,10 @@ Map<String, Object> readAttributes(String attributes, LinkOption... options)
844842
return getFileAttributeView(view).readAttributes(attrs);
845843
}
846844

845+
ZipFileAttributes readAttributesIfExists() throws IOException {
846+
return zfs.getFileAttributes(getResolvedPath());
847+
}
848+
847849
FileStore getFileStore() throws IOException {
848850
// each ZipFileSystem only has one root (as requested for now)
849851
if (exists())
@@ -901,7 +903,7 @@ void checkAccess(AccessMode... modes) throws IOException {
901903
}
902904
}
903905

904-
private boolean exists() {
906+
boolean exists() {
905907
return zfs.exists(getResolvedPath());
906908
}
907909

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,220 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import org.testng.annotations.BeforeClass;
25+
import org.testng.annotations.BeforeMethod;
26+
import org.testng.annotations.DataProvider;
27+
import org.testng.annotations.Test;
28+
29+
import java.io.IOException;
30+
import java.net.URI;
31+
import java.nio.file.*;
32+
import java.nio.file.attribute.BasicFileAttributes;
33+
import java.util.ArrayList;
34+
import java.util.HashMap;
35+
import java.util.List;
36+
import java.util.Map;
37+
38+
import static org.testng.AssertJUnit.assertEquals;
39+
40+
/**
41+
* @test
42+
* @summary Verifies that a FileSystemProvider's implementation of the exists
43+
* and readAttributesIfExists methods are invoked
44+
* @build TestDelegation TestProvider
45+
* @run testng/othervm TestDelegation
46+
*/
47+
public class TestDelegation {
48+
49+
// Non-existent Path to be used by the test
50+
private Path nonExistentFile;
51+
// Path to Temp directory used by the test
52+
private Path tempDirectory;
53+
// Valid file Path used by the test
54+
private Path fileThatExists;
55+
// The FileSystemProvider used by the test
56+
private MyProvider myProvider;
57+
58+
59+
/**
60+
* Create the FileSystemProvider, the FileSystem and
61+
* Path's used by the test.
62+
*
63+
* @throws IOException if an error occurs
64+
*/
65+
@BeforeClass
66+
public void setup() throws IOException {
67+
myProvider = new MyProvider();
68+
FileSystem fs = myProvider.getFileSystem(URI.create("/"));
69+
// Path to Current Working Directory
70+
Path cwd = fs.getPath(".");
71+
tempDirectory = Files.createTempDirectory(cwd, "tmp");
72+
fileThatExists = Files.createFile(tempDirectory.resolve("file"));
73+
nonExistentFile = tempDirectory.resolve("doesNotExist");
74+
}
75+
76+
/**
77+
* DataProvider that is used to test Files::exists. The DataProvider's
78+
* elements are:
79+
* <UL>
80+
* <li>Path to validate</li>
81+
* <li>Does the Path Exist</li>
82+
* </UL>
83+
* @return The test parameter data
84+
*/
85+
@DataProvider
86+
private Object[][] testExists() {
87+
return new Object[][]{
88+
{tempDirectory, true},
89+
{fileThatExists, true},
90+
{nonExistentFile, false}
91+
};
92+
}
93+
94+
/**
95+
* DataProvider that is used to test Files::isDirectory. The DataProvider's
96+
* elements are:
97+
* <UL>
98+
* <li>Path to validate</li>
99+
* <li>Is the Path a Directory</li>
100+
* </UL>
101+
* @return The test parameter data
102+
*/
103+
@DataProvider
104+
private Object[][] testIsDirectory() {
105+
return new Object[][]{
106+
{tempDirectory, true},
107+
{fileThatExists, false},
108+
{nonExistentFile, false}
109+
};
110+
}
111+
/**
112+
* DataProvider that is used to test Files::isRegularFile. The DataProvider's
113+
* elements are:
114+
* <UL>
115+
* <li>Path to validate</li>
116+
* <li>Is the Path a regular file</li>
117+
* </UL>
118+
* @return The test parameter data
119+
*/
120+
@DataProvider
121+
private Object[][] testIsRegularFile() {
122+
return new Object[][]{
123+
{tempDirectory, false},
124+
{fileThatExists, true},
125+
{nonExistentFile, false}
126+
};
127+
}
128+
129+
/**
130+
* Clear our Map prior to each test run
131+
*/
132+
@BeforeMethod
133+
public void resetParams() {
134+
myProvider.resetCalls();
135+
}
136+
137+
/**
138+
* Validate that Files::exists delegates to the FileSystemProvider's
139+
* implementation of exists.
140+
*
141+
* @param p the path to the file to test
142+
* @param exists does the path exist
143+
*/
144+
@Test(dataProvider = "testExists")
145+
public void testExists(Path p, boolean exists) {
146+
assertEquals(Files.exists(p), exists);
147+
// We should only have called exists once
148+
assertEquals(1, myProvider.findCall("exists").size());
149+
assertEquals(0, myProvider.findCall("readAttributesIfExists").size());
150+
}
151+
152+
/**
153+
* Validate that Files::isDirectory delegates to the FileSystemProvider's
154+
* implementation readAttributesIfExists.
155+
*
156+
* @param p the path to the file to test
157+
* @param isDir is the path a directory
158+
*/
159+
@Test(dataProvider = "testIsDirectory")
160+
public void testIsDirectory(Path p, boolean isDir) {
161+
assertEquals(Files.isDirectory(p), isDir);
162+
// We should only have called readAttributesIfExists once
163+
assertEquals(0, myProvider.findCall("exists").size());
164+
assertEquals(1, myProvider.findCall("readAttributesIfExists").size());
165+
}
166+
167+
/**
168+
* Validate that Files::isRegularFile delegates to the FileSystemProvider's
169+
* implementation readAttributesIfExists.
170+
*
171+
* @param p the path to the file to test
172+
* @param isFile is the path a regular file
173+
*/
174+
@Test(dataProvider = "testIsRegularFile")
175+
public void testIsRegularFile(Path p, boolean isFile) {
176+
assertEquals(Files.isRegularFile(p), isFile);
177+
// We should only have called readAttributesIfExists once
178+
assertEquals(0, myProvider.findCall("exists").size());
179+
assertEquals(1, myProvider.findCall("readAttributesIfExists").size());
180+
}
181+
182+
/**
183+
* The FileSystemProvider implementation used by the test
184+
*/
185+
static class MyProvider extends TestProvider {
186+
private final Map<String, List<Path>> calls = new HashMap<>();
187+
188+
private MyProvider() {
189+
super(FileSystems.getDefault().provider());
190+
}
191+
192+
private void recordCall(String op, Path path) {
193+
calls.computeIfAbsent(op, k -> new ArrayList<>()).add(path);
194+
}
195+
196+
List<Path> findCall(String op) {
197+
return calls.getOrDefault(op, List.of());
198+
}
199+
200+
void resetCalls() {
201+
calls.clear();
202+
}
203+
204+
@Override
205+
public boolean exists(Path path, LinkOption... options) {
206+
recordCall("exists", path);
207+
return super.exists(path, options);
208+
}
209+
210+
@Override
211+
public <A extends BasicFileAttributes> A readAttributesIfExists(Path path,
212+
Class<A> type,
213+
LinkOption... options)
214+
throws IOException {
215+
recordCall("readAttributesIfExists", path);
216+
return super.readAttributesIfExists(path, type, options);
217+
}
218+
}
219+
}
220+

‎test/jdk/java/nio/file/spi/TestProvider.java

+4-7
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2008, 2011, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2008, 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -32,7 +32,6 @@
3232
import java.nio.channels.SeekableByteChannel;
3333
import java.net.URI;
3434
import java.io.IOException;
35-
import java.util.Collections;
3635
import java.util.Iterator;
3736
import java.util.Map;
3837
import java.util.Set;
@@ -48,10 +47,6 @@ public TestProvider(FileSystemProvider defaultProvider) {
4847
this.theFileSystem = new TestFileSystem(fs, this);
4948
}
5049

51-
FileSystemProvider defaultProvider() {
52-
return defaultProvider;
53-
}
54-
5550
@Override
5651
public String getScheme() {
5752
return "file";
@@ -203,7 +198,8 @@ public boolean isSameFile(Path file, Path other) throws IOException {
203198
public void checkAccess(Path file, AccessMode... modes)
204199
throws IOException
205200
{
206-
throw new RuntimeException("not implemented");
201+
Path delegate = theFileSystem.unwrap(file);
202+
defaultProvider.checkAccess(delegate, modes);
207203
}
208204

209205
static class TestFileSystem extends FileSystem {
@@ -469,4 +465,5 @@ public WatchKey register(WatchService watcher,
469465
throw new UnsupportedOperationException();
470466
}
471467
}
468+
472469
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,133 @@
1+
/*
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package org.openjdk.bench.jdk.nio.zipfs;
24+
25+
import org.openjdk.jmh.annotations.*;
26+
27+
import java.io.IOException;
28+
import java.net.URI;
29+
import java.nio.charset.StandardCharsets;
30+
import java.nio.file.*;
31+
import java.util.HashMap;
32+
import java.util.Map;
33+
import java.util.Random;
34+
import java.util.concurrent.TimeUnit;
35+
36+
/**
37+
* Simple benchmark measuring cost of Files::exist, Files::isDirectory and
38+
* Files::isRegularFile with ZipFileSystem.
39+
*/
40+
@BenchmarkMode(Mode.AverageTime)
41+
@OutputTimeUnit(TimeUnit.NANOSECONDS)
42+
@State(Scope.Thread)
43+
@Warmup(iterations = 10, time = 500, timeUnit = TimeUnit.MILLISECONDS)
44+
@Measurement(iterations = 5, time = 1000, timeUnit = TimeUnit.MILLISECONDS)
45+
@Fork(3)
46+
public class ZipfileSystemProviderDelegation {
47+
48+
public static final String ZIP_FILE = "zipfsprovider-delegation-benchmark.zip";
49+
public static final String NON_EXISTENT_SUFFIX = "-nope";
50+
@Param({"256", "512"})
51+
private int entriesToTest;
52+
public String[] entries;
53+
private FileSystem zipfs;
54+
55+
private int index = 0;
56+
57+
@Setup
58+
public void setup() throws IOException {
59+
Path zip = Paths.get(ZIP_FILE);
60+
Files.deleteIfExists(zip);
61+
Random random = new Random(4711);
62+
entries = new String[entriesToTest];
63+
URI zipURI = URI.create("jar:file:"+ zip.toAbsolutePath().toString());
64+
Map<String, String> env = new HashMap<>();
65+
env.put("create", "true");
66+
zipfs = FileSystems.newFileSystem(zipURI, env);
67+
for (int i = 0; i < entriesToTest; i++) {
68+
Path dir = zipfs.getPath("dir-" + (random.nextInt(90000) + 10000)
69+
+ "-" + i);
70+
Files.createDirectory(dir);
71+
Path entry = dir.resolve("entry-" +
72+
(random.nextInt(90000) + 10000)
73+
+ "-" + i);
74+
Files.write(entry, "".getBytes(StandardCharsets.UTF_8));
75+
entries[i] = entry.toString();
76+
}
77+
}
78+
79+
@TearDown
80+
public void cleanup() throws IOException {
81+
zipfs.close();
82+
Files.deleteIfExists(Paths.get(ZIP_FILE));
83+
}
84+
85+
@Benchmark
86+
public void existsWithEntry() {
87+
if (index >= entriesToTest) {
88+
index = 0;
89+
}
90+
Files.exists(zipfs.getPath(entries[index++]));
91+
}
92+
93+
@Benchmark
94+
public void existsWithNonExistingEntry() {
95+
if (index >= entriesToTest) {
96+
index = 0;
97+
}
98+
Files.exists(zipfs.getPath(entries[index++] + NON_EXISTENT_SUFFIX));
99+
}
100+
101+
@Benchmark
102+
public void isDirectoryWithEntry() {
103+
if (index >= entriesToTest) {
104+
index = 0;
105+
}
106+
Files.isDirectory(zipfs.getPath(entries[index++]));
107+
}
108+
109+
@Benchmark
110+
public void isDirectoryWithNonExistingEntry() {
111+
if (index >= entriesToTest) {
112+
index = 0;
113+
}
114+
Files.isDirectory(zipfs.getPath(entries[index++] + NON_EXISTENT_SUFFIX));
115+
}
116+
117+
@Benchmark
118+
public void isRegularFileWithEntry() {
119+
if (index >= entriesToTest) {
120+
index = 0;
121+
}
122+
Files.isRegularFile(zipfs.getPath(entries[index++]));
123+
}
124+
125+
@Benchmark
126+
public void isRegularFileWithNonExistingEntry() {
127+
if (index >= entriesToTest) {
128+
index = 0;
129+
}
130+
Files.isRegularFile(zipfs.getPath(entries[index++] + NON_EXISTENT_SUFFIX));
131+
}
132+
133+
}

0 commit comments

Comments
 (0)
Please sign in to comment.