From ab7cfde85c5517cfd9d38af3ddbab7187db1bfb5 Mon Sep 17 00:00:00 2001 From: Lance Andersen Date: Wed, 18 Dec 2019 12:03:56 -0500 Subject: [PATCH] 8229888: (zipfs) Updating an existing zip file does not preserve original permissions Reviewed-by: clanger, alanb, bpb --- .../classes/jdk/nio/zipfs/ZipFileSystem.java | 33 ++- .../jdk/nio/zipfs/ZipFSPermissionsTest.java | 229 ++++++++++++++++++ .../jdk/nio/zipfs/ZipFSPermissionsTest.policy | 5 + 3 files changed, 264 insertions(+), 3 deletions(-) create mode 100644 test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.java create mode 100644 test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.policy diff --git a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystem.java b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystem.java index 14d74c8603e..f9553e8c409 100644 --- a/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystem.java +++ b/src/jdk.zipfs/share/classes/jdk/nio/zipfs/ZipFileSystem.java @@ -1567,9 +1567,7 @@ class ZipFileSystem extends FileSystem { // Creates a new empty temporary file in the same directory as the // specified file. A variant of Files.createTempFile. - private Path createTempFileInSameDirectoryAs(Path path) - throws IOException - { + private Path createTempFileInSameDirectoryAs(Path path) throws IOException { Path parent = path.toAbsolutePath().getParent(); Path dir = (parent == null) ? path.getFileSystem().getPath(".") : parent; Path tmpPath = Files.createTempFile(dir, "zipfstmp", null); @@ -1715,6 +1713,7 @@ class ZipFileSystem extends FileSystem { } if (!hasUpdate) return; + PosixFileAttributes attrs = getPosixAttributes(zfpath); Path tmpFile = createTempFileInSameDirectoryAs(zfpath); try (OutputStream os = new BufferedOutputStream(Files.newOutputStream(tmpFile, WRITE))) { ArrayList elist = new ArrayList<>(inodes.size()); @@ -1791,10 +1790,38 @@ class ZipFileSystem extends FileSystem { Files.delete(zfpath); } + // Set the POSIX permissions of the original Zip File if available + // before moving the temp file + if (attrs != null) { + Files.setPosixFilePermissions(tmpFile, attrs.permissions()); + } Files.move(tmpFile, zfpath, REPLACE_EXISTING); hasUpdate = false; // clear } + /** + * Returns a file's POSIX file attributes. + * @param path The path to the file + * @return The POSIX file attributes for the specified file or + * null if the POSIX attribute view is not available + * @throws IOException If an error occurs obtaining the POSIX attributes for + * the specified file + */ + private PosixFileAttributes getPosixAttributes(Path path) throws IOException { + try { + PosixFileAttributeView view = + Files.getFileAttributeView(path, PosixFileAttributeView.class); + // Return if the attribute view is not supported + if (view == null) { + return null; + } + return view.readAttributes(); + } catch (UnsupportedOperationException e) { + // PosixFileAttributes not available + return null; + } + } + private IndexNode getInode(byte[] path) { return inodes.get(IndexNode.keyOf(Objects.requireNonNull(entryLookup.apply(path), "path"))); } diff --git a/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.java b/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.java new file mode 100644 index 00000000000..487e21b89e7 --- /dev/null +++ b/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.java @@ -0,0 +1,229 @@ +/* + * Copyright (c) 2019, 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 + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + * + */ + +import org.testng.SkipException; +import org.testng.annotations.*; + +import java.io.IOException; +import java.nio.file.FileSystem; +import java.nio.file.FileSystems; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.attribute.PosixFileAttributeView; +import java.nio.file.attribute.PosixFileAttributes; +import java.nio.file.attribute.PosixFilePermission; +import java.nio.file.attribute.PosixFilePermissions; +import java.util.Map; +import java.util.Set; + +import static java.nio.file.attribute.PosixFilePermission.*; +import static org.testng.Assert.assertEquals; + +/** + * @test + * @bug 8229888 + * @summary Updating an existing zip file does not preserve original permissions + * @library /test/lib + * @modules jdk.zipfs + * @run testng/othervm ZipFSPermissionsTest + * @run testng/othervm/java.security.policy=ZipFSPermissionsTest.policy ZipFSPermissionsTest + */ +public class ZipFSPermissionsTest { + + // Files used for testing + private static final Path zipFile = Path.of("zipPermsTest.zip"); + private static final Path entry0 = Path.of("Entry-0.txt"); + // Path of 2nd file to add to the Zip file + private static final Path entry1 = Path.of("Entry-1.txt"); + + // Enable for permissions output + private static final boolean DEBUG = false; + + /** + * Create the files used by the test + */ + @BeforeSuite + public void setUp() throws Exception { + boolean supportsPosix = FileSystems.getDefault() + .supportedFileAttributeViews().contains("posix"); + + // Check to see if File System supports POSIX permissions + if (supportsPosix) { + System.out.println("File Store Supports Posix"); + } else { + // As there is no POSIX permission support, skip running the test + throw new SkipException("Cannot set permissions on this File Store"); + } + Files.writeString(entry0, "Tennis Pro"); + Files.writeString(entry1, "Tennis is a lifetime sport!"); + } + + /** + * Re-create the initial Zip file prior to each run. + */ + @BeforeMethod + public void before() throws Exception { + Files.deleteIfExists(zipFile); + zip(zipFile, Map.of("create", "true"), entry0); + } + + /** + * Remove Zip file used by test after each run. + */ + @AfterMethod + public void tearDown() throws Exception { + Files.deleteIfExists(zipFile); + } + + /** + * Remove files used by test as part of final test run clean-up + */ + @AfterSuite + public void suiteCleanUp() throws Exception { + Files.deleteIfExists(zipFile); + Files.deleteIfExists(entry0); + Files.deleteIfExists(entry1); + } + + /** + * Validate that the Zip file permissions are as expected after updating the + * Zip file + * @param newPerms The permissions to set on the Zip File before updating the + * file + * @throws Exception If an error occurs + */ + @Test(dataProvider = "posixPermissions") + public void testZipPerms(Set newPerms) throws Exception { + if (DEBUG) { + System.out.printf("Test Run with perms= %s%n", newPerms); + } + + PosixFileAttributes attrs = getPosixAttributes(zipFile); + + // Permissions used to verify the results of updating the Zip file + if (newPerms == null) { + // Use current Zip File permissions; + newPerms = attrs.permissions(); + } + displayPermissions("Original permissions", zipFile); + + // Now set the new permissions + Files.setPosixFilePermissions(zipFile, newPerms); + displayPermissions("Revised permissions", zipFile); + + // Update the Zip file + zip(zipFile, Map.of(), entry1); + + // Validate that the permissions are as expected after updating the + // Zip file + PosixFileAttributes afterAttrs = getPosixAttributes(zipFile); + displayPermissions("Permissions after updating the Zip File", zipFile); + assertEquals(afterAttrs.permissions(), newPerms, + "Permissions were not updated as expected!"); + } + + /** + * Display the permissions for the specified Zip file when {@code DEBUG} + * is set to {@code true} + * + * @param msg String to include in the message + * @param zipFile Path to the Zip File + * @throws IOException If an error occurs obtaining the permissions + */ + public void displayPermissions(String msg, Path zipFile) throws IOException { + if (DEBUG) { + PosixFileAttributeView view = Files.getFileAttributeView(zipFile, + PosixFileAttributeView.class); + if (view == null) { + System.out.println("Could not obtain a PosixFileAttributeView!"); + return; + } + PosixFileAttributes attrs = view.readAttributes(); + System.out.printf("%s: %s, Owner: %s, Group:%s, permissions: %s%n", msg, + zipFile.getFileName(), attrs.owner().getName(), + attrs.group().getName(), PosixFilePermissions.toString(attrs.permissions())); + } + } + + /** + * Create a Zip File System using the specified properties and a Zip file + * with the specified number of entries + * + * @param zipFile Path to the Zip File to create/update + * @param env Properties used for creating the Zip Filesystem + * @param source The path of the file to add to the Zip File + * @throws IOException If an error occurs while creating/updating the Zip file + */ + public void zip(Path zipFile, Map env, Path source) throws IOException { + if (DEBUG) { + System.out.printf("File:%s, adding:%s%n", zipFile.toAbsolutePath(), source); + } + try (FileSystem zipfs = + FileSystems.newFileSystem(zipFile, env)) { + Files.copy(source, zipfs.getPath(source.getFileName().toString())); + } + } + + /** + * Returns a file's POSIX file attributes. + * + * @param path The path to the Zip file + * @return The POSIX file attributes for the specified file or + * null if the POSIX attribute view is not available + * @throws IOException If an error occurs obtaining the POSIX attributes for + * the specified file + */ + public PosixFileAttributes getPosixAttributes(Path path) throws IOException { + PosixFileAttributes attrs = null; + PosixFileAttributeView view = + Files.getFileAttributeView(path, PosixFileAttributeView.class); + // Return if the attribute view is not supported + if (view == null) { + return null; + } + attrs = view.readAttributes(); + return attrs; + } + + /* + * DataProvider used to verify the permissions on a Zip file + * are as expected after updating the Zip file + */ + @DataProvider(name = "posixPermissions") + private Object[][] posixPermissions() { + return new Object[][]{ + {null}, + {Set.of(OWNER_READ, OWNER_WRITE, OTHERS_READ)}, + {Set.of(OWNER_READ, OWNER_WRITE, OWNER_EXECUTE)}, + {Set.of(OWNER_READ, OWNER_WRITE, OTHERS_READ, OTHERS_WRITE)}, + {Set.of(OWNER_READ, OWNER_WRITE, OWNER_EXECUTE, OTHERS_READ, + OTHERS_WRITE, OTHERS_EXECUTE)}, + {Set.of(OWNER_READ, OWNER_WRITE, OWNER_EXECUTE, + GROUP_READ, GROUP_WRITE,GROUP_EXECUTE, OTHERS_READ, + OTHERS_WRITE, OTHERS_EXECUTE)}, + {Set.of(OWNER_READ, OWNER_WRITE, GROUP_READ, GROUP_WRITE, + OTHERS_READ, OTHERS_WRITE)}, + }; + } +} diff --git a/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.policy b/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.policy new file mode 100644 index 00000000000..8bd0028ca85 --- /dev/null +++ b/test/jdk/jdk/nio/zipfs/ZipFSPermissionsTest.policy @@ -0,0 +1,5 @@ +grant { + permission java.io.FilePermission "<>","read,write,delete"; + permission java.util.PropertyPermission "user.dir","read"; + permission java.lang.RuntimePermission "accessUserInformation"; +};