/* * Copyright (c) 2016, 2023, 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. */ /* * @test * @bug 8165944 8318971 * @summary test several jar tool input file scenarios with variations on -C * options with/without a --release option. Some input files are * duplicates that sometimes cause exceptions and other times do not, * demonstrating identical behavior to JDK 8 jar tool. * @library /test/lib * @modules jdk.jartool * @build jdk.test.lib.Platform * jdk.test.lib.util.FileUtils * @run testng InputFilesTest */ import org.testng.Assert; import org.testng.annotations.AfterMethod; import org.testng.annotations.BeforeMethod; import org.testng.annotations.Test; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.PrintStream; import java.io.UncheckedIOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.util.Arrays; import java.util.spi.ToolProvider; import java.util.stream.Stream; import java.util.zip.ZipException; import jdk.test.lib.util.FileUtils; public class InputFilesTest { private static final ToolProvider JAR_TOOL = ToolProvider.findFirst("jar") .orElseThrow(() -> new RuntimeException("jar tool not found") ); private final String nl = System.lineSeparator(); private final ByteArrayOutputStream baos = new ByteArrayOutputStream(); private final PrintStream out = new PrintStream(baos); private Runnable onCompletion; @BeforeMethod public void reset() { onCompletion = null; } @AfterMethod public void run() { if (onCompletion != null) { onCompletion.run(); } } @Test public void test1() throws IOException { mkdir("test1 test2"); touch("test1/testfile1 test2/testfile2"); jar("cf test.jar -C test1 . -C test2 ."); jar("tf test.jar"); println(); String output = "META-INF/" + nl + "META-INF/MANIFEST.MF" + nl + "testfile1" + nl + "testfile2" + nl; rm("test.jar test1 test2"); Assert.assertEquals(baos.toByteArray(), output.getBytes()); } @Test public void test2() throws IOException { mkdir("test1 test2 test3 test4"); touch("test1/testfile1 test2/testfile2 test3/testfile3 test4/testfile4"); jar("cf test.jar -C test1 . -C test2 . --release 9 -C test3 . -C test4 ."); jar("tf test.jar"); println(); String output = "META-INF/" + nl + "META-INF/MANIFEST.MF" + nl + "testfile1" + nl + "testfile2" + nl + "META-INF/versions/9/" + nl + "META-INF/versions/9/testfile3" + nl + "META-INF/versions/9/testfile4" + nl; rm("test.jar test1 test2 test3 test4"); Assert.assertEquals(baos.toByteArray(), output.getBytes()); } @Test public void test3() throws IOException { touch("test"); jar("cf test.jar test test"); jar("tf test.jar"); println(); String output = "META-INF/" + nl + "META-INF/MANIFEST.MF" + nl + "test" + nl; rm("test.jar test"); Assert.assertEquals(baos.toByteArray(), output.getBytes()); } @Test public void test4() throws IOException { mkdir("a"); touch("a/test"); jar("cf test.jar -C a test -C a test"); jar("tf test.jar"); println(); String output = "META-INF/" + nl + "META-INF/MANIFEST.MF" + nl + "test" + nl; rm("test.jar a"); Assert.assertEquals(baos.toByteArray(), output.getBytes()); } @Test(expectedExceptions = {ZipException.class}) public void test5() throws IOException { mkdir("a"); touch("test a/test"); onCompletion = () -> rm("test a"); jar("cf test.jar -C a test test"); } @Test(expectedExceptions = {ZipException.class}) public void test6() throws IOException { mkdir("test1 test2"); touch("test1/a test2/a"); onCompletion = () -> rm("test1 test2"); jar("cf test.jar --release 9 -C test1 a -C test2 a"); } /** * Containing non-existent file in the file list * The final jar should not be created and correct error message should be caught. * IOException is triggered as expected. */ @Test public void testNonExistentFileInput() throws IOException { touch("existingTestFile.txt"); onCompletion = () -> rm("existingTestFile.txt"); try { jar("cf test.jar existingTestFile.txt nonExistentTestFile.txt"); Assert.fail("jar tool unexpectedly completed successfully"); } catch (IOException e) { Assert.assertEquals(e.getMessage().trim(), "nonExistentTestFile.txt : no such file or directory"); Assert.assertTrue(Files.notExists(Path.of("test.jar")), "Jar file should not be created."); } } /** * With @File as a part of jar command line, where the File is containing one or more * non-existent files or directories * The final jar should not be created and correct error message should be caught. * IOException is triggered as expected. */ @Test public void testNonExistentFileInputClassList() throws IOException { touch("existingTestFile.txt"); touch("classes.list"); Files.writeString(Path.of("classes.list"), """ existingTestFile.txt nonExistentTestFile.txt nonExistentDirectory """); onCompletion = () -> rm("existingTestFile.txt classes.list"); try { jar("cf test.jar @classes.list"); Assert.fail("jar tool unexpectedly completed successfully"); } catch (IOException e) { String msg = e.getMessage().trim(); Assert.assertTrue(msg.contains("nonExistentTestFile.txt : no such file or directory")); Assert.assertTrue(msg.trim().contains("nonExistentDirectory : no such file or directory")); Assert.assertTrue(Files.notExists(Path.of("test.jar")), "Jar file should not be created."); } } /** * Create a jar file; then with @File as a part of jar command line, where the File is containing one or more * non-existent files or directories * The final jar should not be created and correct error message should be caught. * IOException is triggered as expected. */ @Test public void testUpdateNonExistentFileInputClassList() throws IOException { touch("existingTestFileUpdate.txt"); touch("existingTestFileUpdate2.txt"); touch("classesUpdate.list"); Files.writeString(Path.of("classesUpdate.list"), """ existingTestFileUpdate2.txt nonExistentTestFileUpdate.txt nonExistentDirectoryUpdate """); onCompletion = () -> rm("existingTestFileUpdate.txt existingTestFileUpdate2.txt " + "classesUpdate.list testUpdate.jar"); try { jar("cf testUpdate.jar existingTestFileUpdate.txt"); Assert.assertTrue(Files.exists(Path.of("testUpdate.jar"))); jar("uf testUpdate.jar @classesUpdate.list"); Assert.fail("jar tool unexpectedly completed successfully"); } catch (IOException e) { String msg = e.getMessage().trim(); Assert.assertFalse(msg.contains("existingTestFileUpdate.txt : no such file or directory")); Assert.assertTrue(msg.contains("nonExistentTestFileUpdate.txt : no such file or directory")); Assert.assertTrue(msg.trim().contains("nonExistentDirectoryUpdate : no such file or directory")); } } private Stream mkpath(String... args) { return Arrays.stream(args).map(d -> Paths.get(".", d.split("/"))); } private void mkdir(String cmdline) { System.out.println("mkdir -p " + cmdline); mkpath(cmdline.split(" +")).forEach(p -> { try { Files.createDirectories(p); } catch (IOException x) { throw new UncheckedIOException(x); } }); } private void touch(String cmdline) { System.out.println("touch " + cmdline); mkpath(cmdline.split(" +")).forEach(p -> { try { Files.createFile(p); } catch (IOException x) { throw new UncheckedIOException(x); } }); } private void rm(String cmdline) { System.out.println("rm -rf " + cmdline); mkpath(cmdline.split(" +")).forEach(p -> { try { if (Files.isDirectory(p)) { FileUtils.deleteFileTreeWithRetry(p); } else { FileUtils.deleteFileIfExistsWithRetry(p); } } catch (IOException x) { throw new UncheckedIOException(x); } }); } private void jar(String cmdline) throws IOException { System.out.println("jar " + cmdline); baos.reset(); // the run method catches IOExceptions, we need to expose them ByteArrayOutputStream baes = new ByteArrayOutputStream(); PrintStream err = new PrintStream(baes); PrintStream saveErr = System.err; System.setErr(err); int rc = JAR_TOOL.run(out, err, cmdline.split(" +")); System.setErr(saveErr); if (rc != 0) { String s = baes.toString(); if (s.startsWith("java.util.zip.ZipException: duplicate entry: ")) { throw new ZipException(s); } throw new IOException(s); } } private void println() throws IOException { System.out.println(new String(baos.toByteArray())); } }