517 lines
25 KiB
Java
517 lines
25 KiB
Java
|
/*
|
||
|
* Copyright (c) 2021, 2024, 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 java.io.ByteArrayOutputStream;
|
||
|
import java.io.File;
|
||
|
import java.io.IOException;
|
||
|
import java.io.PrintStream;
|
||
|
import java.nio.charset.StandardCharsets;
|
||
|
import java.nio.file.Files;
|
||
|
import java.nio.file.Path;
|
||
|
import java.util.ArrayList;
|
||
|
import java.util.Collection;
|
||
|
import java.util.List;
|
||
|
import java.util.spi.ToolProvider;
|
||
|
import java.util.stream.Stream;
|
||
|
|
||
|
import jdk.test.lib.util.JarBuilder;
|
||
|
import org.junit.jupiter.api.AfterEach;
|
||
|
import org.junit.jupiter.api.Assumptions;
|
||
|
import org.junit.jupiter.api.BeforeEach;
|
||
|
import org.junit.jupiter.api.Test;
|
||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||
|
import org.junit.jupiter.params.provider.Arguments;
|
||
|
import org.junit.jupiter.params.provider.MethodSource;
|
||
|
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
|
||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||
|
import static org.junit.jupiter.api.Assertions.assertNotEquals;
|
||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||
|
|
||
|
/*
|
||
|
* @test
|
||
|
* @bug 8173970
|
||
|
* @summary jar tool should allow extracting to specific directory
|
||
|
* @library /test/lib
|
||
|
* @comment The test relies on verification of error messages generated by jar tool, so we use
|
||
|
* a fixed en_US locale for this test.
|
||
|
* @run junit/othervm -Duser.language=en -Duser.country=US JarExtractTest
|
||
|
*/
|
||
|
public class JarExtractTest {
|
||
|
private static final ToolProvider JAR_TOOL = ToolProvider.findFirst("jar")
|
||
|
.orElseThrow(() ->
|
||
|
new RuntimeException("jar tool not found")
|
||
|
);
|
||
|
|
||
|
private static final byte[] FILE_CONTENT = "Hello world!!!".getBytes(StandardCharsets.UTF_8);
|
||
|
// the jar that will get extracted in the tests
|
||
|
private Path testJarPath;
|
||
|
private static Collection<Path> filesToDelete = new ArrayList<>();
|
||
|
|
||
|
@BeforeEach
|
||
|
public void createTestJar() throws Exception {
|
||
|
final String tmpDir = Files.createTempDirectory("8173970-").toString();
|
||
|
testJarPath = Path.of(tmpDir, "8173970-test.jar");
|
||
|
final JarBuilder builder = new JarBuilder(testJarPath.toString());
|
||
|
// d1
|
||
|
// |--- d2
|
||
|
// | |--- d3
|
||
|
// | | |--- f2.txt
|
||
|
// |
|
||
|
// |--- d4
|
||
|
// ...
|
||
|
// f1.txt
|
||
|
|
||
|
builder.addEntry("d1/", new byte[0]);
|
||
|
builder.addEntry("f1.txt", FILE_CONTENT);
|
||
|
builder.addEntry("d1/d2/d3/f2.txt", FILE_CONTENT);
|
||
|
builder.addEntry("d1/d4/", new byte[0]);
|
||
|
builder.build();
|
||
|
}
|
||
|
|
||
|
@AfterEach
|
||
|
public void cleanup() {
|
||
|
for (final Path p : filesToDelete) {
|
||
|
try {
|
||
|
System.out.println("Deleting file/dir " + p);
|
||
|
Files.delete(p);
|
||
|
} catch (IOException ioe) {
|
||
|
//ignore
|
||
|
System.err.println("ignoring exception: " + ioe
|
||
|
+ " that happened when deleting: " + p);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates and returns various relative paths, to which the jar will be extracted in the tests
|
||
|
*/
|
||
|
static Stream<Arguments> provideRelativeExtractLocations() throws Exception {
|
||
|
// create some dirs so that they already exist when the jar is being extracted
|
||
|
final String existing1 = "." + File.separator + "8173970-existing-1";
|
||
|
Files.createDirectories(Path.of(existing1));
|
||
|
final String existing2 = "." + File.separator + "foo" + File.separator + "8173970-existing-2";
|
||
|
Files.createDirectories(Path.of(existing2));
|
||
|
final Path dirOutsideScratchDir = Files.createTempDirectory(Path.of(".."), "8173970");
|
||
|
// we need to explicitly delete this dir after the tests end
|
||
|
filesToDelete.add(dirOutsideScratchDir);
|
||
|
final String existing3 = dirOutsideScratchDir.toString() + File.separator + "8173970-existing-3";
|
||
|
Files.createDirectories(Path.of(existing3));
|
||
|
|
||
|
final String anotherDirOutsideScratchDir = ".." + File.separator + "8173970-non-existent";
|
||
|
filesToDelete.add(Path.of(anotherDirOutsideScratchDir));
|
||
|
|
||
|
final List<Arguments> args = new ArrayList<>();
|
||
|
args.add(Arguments.of(".")); // current dir
|
||
|
// (explicitly) relative to current dir
|
||
|
args.add(Arguments.of("." + File.separator + "8173970-extract-1"));
|
||
|
// (implicitly) relative to current dir
|
||
|
args.add(Arguments.of("8173970-extract-2"));
|
||
|
// sibling to current dir
|
||
|
args.add(Arguments.of(anotherDirOutsideScratchDir));
|
||
|
// some existing dirs
|
||
|
args.add(Arguments.of(existing1));
|
||
|
args.add(Arguments.of(existing2));
|
||
|
args.add(Arguments.of(existing3));
|
||
|
// a non-existent dir within an existing dir
|
||
|
args.add(Arguments.of(existing1 + File.separator
|
||
|
+ "non-existing" + File.separator + "foo"));
|
||
|
return args.stream();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates and returns various absolute paths, to which the jar will be extracted in the tests
|
||
|
*/
|
||
|
static Stream<Arguments> provideAbsoluteExtractLocations() throws Exception {
|
||
|
final Stream<Arguments> relative = provideRelativeExtractLocations();
|
||
|
return relative.map((arg) -> {
|
||
|
final String relPath = (String) arg.get()[0];
|
||
|
return Arguments.of(Path.of(relPath).toAbsolutePath().toString());
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates and returns various normalized paths, to which the jar will be extracted in the tests
|
||
|
*/
|
||
|
static Stream<Arguments> provideAbsoluteNormalizedExtractLocations() throws Exception {
|
||
|
final Stream<Arguments> relative = provideRelativeExtractLocations();
|
||
|
return relative.map((arg) -> {
|
||
|
final String relPath = (String) arg.get()[0];
|
||
|
return Arguments.of(Path.of(relPath).toAbsolutePath().normalize().toString());
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extracts a jar to various relative paths, using the -C/--dir option and then
|
||
|
* verifies that the extracted content is at the expected locations with the correct
|
||
|
* content
|
||
|
*/
|
||
|
@ParameterizedTest
|
||
|
@MethodSource("provideRelativeExtractLocations")
|
||
|
public void testExtractToRelativeDir(final String dest) throws Exception {
|
||
|
testLongFormExtract(dest);
|
||
|
testExtract(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extracts a jar to various absolute paths, using the -C/--dir option and then
|
||
|
* verifies that the extracted content is at the expected locations with the correct
|
||
|
* content
|
||
|
*/
|
||
|
@ParameterizedTest
|
||
|
@MethodSource("provideAbsoluteExtractLocations")
|
||
|
public void testExtractToAbsoluteDir(final String dest) throws Exception {
|
||
|
testExtract(dest);
|
||
|
testLongFormExtract(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extracts a jar to various normalized paths (i.e. no {@code .} or @{code ..} in the path components),
|
||
|
* using the -C/--dir option and then verifies that the extracted content is at the expected locations
|
||
|
* with the correct content
|
||
|
*/
|
||
|
@ParameterizedTest
|
||
|
@MethodSource("provideAbsoluteNormalizedExtractLocations")
|
||
|
public void testExtractToAbsoluteNormalizedDir(final String dest) throws Exception {
|
||
|
testExtract(dest);
|
||
|
testLongFormExtract(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Test that extracting a jar with {@code jar -x -f --dir} works as expected
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractLongFormDir() throws Exception {
|
||
|
final String dest = "foo-bar";
|
||
|
System.out.println("Extracting " + testJarPath + " to " + dest);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, "-x", "-f", testJarPath.toString(),
|
||
|
"--dir", dest);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath + " to " + dest);
|
||
|
verifyExtractedContent(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Verifies that the {@code jar --help} output contains the --dir option
|
||
|
*/
|
||
|
@Test
|
||
|
public void testHelpOutput() {
|
||
|
final ByteArrayOutputStream outStream = new ByteArrayOutputStream();
|
||
|
final int exitCode = JAR_TOOL.run(new PrintStream(outStream), System.err, "--help");
|
||
|
assertEquals(0, exitCode, "jar --help command failed");
|
||
|
final String output = outStream.toString();
|
||
|
// this message is expected to be the one from the jar --help output which is sourced from
|
||
|
// jar.properties
|
||
|
final String expectedMsg = "--dir Directory into which the jar will be extracted";
|
||
|
assertTrue(output.contains(expectedMsg), "jar --help didn't contain --dir option");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that {@code jar -x -f} command works fine even when the -C or --dir option
|
||
|
* isn't specified
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractWithoutOutputDir() throws Exception {
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, "-x", "-f", testJarPath.toString());
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath);
|
||
|
// the content would have been extracted to current dir
|
||
|
verifyExtractedContent(".");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that {@code jar --extract -f} command works fine even when the -C or --dir option
|
||
|
* isn't specified
|
||
|
*/
|
||
|
@Test
|
||
|
public void testLongFormExtractWithoutOutputDir() throws Exception {
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, "--extract", "-f", testJarPath.toString());
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath);
|
||
|
// the content would have been extracted to current dir
|
||
|
verifyExtractedContent(".");
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that when the destination directory specified for jar extract is actually a file
|
||
|
* or one of the path component in the specified destination path is a file, then the
|
||
|
* extraction fails.
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractToNonDirectory() throws Exception {
|
||
|
final String expectedErrMsg = "could not create directory";
|
||
|
final Path notADir1 = Files.createTempFile(Path.of("."), "8173970", ".txt");
|
||
|
final Path notADir2 = notADir1.resolve("foobar");
|
||
|
for (final Path dest : List.of(notADir1, notADir2)) {
|
||
|
final String[] args = {"-x", "-f", testJarPath.toString(), "-C", dest.toString()};
|
||
|
final ByteArrayOutputStream err = new ByteArrayOutputStream();
|
||
|
printJarCommand(args);
|
||
|
int exitCode = JAR_TOOL.run(System.out, new PrintStream(err), args);
|
||
|
assertNotEquals(0, exitCode, "jar extraction was expected to fail but didn't");
|
||
|
// verify it did indeed fail due to the right reason
|
||
|
assertTrue(err.toString(StandardCharsets.UTF_8).contains(expectedErrMsg));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that extracting a jar using {@code -P} flag and without any explicit destination
|
||
|
* directory works correctly if the jar contains entries with leading slashes and/or {@code ..}
|
||
|
* parts preserved.
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractNoDestDirWithPFlag() throws Exception {
|
||
|
// run this test only on those systems where "/tmp" directory is available and we
|
||
|
// can write to it
|
||
|
Assumptions.assumeTrue(Files.isDirectory(Path.of("/tmp")),
|
||
|
"skipping test, since /tmp isn't a directory");
|
||
|
// try and write into "/tmp"
|
||
|
final Path tmpDir;
|
||
|
try {
|
||
|
tmpDir = Files.createTempDirectory(Path.of("/tmp"), "8173970-").toAbsolutePath();
|
||
|
} catch (IOException ioe) {
|
||
|
Assumptions.abort("skipping test, since /tmp cannot be written to: " + ioe);
|
||
|
return;
|
||
|
}
|
||
|
final String leadingSlashEntryName = tmpDir.toString() + "/foo/f1.txt";
|
||
|
// create a jar which has leading slash (/) and dot-dot (..) preserved in entry names
|
||
|
final Path jarPath = createJarWithPFlagSemantics(leadingSlashEntryName);
|
||
|
final List<String[]> cmdArgs = new ArrayList<>();
|
||
|
cmdArgs.add(new String[]{"-xvfP", jarPath.toString()});
|
||
|
cmdArgs.add(new String[]{"--extract", "-v", "-P", "-f", jarPath.toString()});
|
||
|
try {
|
||
|
for (final String[] args : cmdArgs) {
|
||
|
printJarCommand(args);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, args);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + jarPath);
|
||
|
final String dest = ".";
|
||
|
assertTrue(Files.isDirectory(Path.of(dest)), dest + " is not a directory");
|
||
|
final Path d1 = Path.of(dest, "d1");
|
||
|
assertTrue(Files.isDirectory(d1), d1 + " directory is missing or not a directory");
|
||
|
final Path d2 = Path.of(dest, "d1", "d2");
|
||
|
assertTrue(Files.isDirectory(d2), d2 + " directory is missing or not a directory");
|
||
|
final Path f1 = Path.of(leadingSlashEntryName);
|
||
|
assertTrue(Files.isRegularFile(f1), f1 + " is missing or not a file");
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f1),
|
||
|
"Unexpected content in file " + f1);
|
||
|
final Path f2 = Path.of("d1/d2/../f2.txt");
|
||
|
assertTrue(Files.isRegularFile(f2), f2 + " is missing or not a file");
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f2),
|
||
|
"Unexpected content in file " + f2);
|
||
|
}
|
||
|
} finally {
|
||
|
// clean up the file that might have been extracted into "/tmp/...." directory
|
||
|
Files.deleteIfExists(Path.of(leadingSlashEntryName));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that the {@code -P} option cannot be used during jar extraction when the {@code -C} and/or
|
||
|
* {@code --dir} option is used
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractWithDirPFlagNotAllowed() throws Exception {
|
||
|
// this error message is expected to be the one from the jar --help output which is sourced from
|
||
|
// jar.properties
|
||
|
final String expectedErrMsg = "You may not specify '-Px' with the '-C' or '--dir' options";
|
||
|
final String tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toString();
|
||
|
final List<String[]> cmdArgs = new ArrayList<>();
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "-P", "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "-P", "--dir", tmpDir});
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "-P", "-C", "."});
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "-P", "--dir", "."});
|
||
|
cmdArgs.add(new String[]{"-xvfP", testJarPath.toString(), "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "-P", "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "-P", "--dir", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "-P", "-C", "."});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "-P", "--dir", "."});
|
||
|
for (final String[] args : cmdArgs) {
|
||
|
final ByteArrayOutputStream err = new ByteArrayOutputStream();
|
||
|
printJarCommand(args);
|
||
|
int exitCode = JAR_TOOL.run(System.out, new PrintStream(err), args);
|
||
|
assertNotEquals(0, exitCode, "jar extraction was expected to fail but didn't");
|
||
|
// verify it did indeed fail due to the right reason
|
||
|
assertTrue(err.toString(StandardCharsets.UTF_8).contains(expectedErrMsg));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that {@code jar -xvf <jarname> -C <dir>} works fine too
|
||
|
*/
|
||
|
@Test
|
||
|
public void testLegacyCompatibilityMode() throws Exception {
|
||
|
final String tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toString();
|
||
|
final String[] args = new String[]{"-xvf", testJarPath.toString(), "-C", tmpDir};
|
||
|
printJarCommand(args);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, args);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath);
|
||
|
verifyExtractedContent(tmpDir);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that when multiple directories are specified for extracting the jar, the jar extraction
|
||
|
* fails
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractFailWithMultipleDir() throws Exception {
|
||
|
// this error message is expected to be the one from the jar --help output which is sourced from
|
||
|
// jar.properties
|
||
|
final String expectedErrMsg = "You may not specify the '-C' or '--dir' option more than once with the '-x' option";
|
||
|
final String tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toString();
|
||
|
final List<String[]> cmdArgs = new ArrayList<>();
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "-C", tmpDir, "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "--dir", tmpDir, "--dir", tmpDir});
|
||
|
cmdArgs.add(new String[]{"-x", "-f", testJarPath.toString(), "--dir", tmpDir, "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "-C", tmpDir, "-C", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "--dir", tmpDir, "--dir", tmpDir});
|
||
|
cmdArgs.add(new String[]{"--extract", "-f", testJarPath.toString(), "--dir", tmpDir, "-C", tmpDir});
|
||
|
for (final String[] args : cmdArgs) {
|
||
|
final ByteArrayOutputStream err = new ByteArrayOutputStream();
|
||
|
printJarCommand(args);
|
||
|
int exitCode = JAR_TOOL.run(System.out, new PrintStream(err), args);
|
||
|
assertNotEquals(0, exitCode, "jar extraction was expected to fail but didn't");
|
||
|
// verify it did indeed fail due to the right reason
|
||
|
assertTrue(err.toString(StandardCharsets.UTF_8).contains(expectedErrMsg));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Tests that extracting only specific files from a jar, into a specific destination directory,
|
||
|
* works as expected
|
||
|
*/
|
||
|
@Test
|
||
|
public void testExtractPartialContent() throws Exception {
|
||
|
String tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toString();
|
||
|
String[] cmdArgs = new String[]{"-x", "-f", testJarPath.toString(), "--dir", tmpDir,
|
||
|
"f1.txt", "d1/d2/d3/f2.txt"};
|
||
|
testExtractPartialContent(tmpDir, cmdArgs);
|
||
|
|
||
|
tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toString();
|
||
|
cmdArgs = new String[]{"--extract", "-f", testJarPath.toString(), "--dir", tmpDir,
|
||
|
"f1.txt", "d1/d2/d3/f2.txt"};
|
||
|
testExtractPartialContent(tmpDir, cmdArgs);
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extract to destDir using the passed command arguments and verify the extracted content
|
||
|
*/
|
||
|
private void testExtractPartialContent(final String destDir, final String[] extractArgs) throws Exception {
|
||
|
printJarCommand(extractArgs);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, extractArgs);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath);
|
||
|
// make sure only the specific files were extracted
|
||
|
final Stream<Path> paths = Files.walk(Path.of(destDir));
|
||
|
// files/dirs count expected to be found when the location to which the jar was extracted
|
||
|
// is walked.
|
||
|
// 1) The top level dir being walked 2) f1.txt file 3) d1 dir 4) d1/d2 dir
|
||
|
// 5) d1/d2/d3 dir 6) d1/d2/d3/f2.txt file
|
||
|
final int numExpectedFiles = 6;
|
||
|
assertEquals(numExpectedFiles, paths.count(), "Unexpected number of files/dirs in " + destDir);
|
||
|
final Path f1 = Path.of(destDir, "f1.txt");
|
||
|
assertTrue(Files.isRegularFile(f1), f1.toString() + " wasn't extracted from " + testJarPath);
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f1), "Unexpected content in file " + f1);
|
||
|
final Path d1 = Path.of(destDir, "d1");
|
||
|
assertTrue(Files.isDirectory(d1), d1.toString() + " wasn't extracted from " + testJarPath);
|
||
|
assertEquals(2, Files.walk(d1, 1).count(), "Unexpected number " +
|
||
|
"of files/dirs in " + d1);
|
||
|
final Path d2 = Path.of(d1.toString(), "d2");
|
||
|
assertTrue(Files.isDirectory(d2), d2.toString() + " wasn't extracted from " + testJarPath);
|
||
|
assertEquals(2, Files.walk(d2, 1).count(), "Unexpected number " +
|
||
|
"of files/dirs in " + d2);
|
||
|
final Path d3 = Path.of(d2.toString(), "d3");
|
||
|
assertTrue(Files.isDirectory(d3), d3.toString() + " wasn't extracted from " + testJarPath);
|
||
|
assertEquals(2, Files.walk(d3, 1).count(), "Unexpected number " +
|
||
|
"of files/dirs in " + d3);
|
||
|
final Path f2 = Path.of(d3.toString(), "f2.txt");
|
||
|
assertTrue(Files.isRegularFile(f2), f2.toString() + " wasn't extracted from " + testJarPath);
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f2), "Unexpected content in file " + f2);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extracts the jar file using {@code jar -x -f <jarfile> -C <dest>} and verifies the extracted content
|
||
|
*/
|
||
|
private void testExtract(final String dest) throws Exception {
|
||
|
final String[] args = new String[]{"-x", "-f", testJarPath.toString(), "-C", dest};
|
||
|
printJarCommand(args);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, args);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath + " to " + dest);
|
||
|
verifyExtractedContent(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Extracts the jar file using {@code jar --extract -f <jarfile> -C <dest>} and verifies the
|
||
|
* extracted content
|
||
|
*/
|
||
|
private void testLongFormExtract(final String dest) throws Exception {
|
||
|
final String[] args = new String[]{"--extract", "-f", testJarPath.toString(), "-C", dest};
|
||
|
printJarCommand(args);
|
||
|
final int exitCode = JAR_TOOL.run(System.out, System.err, args);
|
||
|
assertEquals(0, exitCode, "Failed to extract " + testJarPath + " to " + dest);
|
||
|
verifyExtractedContent(dest);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Verifies that the extracted jar content matches what was present in the original jar
|
||
|
*/
|
||
|
private void verifyExtractedContent(final String dest) throws IOException {
|
||
|
assertTrue(Files.isDirectory(Path.of(dest)), dest + " is not a directory");
|
||
|
final Path d1 = Path.of(dest, "d1");
|
||
|
assertTrue(Files.isDirectory(d1), d1 + " directory is missing or not a directory");
|
||
|
final Path d2 = Path.of(dest, "d1", "d2");
|
||
|
assertTrue(Files.isDirectory(d2), d2 + " directory is missing or not a directory");
|
||
|
final Path d3 = Path.of(dest, "d1", "d2", "d3");
|
||
|
assertTrue(Files.isDirectory(d3), d3 + " directory is missing or not a directory");
|
||
|
final Path d4 = Path.of(dest, "d1", "d4");
|
||
|
assertTrue(Files.isDirectory(d4), d4 + " directory is missing or not a directory");
|
||
|
// d1/d4 is expected to be empty directory
|
||
|
final List<Path> d4Children;
|
||
|
try (final Stream<Path> s = Files.walk(d4, 1)) {
|
||
|
d4Children = s.toList();
|
||
|
}
|
||
|
assertEquals(1, d4Children.size(), "Directory " + d4
|
||
|
+ " has unexpected files/dirs: " + d4Children);
|
||
|
final Path f1 = Path.of(dest, "f1.txt");
|
||
|
assertTrue(Files.isRegularFile(f1), f1 + " is missing or not a file");
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f1), "Unexpected content in file " + f1);
|
||
|
final Path f2 = Path.of(d3.toString(), "f2.txt");
|
||
|
assertTrue(Files.isRegularFile(f2), f2 + " is missing or not a file");
|
||
|
assertArrayEquals(FILE_CONTENT, Files.readAllBytes(f2), "Unexpected content in file " + f2);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Creates a jar whose entries have a leading slash and the dot-dot character preserved.
|
||
|
* This is the same as creating a jar using {@code jar -cfP somejar.jar <file1> <file2> ...}
|
||
|
*/
|
||
|
private static Path createJarWithPFlagSemantics(String leadingSlashEntryName)
|
||
|
throws IOException {
|
||
|
final Path tmpDir = Files.createTempDirectory(Path.of("."), "8173970-").toAbsolutePath();
|
||
|
final Path jarPath = tmpDir.resolve("8173970-test-withpflag.jar");
|
||
|
final JarBuilder builder = new JarBuilder(jarPath.toString());
|
||
|
builder.addEntry("d1/", new byte[0]);
|
||
|
builder.addEntry("d1/d2/", new byte[0]);
|
||
|
builder.addEntry(leadingSlashEntryName, FILE_CONTENT);
|
||
|
builder.addEntry("d1/d2/../f2.txt", FILE_CONTENT);
|
||
|
builder.build();
|
||
|
return jarPath;
|
||
|
}
|
||
|
|
||
|
private static void printJarCommand(final String[] cmdArgs) {
|
||
|
System.out.println("Running 'jar " + String.join(" ", cmdArgs) + "'");
|
||
|
}
|
||
|
}
|