7700b25460
Reviewed-by: dfuchs, lancea, alanb, mchung
217 lines
8.0 KiB
Java
217 lines
8.0 KiB
Java
/*
|
|
* Copyright (c) 2012, 2021, 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 6901992
|
|
* @summary InvalidJarIndexException due to bug in sun.misc.JarIndex.merge()
|
|
* Test URLClassLoader usage of the merge method when using indexes
|
|
* @author Diego Belfer
|
|
* @run main/othervm -Djdk.net.URLClassPath.enableJarIndex=true JarIndexMergeForClassLoaderTest
|
|
*/
|
|
import java.io.BufferedReader;
|
|
import java.io.File;
|
|
import java.io.FileNotFoundException;
|
|
import java.io.FileOutputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.io.InputStreamReader;
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.util.jar.JarEntry;
|
|
import java.util.jar.JarOutputStream;
|
|
|
|
public class JarIndexMergeForClassLoaderTest {
|
|
static final String slash = File.separator;
|
|
static final String testClassesDir = System.getProperty("test.classes", ".");
|
|
static final String jar;
|
|
static final boolean debug = true;
|
|
static final File tmpFolder = new File(testClassesDir);
|
|
|
|
static {
|
|
jar = System.getProperty("java.home") + slash + "bin" + slash + "jar";
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
// Create the jars file
|
|
File jar1 = buildJar1();
|
|
File jar2 = buildJar2();
|
|
File jar3 = buildJar3();
|
|
|
|
// Index jar files in two levels: jar1 -> jar2 -> jar3
|
|
createIndex(jar2.getName(), jar3.getName());
|
|
createIndex(jar1.getName(), jar2.getName());
|
|
|
|
// Get root jar of the URLClassLoader
|
|
URL url = jar1.toURI().toURL();
|
|
|
|
URLClassLoader classLoader = new URLClassLoader(new URL[] { url });
|
|
|
|
assertResource(classLoader, "com/jar1/resource.file", "jar1");
|
|
assertResource(classLoader, "com/test/resource1.file", "resource1");
|
|
assertResource(classLoader, "com/jar2/resource.file", "jar2");
|
|
assertResource(classLoader, "com/test/resource2.file", "resource2");
|
|
assertResource(classLoader, "com/test/resource3.file", "resource3");
|
|
|
|
/*
|
|
* The following two asserts failed before the fix of the bug 6901992
|
|
*/
|
|
// Check that an existing file is found using the merged index
|
|
assertResource(classLoader, "com/missing/jar3/resource.file", "jar3");
|
|
// Check that a non existent file in directory which does not contain
|
|
// any file is not found and it does not throw InvalidJarIndexException
|
|
assertResource(classLoader, "com/missing/nofile", null);
|
|
}
|
|
|
|
private static File buildJar3() throws FileNotFoundException, IOException {
|
|
JarBuilder jar3Builder = new JarBuilder(tmpFolder, "jar3.jar");
|
|
jar3Builder.addResourceFile("com/test/resource3.file", "resource3");
|
|
jar3Builder.addResourceFile("com/missing/jar3/resource.file", "jar3");
|
|
return jar3Builder.build();
|
|
}
|
|
|
|
private static File buildJar2() throws FileNotFoundException, IOException {
|
|
JarBuilder jar2Builder = new JarBuilder(tmpFolder, "jar2.jar");
|
|
jar2Builder.addResourceFile("com/jar2/resource.file", "jar2");
|
|
jar2Builder.addResourceFile("com/test/resource2.file", "resource2");
|
|
return jar2Builder.build();
|
|
}
|
|
|
|
private static File buildJar1() throws FileNotFoundException, IOException {
|
|
JarBuilder jar1Builder = new JarBuilder(tmpFolder, "jar1.jar");
|
|
jar1Builder.addResourceFile("com/jar1/resource.file", "jar1");
|
|
jar1Builder.addResourceFile("com/test/resource1.file", "resource1");
|
|
return jar1Builder.build();
|
|
}
|
|
|
|
/* create the index */
|
|
static void createIndex(String parentJar, String childJar) {
|
|
// ProcessBuilder is used so that the current directory can be set
|
|
// to the directory that directly contains the jars.
|
|
debug("Running jar to create the index for: " + parentJar + " and "
|
|
+ childJar);
|
|
ProcessBuilder pb = new ProcessBuilder(jar, "-i", parentJar, childJar);
|
|
|
|
pb.directory(tmpFolder);
|
|
// pd.inheritIO();
|
|
try {
|
|
Process p = pb.start();
|
|
if (p.waitFor() != 0)
|
|
throw new RuntimeException("jar indexing failed");
|
|
|
|
if (debug && p != null) {
|
|
debugStream(p.getInputStream());
|
|
debugStream(p.getErrorStream());
|
|
}
|
|
} catch (InterruptedException | IOException x) {
|
|
throw new RuntimeException(x);
|
|
}
|
|
}
|
|
|
|
private static void debugStream(InputStream is) throws IOException {
|
|
try (BufferedReader reader = new BufferedReader(new InputStreamReader(is))) {
|
|
String line;
|
|
while ((line = reader.readLine()) != null) {
|
|
debug(line);
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void assertResource(URLClassLoader classLoader, String file,
|
|
String expectedContent) throws IOException {
|
|
InputStream fileStream = classLoader.getResourceAsStream(file);
|
|
|
|
if (fileStream == null && expectedContent == null) {
|
|
return;
|
|
}
|
|
if (fileStream == null && expectedContent != null) {
|
|
throw new RuntimeException(
|
|
buildMessage(file, expectedContent, null));
|
|
}
|
|
try {
|
|
String actualContent = readAsString(fileStream);
|
|
|
|
if (fileStream != null && expectedContent == null) {
|
|
throw new RuntimeException(buildMessage(file, null,
|
|
actualContent));
|
|
}
|
|
if (!expectedContent.equals(actualContent)) {
|
|
throw new RuntimeException(buildMessage(file, expectedContent,
|
|
actualContent));
|
|
}
|
|
} finally {
|
|
fileStream.close();
|
|
}
|
|
}
|
|
|
|
private static String buildMessage(String file, String expectedContent,
|
|
String actualContent) {
|
|
return "Expected: " + expectedContent + " for: " + file + " was: "
|
|
+ actualContent;
|
|
}
|
|
|
|
private static String readAsString(InputStream fileStream)
|
|
throws IOException {
|
|
byte[] buffer = new byte[1024];
|
|
int count, len = 0;
|
|
while ((count = fileStream.read(buffer, len, buffer.length-len)) != -1)
|
|
len += count;
|
|
return new String(buffer, 0, len, "ASCII");
|
|
}
|
|
|
|
static void debug(Object message) {
|
|
if (debug)
|
|
System.out.println(message);
|
|
}
|
|
|
|
/*
|
|
* Helper class for building jar files
|
|
*/
|
|
public static class JarBuilder {
|
|
private JarOutputStream os;
|
|
private File jarFile;
|
|
|
|
public JarBuilder(File tmpFolder, String jarName)
|
|
throws FileNotFoundException, IOException
|
|
{
|
|
this.jarFile = new File(tmpFolder, jarName);
|
|
this.os = new JarOutputStream(new FileOutputStream(jarFile));
|
|
}
|
|
|
|
public void addResourceFile(String pathFromRoot, String content)
|
|
throws IOException
|
|
{
|
|
JarEntry entry = new JarEntry(pathFromRoot);
|
|
os.putNextEntry(entry);
|
|
os.write(content.getBytes("ASCII"));
|
|
os.closeEntry();
|
|
}
|
|
|
|
public File build() throws IOException {
|
|
os.close();
|
|
return jarFile;
|
|
}
|
|
}
|
|
}
|
|
|