Skip to content

Commit 725079b

Browse files
committedDec 17, 2024
8345506: jar --validate may lead to java.nio.file.FileAlreadyExistsException
Reviewed-by: lancea
1 parent 5e25c48 commit 725079b

File tree

2 files changed

+169
-2
lines changed

2 files changed

+169
-2
lines changed
 

‎src/jdk.jartool/share/classes/sun/tools/jar/Main.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -420,8 +420,9 @@ public synchronized boolean run(String[] args) {
420420
file = new File(fname);
421421
} else {
422422
file = createTemporaryFile("tmpJar", ".jar");
423-
try (InputStream in = new FileInputStream(FileDescriptor.in)) {
424-
Files.copy(in, file.toPath());
423+
try (InputStream in = new FileInputStream(FileDescriptor.in);
424+
OutputStream os = Files.newOutputStream(file.toPath())) {
425+
in.transferTo(os);
425426
}
426427
}
427428
ok = validateJar(file);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,166 @@
1+
/*
2+
* Copyright (c) 2024, 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 java.io.IOException;
25+
import java.io.OutputStream;
26+
import java.nio.file.Files;
27+
import java.nio.file.Path;
28+
import java.util.jar.JarFile;
29+
import java.util.jar.JarOutputStream;
30+
import java.util.jar.Manifest;
31+
import java.util.zip.ZipEntry;
32+
33+
import jdk.test.lib.JDKToolFinder;
34+
import jdk.test.lib.process.OutputAnalyzer;
35+
import jdk.test.lib.process.ProcessTools;
36+
import org.junit.jupiter.api.BeforeAll;
37+
import org.junit.jupiter.api.Test;
38+
import static java.nio.charset.StandardCharsets.US_ASCII;
39+
import static org.junit.jupiter.api.Assertions.assertTrue;
40+
41+
/*
42+
* @test
43+
* @bug 8345506
44+
* @summary verifies that the "jar" operations that are expected to work
45+
* without the "--file" option work as expected
46+
* @library /test/lib
47+
* @run junit JarNoFileArgOperations
48+
*/
49+
public class JarNoFileArgOperations {
50+
51+
private static final Path SCRATCH_DIR = Path.of(".");
52+
private static final Path JAR_TOOL = Path.of(JDKToolFinder.getJDKTool("jar")).toAbsolutePath();
53+
private static final String JAR_ENTRY_NAME = "foobarhello.txt";
54+
55+
private static Path SIMPLE_JAR;
56+
57+
private static void makeSimpleJar(final Path path) throws IOException {
58+
final Manifest manifest = new Manifest();
59+
manifest.getMainAttributes().putValue("Manifest-Version", "1.0");
60+
try (OutputStream fos = Files.newOutputStream(path);
61+
JarOutputStream jaros = new JarOutputStream(fos, manifest)) {
62+
jaros.putNextEntry(new ZipEntry(JAR_ENTRY_NAME));
63+
jaros.write("foobar-8345506".getBytes(US_ASCII));
64+
jaros.closeEntry();
65+
}
66+
}
67+
68+
@BeforeAll
69+
static void beforeAll() throws Exception {
70+
final Path jarFile = Files.createTempFile(SCRATCH_DIR, "8345506", ".jar");
71+
makeSimpleJar(jarFile);
72+
SIMPLE_JAR = jarFile;
73+
System.out.println("created JAR file " + jarFile);
74+
}
75+
76+
/*
77+
* Launches "jar --validate" by streaming the JAR file content through the "jar" tool
78+
* process' STDIN and expects that the command completes normally.
79+
*/
80+
@Test
81+
public void testValidate() throws Exception {
82+
System.out.println("launching jar --validate");
83+
final ProcessBuilder pb = new ProcessBuilder()
84+
.command(JAR_TOOL.toString(), "--validate")
85+
// stream the JAR file content to the jar command through the process' STDIN
86+
.redirectInput(SIMPLE_JAR.toFile());
87+
final OutputAnalyzer oa = ProcessTools.executeCommand(pb);
88+
oa.shouldHaveExitValue(0);
89+
}
90+
91+
/*
92+
* Launches "jar --list" and "jar -t" by streaming the JAR file content through the "jar" tool
93+
* process' STDIN and expects that the command completes normally.
94+
*/
95+
@Test
96+
public void testListing() throws Exception {
97+
for (String opt : new String[]{"-t", "--list"}) {
98+
final ProcessBuilder pb = new ProcessBuilder()
99+
.command(JAR_TOOL.toString(), opt)
100+
// stream the JAR file content to the jar command through the process' STDIN
101+
.redirectInput(SIMPLE_JAR.toFile());
102+
final OutputAnalyzer oa = ProcessTools.executeCommand(pb);
103+
oa.shouldHaveExitValue(0);
104+
// verify the listing contained the JAR entry name
105+
oa.contains(JAR_ENTRY_NAME);
106+
}
107+
}
108+
109+
/*
110+
* Launches "jar --extract" and "jar -x" by streaming the JAR file content through
111+
* the "jar" tool process' STDIN and expects that the command completes normally.
112+
*/
113+
@Test
114+
public void testExtract() throws Exception {
115+
for (String opt : new String[]{"-x", "--extract"}) {
116+
final Path tmpDestDir = Files.createTempDirectory(SCRATCH_DIR, "8345506");
117+
final ProcessBuilder pb = new ProcessBuilder()
118+
.command(JAR_TOOL.toString(), opt, "--dir", tmpDestDir.toString())
119+
// stream the JAR file content to the jar command through the process' STDIN
120+
.redirectInput(SIMPLE_JAR.toFile());
121+
final OutputAnalyzer oa = ProcessTools.executeCommand(pb);
122+
oa.shouldHaveExitValue(0);
123+
// verify the file content was extracted
124+
assertTrue(Files.exists(tmpDestDir.resolve(JAR_ENTRY_NAME)),
125+
JAR_ENTRY_NAME + " wasn't extracted to " + tmpDestDir);
126+
}
127+
}
128+
129+
/*
130+
* Launches "jar --update" and "jar -u" by streaming the JAR file content through
131+
* the "jar" tool process' STDIN and expects that the command completes normally.
132+
*/
133+
@Test
134+
public void testUpdate() throws Exception {
135+
for (String opt : new String[]{"-u", "--update"}) {
136+
// the updated JAR will be written out to this file
137+
final Path destUpdatedJar = Files.createTempFile(SCRATCH_DIR, "8345506", ".jar");
138+
// an arbitrary file that will be added to the JAR file as
139+
// part of the update operation
140+
final Path fileToAdd = Files.createTempFile(SCRATCH_DIR, "8345506", ".txt");
141+
final String expectedNewEntry = fileToAdd.getFileName().toString();
142+
final ProcessBuilder pb = new ProcessBuilder()
143+
.command(JAR_TOOL.toString(), opt, expectedNewEntry)
144+
// stream the JAR file content to the jar command through the process' STDIN
145+
.redirectInput(SIMPLE_JAR.toFile())
146+
// redirect the updated JAR to a file so that its contents can be verified
147+
// later
148+
.redirectOutput(destUpdatedJar.toFile());
149+
final OutputAnalyzer oa = ProcessTools.executeProcess(pb);
150+
oa.shouldHaveExitValue(0);
151+
System.out.println("updated JAR file at " + destUpdatedJar.toAbsolutePath());
152+
// verify, by listing the updated JAR file contents,
153+
// that the JAR file has been updated to include the new file
154+
try (final JarFile jar = new JarFile(destUpdatedJar.toFile())) {
155+
jar.stream()
156+
.map(ZipEntry::getName)
157+
.filter((name) -> name.equals(expectedNewEntry))
158+
.findFirst()
159+
.orElseThrow(
160+
() -> new AssertionError("missing entry " + expectedNewEntry
161+
+ " in updated JAR file " + destUpdatedJar)
162+
);
163+
}
164+
}
165+
}
166+
}

0 commit comments

Comments
 (0)
Please sign in to comment.