From 177d71c3b78c50b963f90efe425dd60a7e3c66ae Mon Sep 17 00:00:00 2001 From: Sebastian Hartte Date: Fri, 12 Apr 2024 16:16:47 +0200 Subject: [PATCH 1/5] Add close() methods throughout the SecureJar->JarContents chain to allow closing the backing file systems. --- src/main/java/cpw/mods/jarhandling/JarContents.java | 9 ++++++++- src/main/java/cpw/mods/jarhandling/SecureJar.java | 9 ++++++++- src/main/java/cpw/mods/jarhandling/VirtualJar.java | 6 ++++++ src/main/java/cpw/mods/jarhandling/impl/Jar.java | 6 ++++++ .../java/cpw/mods/jarhandling/impl/JarContentsImpl.java | 5 +++++ src/main/java/cpw/mods/niofs/union/UnionFileSystem.java | 5 ++++- .../cpw/mods/jarhandling/impl/TestDummyJarProvider.java | 6 ++++-- 7 files changed, 41 insertions(+), 5 deletions(-) diff --git a/src/main/java/cpw/mods/jarhandling/JarContents.java b/src/main/java/cpw/mods/jarhandling/JarContents.java index a9bf818..bcbb675 100644 --- a/src/main/java/cpw/mods/jarhandling/JarContents.java +++ b/src/main/java/cpw/mods/jarhandling/JarContents.java @@ -2,6 +2,7 @@ import org.jetbrains.annotations.ApiStatus; +import java.io.IOException; import java.net.URI; import java.nio.file.Path; import java.util.Collection; @@ -18,7 +19,7 @@ * Convert to a full jar with {@link SecureJar#from(JarContents)}. */ @ApiStatus.NonExtendable -public interface JarContents { +public interface JarContents extends AutoCloseable { /** * @see SecureJar#getPrimaryPath() */ @@ -69,4 +70,10 @@ static JarContents of(Path fileOrFolder) { static JarContents of(Collection filesOrFolders) { return new JarContentsBuilder().paths(filesOrFolders.toArray(new Path[0])).build(); } + + /** + * Closes the underlying resources. + */ + @Override + void close() throws IOException; } diff --git a/src/main/java/cpw/mods/jarhandling/SecureJar.java b/src/main/java/cpw/mods/jarhandling/SecureJar.java index 75d443e..9d4e69e 100644 --- a/src/main/java/cpw/mods/jarhandling/SecureJar.java +++ b/src/main/java/cpw/mods/jarhandling/SecureJar.java @@ -28,7 +28,7 @@ * A secure jar is the full definition for a module, * including all its paths and code signing metadata. */ -public interface SecureJar { +public interface SecureJar extends AutoCloseable { /** * Creates a jar from a list of paths. * See {@link JarContentsBuilder} for more configuration options. @@ -86,6 +86,13 @@ static SecureJar from(JarContents contents, JarMetadata metadata) { */ Path getRootPath(); + /** + * Closes the underlying file system resources (if any). + * Renders this object unusable. + */ + @Override + void close() throws IOException; + /** * All the functions that are necessary to turn a {@link SecureJar} into a module. */ diff --git a/src/main/java/cpw/mods/jarhandling/VirtualJar.java b/src/main/java/cpw/mods/jarhandling/VirtualJar.java index f6fc84c..5b22482 100644 --- a/src/main/java/cpw/mods/jarhandling/VirtualJar.java +++ b/src/main/java/cpw/mods/jarhandling/VirtualJar.java @@ -4,6 +4,7 @@ import cpw.mods.niofs.union.UnionFileSystemProvider; import org.jetbrains.annotations.Nullable; +import java.io.IOException; import java.io.InputStream; import java.lang.module.ModuleDescriptor; import java.net.URI; @@ -106,6 +107,11 @@ public Path getRootPath() { return dummyFileSystem.getRoot(); } + @Override + public void close() throws IOException { + dummyFileSystem.close(); + } + private class VirtualJarModuleDataProvider implements ModuleDataProvider { @Override public String name() { diff --git a/src/main/java/cpw/mods/jarhandling/impl/Jar.java b/src/main/java/cpw/mods/jarhandling/impl/Jar.java index 7ec50eb..b438aa9 100644 --- a/src/main/java/cpw/mods/jarhandling/impl/Jar.java +++ b/src/main/java/cpw/mods/jarhandling/impl/Jar.java @@ -7,6 +7,7 @@ import cpw.mods.util.LambdaExceptionUtils; import org.jetbrains.annotations.Nullable; +import java.io.IOException; import java.io.InputStream; import java.lang.module.ModuleDescriptor; import java.net.URI; @@ -104,6 +105,11 @@ public Path getRootPath() { return filesystem.getPath(""); } + @Override + public void close() throws IOException { + contents.close(); + } + @Override public String toString() { return "Jar[" + getURI() + "]"; diff --git a/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java b/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java index 2100d12..c019e56 100644 --- a/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java +++ b/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java @@ -218,4 +218,9 @@ public List getMetaInfServices() { } return this.providers; } + + @Override + public void close() throws IOException { + filesystem.close(); + } } diff --git a/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java b/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java index a89da3e..03bd167 100644 --- a/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java +++ b/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java @@ -155,8 +155,11 @@ public UnionFileSystemProvider provider() { } @Override - public void close() { + public void close() throws IOException { provider().removeFileSystem(this); + for (var embeddedFs : embeddedFileSystems.values()) { + embeddedFs.fs.close(); + } } @Override diff --git a/src/test/java/cpw/mods/jarhandling/impl/TestDummyJarProvider.java b/src/test/java/cpw/mods/jarhandling/impl/TestDummyJarProvider.java index e67909b..ccba0c9 100644 --- a/src/test/java/cpw/mods/jarhandling/impl/TestDummyJarProvider.java +++ b/src/test/java/cpw/mods/jarhandling/impl/TestDummyJarProvider.java @@ -13,11 +13,9 @@ import java.lang.module.Configuration; import java.lang.module.ModuleDescriptor; import java.lang.module.ModuleFinder; -import java.lang.module.ResolvedModule; import java.net.URI; import java.nio.file.Path; import java.security.CodeSigner; -import java.util.Arrays; import java.util.List; import java.util.Optional; import java.util.Set; @@ -133,5 +131,9 @@ public Path getPath(final String first, final String... rest) { public Path getRootPath() { return null; } + + @Override + public void close() { + } } } From 797e232b4de40f2b07372ed438d1644e4c53ec74 Mon Sep 17 00:00:00 2001 From: Sebastian Hartte Date: Sun, 2 Jun 2024 17:55:42 +0200 Subject: [PATCH 2/5] Address closing a filesystem potentially throwing. --- .../java/cpw/mods/niofs/union/UnionFileSystem.java | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java b/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java index 03bd167..4db42d7 100644 --- a/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java +++ b/src/main/java/cpw/mods/niofs/union/UnionFileSystem.java @@ -157,8 +157,20 @@ public UnionFileSystemProvider provider() { @Override public void close() throws IOException { provider().removeFileSystem(this); + IOException closeException = null; for (var embeddedFs : embeddedFileSystems.values()) { - embeddedFs.fs.close(); + try { + embeddedFs.fs.close(); + } catch (IOException e) { + if (closeException != null) { + closeException.addSuppressed(e); + } else { + closeException = e; + } + } + } + if (closeException != null) { + throw closeException; } } From d2c3d97627ab87927a3f3e1089fab0ebb916163e Mon Sep 17 00:00:00 2001 From: Sebastian Hartte Date: Sun, 2 Jun 2024 18:30:44 +0200 Subject: [PATCH 3/5] Remove AutoClosable from SecureJar to prevent spurious warnings on all getters that return it --- src/main/java/cpw/mods/jarhandling/SecureJar.java | 7 +------ src/main/java/cpw/mods/jarhandling/impl/Jar.java | 1 - 2 files changed, 1 insertion(+), 7 deletions(-) diff --git a/src/main/java/cpw/mods/jarhandling/SecureJar.java b/src/main/java/cpw/mods/jarhandling/SecureJar.java index 9d4e69e..40cfcad 100644 --- a/src/main/java/cpw/mods/jarhandling/SecureJar.java +++ b/src/main/java/cpw/mods/jarhandling/SecureJar.java @@ -17,10 +17,6 @@ import java.security.CodeSigner; import java.util.List; import java.util.Optional; -import java.util.Set; -import java.util.function.BiPredicate; -import java.util.function.Function; -import java.util.function.Supplier; import java.util.jar.Attributes; import java.util.jar.Manifest; @@ -28,7 +24,7 @@ * A secure jar is the full definition for a module, * including all its paths and code signing metadata. */ -public interface SecureJar extends AutoCloseable { +public interface SecureJar { /** * Creates a jar from a list of paths. * See {@link JarContentsBuilder} for more configuration options. @@ -90,7 +86,6 @@ static SecureJar from(JarContents contents, JarMetadata metadata) { * Closes the underlying file system resources (if any). * Renders this object unusable. */ - @Override void close() throws IOException; /** diff --git a/src/main/java/cpw/mods/jarhandling/impl/Jar.java b/src/main/java/cpw/mods/jarhandling/impl/Jar.java index b438aa9..c970e4e 100644 --- a/src/main/java/cpw/mods/jarhandling/impl/Jar.java +++ b/src/main/java/cpw/mods/jarhandling/impl/Jar.java @@ -3,7 +3,6 @@ import cpw.mods.jarhandling.JarMetadata; import cpw.mods.jarhandling.SecureJar; import cpw.mods.niofs.union.UnionFileSystem; -import cpw.mods.niofs.union.UnionPathFilter; import cpw.mods.util.LambdaExceptionUtils; import org.jetbrains.annotations.Nullable; From f73803f45be2b958afd34ee6ef384dc720e7d7fd Mon Sep 17 00:00:00 2001 From: Sebastian Hartte Date: Sun, 2 Jun 2024 18:37:00 +0200 Subject: [PATCH 4/5] Remove AutoClosable from SecureJar to prevent spurious warnings on all getters that return it --- src/test/java/cpw/mods/jarhandling/impl/TestMetadata.java | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/src/test/java/cpw/mods/jarhandling/impl/TestMetadata.java b/src/test/java/cpw/mods/jarhandling/impl/TestMetadata.java index 2eb4d76..c762629 100644 --- a/src/test/java/cpw/mods/jarhandling/impl/TestMetadata.java +++ b/src/test/java/cpw/mods/jarhandling/impl/TestMetadata.java @@ -70,5 +70,9 @@ public Set getPackagesExcluding(String... excludedRootPackages) { public List getMetaInfServices() { return List.of(); } + + @Override + public void close() { + } } } From 25afebd3532f106f6a6594eddc59a6a5bbc93ab2 Mon Sep 17 00:00:00 2001 From: Sebastian Hartte Date: Sun, 2 Jun 2024 20:26:29 +0200 Subject: [PATCH 5/5] Make it extend Closeable --- src/main/java/cpw/mods/jarhandling/JarContents.java | 9 ++------- .../java/cpw/mods/jarhandling/impl/JarContentsImpl.java | 1 - 2 files changed, 2 insertions(+), 8 deletions(-) diff --git a/src/main/java/cpw/mods/jarhandling/JarContents.java b/src/main/java/cpw/mods/jarhandling/JarContents.java index bcbb675..da4366c 100644 --- a/src/main/java/cpw/mods/jarhandling/JarContents.java +++ b/src/main/java/cpw/mods/jarhandling/JarContents.java @@ -2,6 +2,7 @@ import org.jetbrains.annotations.ApiStatus; +import java.io.Closeable; import java.io.IOException; import java.net.URI; import java.nio.file.Path; @@ -19,7 +20,7 @@ * Convert to a full jar with {@link SecureJar#from(JarContents)}. */ @ApiStatus.NonExtendable -public interface JarContents extends AutoCloseable { +public interface JarContents extends Closeable { /** * @see SecureJar#getPrimaryPath() */ @@ -70,10 +71,4 @@ static JarContents of(Path fileOrFolder) { static JarContents of(Collection filesOrFolders) { return new JarContentsBuilder().paths(filesOrFolders.toArray(new Path[0])).build(); } - - /** - * Closes the underlying resources. - */ - @Override - void close() throws IOException; } diff --git a/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java b/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java index c019e56..d198444 100644 --- a/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java +++ b/src/main/java/cpw/mods/jarhandling/impl/JarContentsImpl.java @@ -23,7 +23,6 @@ import java.util.Map; import java.util.Optional; import java.util.Set; -import java.util.function.BiPredicate; import java.util.function.Supplier; import java.util.jar.JarFile; import java.util.jar.JarInputStream;