/* * Copyright (C) 2012 The Guava Authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.google.common.reflect; import static org.truth0.Truth.ASSERT; import com.google.common.base.Charsets; import com.google.common.collect.ImmutableMap; import com.google.common.collect.Maps; import com.google.common.collect.Sets; import com.google.common.io.Closer; import com.google.common.io.Resources; import com.google.common.reflect.ClassPath.ClassInfo; import com.google.common.reflect.ClassPath.ResourceInfo; import com.google.common.reflect.subpackage.ClassInSubPackage; import com.google.common.testing.EqualsTester; import com.google.common.testing.NullPointerTester; import junit.framework.TestCase; import org.junit.Test; import java.io.ByteArrayInputStream; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.net.URI; import java.net.URISyntaxException; import java.net.URL; import java.net.URLClassLoader; import java.util.Map; import java.util.Set; import java.util.jar.Attributes; import java.util.jar.JarFile; import java.util.jar.JarOutputStream; import java.util.jar.Manifest; import java.util.zip.ZipEntry; /** * Functional tests of {@link ClassPath}. */ public class ClassPathTest extends TestCase { public void testGetResources() throws Exception { Map<String, ResourceInfo> byName = Maps.newHashMap(); Map<String, ResourceInfo> byToString = Maps.newHashMap(); ClassPath classpath = ClassPath.from(getClass().getClassLoader()); for (ResourceInfo resource : classpath.getResources()) { ASSERT.that(resource.getResourceName()).isNotEqualTo(JarFile.MANIFEST_NAME); ASSERT.that(resource.toString()).isNotEqualTo(JarFile.MANIFEST_NAME); byName.put(resource.getResourceName(), resource); byToString.put(resource.toString(), resource); // TODO: This will fail on maven resources in the classes directory on a mac. // assertNotNull(resource.url()); } String testResourceName = "com/google/common/reflect/test.txt"; ASSERT.that(byName.keySet()).has().allOf( "com/google/common/reflect/ClassPath.class", "com/google/common/reflect/ClassPathTest.class", "com/google/common/reflect/ClassPathTest$Nested.class", testResourceName); ASSERT.that(byToString.keySet()).has().allOf( "com.google.common.reflect.ClassPath", "com.google.common.reflect.ClassPathTest", "com.google.common.reflect.ClassPathTest$Nested", testResourceName); assertEquals(getClass().getClassLoader().getResource(testResourceName), byName.get("com/google/common/reflect/test.txt").url()); } public void testGetAllClasses() throws Exception { Set<String> names = Sets.newHashSet(); Set<String> strings = Sets.newHashSet(); Set<Class<?>> classes = Sets.newHashSet(); Set<String> packageNames = Sets.newHashSet(); Set<String> simpleNames = Sets.newHashSet(); ClassPath classpath = ClassPath.from(getClass().getClassLoader()); for (ClassInfo classInfo : classpath.getAllClasses()) { if (!classInfo.getPackageName().equals(ClassPathTest.class.getPackage().getName())) { continue; } names.add(classInfo.getName()); strings.add(classInfo.toString()); classes.add(classInfo.load()); packageNames.add(classInfo.getPackageName()); simpleNames.add(classInfo.getSimpleName()); } class LocalClass {} Class<?> anonymousClass = new Object() {}.getClass(); ASSERT.that(names).has().allOf(anonymousClass.getName(), LocalClass.class.getName(), ClassPath.class.getName(), ClassPathTest.class.getName()); ASSERT.that(strings).has().allOf(anonymousClass.getName(), LocalClass.class.getName(), ClassPath.class.getName(), ClassPathTest.class.getName()); ASSERT.that(classes).has().allOf(anonymousClass, LocalClass.class, ClassPath.class, ClassPathTest.class); ASSERT.that(packageNames).has().exactly(ClassPath.class.getPackage().getName()); ASSERT.that(simpleNames).has().allOf("", "Local", "ClassPath", "ClassPathTest"); } public void testGetTopLevelClasses() throws Exception { Set<String> names = Sets.newHashSet(); Set<String> strings = Sets.newHashSet(); Set<Class<?>> classes = Sets.newHashSet(); Set<String> packageNames = Sets.newHashSet(); Set<String> simpleNames = Sets.newHashSet(); ClassPath classpath = ClassPath.from(getClass().getClassLoader()); for (ClassInfo classInfo : classpath.getTopLevelClasses(ClassPathTest.class.getPackage().getName())) { names.add(classInfo.getName()); strings.add(classInfo.toString()); classes.add(classInfo.load()); packageNames.add(classInfo.getPackageName()); simpleNames.add(classInfo.getSimpleName()); } ASSERT.that(names).has().allOf(ClassPath.class.getName(), ClassPathTest.class.getName()); ASSERT.that(strings).has().allOf(ClassPath.class.getName(), ClassPathTest.class.getName()); ASSERT.that(classes).has().allOf(ClassPath.class, ClassPathTest.class); ASSERT.that(packageNames).has().item(ClassPath.class.getPackage().getName()); ASSERT.that(simpleNames).has().allOf("ClassPath", "ClassPathTest"); assertFalse(classes.contains(ClassInSubPackage.class)); } public void testGetTopLevelClassesRecursive() throws Exception { Set<Class<?>> classes = Sets.newHashSet(); ClassPath classpath = ClassPath.from(ClassPathTest.class.getClassLoader()); for (ClassInfo classInfo : classpath.getTopLevelClassesRecursive(ClassPathTest.class.getPackage().getName())) { if (classInfo.getName().contains("ClassPathTest")) { System.err.println(""); } classes.add(classInfo.load()); } ASSERT.that(classes).has().allOf(ClassPathTest.class, ClassInSubPackage.class); } public void testGetTopLevelClasses_diamond() throws Exception { ClassLoader parent = ClassPathTest.class.getClassLoader(); ClassLoader sub1 = new ClassLoader(parent) {}; ClassLoader sub2 = new ClassLoader(parent) {}; assertEquals(findClass(ClassPath.from(sub1).getTopLevelClasses(), ClassPathTest.class), findClass(ClassPath.from(sub2).getTopLevelClasses(), ClassPathTest.class)); } public void testEquals() { new EqualsTester() .addEqualityGroup(classInfo(ClassPathTest.class), classInfo(ClassPathTest.class)) .addEqualityGroup(classInfo(Test.class), classInfo(Test.class, getClass().getClassLoader())) .addEqualityGroup( new ResourceInfo("a/b/c.txt", getClass().getClassLoader()), new ResourceInfo("a/b/c.txt", getClass().getClassLoader())) .addEqualityGroup( new ResourceInfo("x.txt", getClass().getClassLoader())) .testEquals(); } public void testClassPathEntries_emptyURLClassLoader_noParent() { ASSERT.that(ClassPath.getClassPathEntries(new URLClassLoader(new URL[0], null)).keySet()) .isEmpty(); } public void testClassPathEntries_URLClassLoader_noParent() throws Exception { URL url1 = new URL("file:/a"); URL url2 = new URL("file:/b"); URLClassLoader classloader = new URLClassLoader(new URL[] {url1, url2}, null); assertEquals( ImmutableMap.of(url1.toURI(), classloader, url2.toURI(), classloader), ClassPath.getClassPathEntries(classloader)); } public void testClassPathEntries_URLClassLoader_withParent() throws Exception { URL url1 = new URL("file:/a"); URL url2 = new URL("file:/b"); URLClassLoader parent = new URLClassLoader(new URL[] {url1}, null); URLClassLoader child = new URLClassLoader(new URL[] {url2}, parent) {}; ImmutableMap<URI, ClassLoader> classPathEntries = ClassPath.getClassPathEntries(child); assertEquals(ImmutableMap.of(url1.toURI(), parent, url2.toURI(), child), classPathEntries); ASSERT.that(classPathEntries.keySet()).has().exactly(url1.toURI(), url2.toURI()).inOrder(); } public void testClassPathEntries_duplicateUri_parentWins() throws Exception { URL url = new URL("file:/a"); URLClassLoader parent = new URLClassLoader(new URL[] {url}, null); URLClassLoader child = new URLClassLoader(new URL[] {url}, parent) {}; assertEquals(ImmutableMap.of(url.toURI(), parent), ClassPath.getClassPathEntries(child)); } public void testClassPathEntries_notURLClassLoader_noParent() { ASSERT.that(ClassPath.getClassPathEntries(new ClassLoader(null) {}).keySet()).isEmpty(); } public void testClassPathEntries_notURLClassLoader_withParent() throws Exception { URL url = new URL("file:/a"); URLClassLoader parent = new URLClassLoader(new URL[] {url}, null); assertEquals( ImmutableMap.of(url.toURI(), parent), ClassPath.getClassPathEntries(new ClassLoader(parent) {})); } public void testClassPathEntries_notURLClassLoader_withParentAndGrandParent() throws Exception { URL url1 = new URL("file:/a"); URL url2 = new URL("file:/b"); URLClassLoader grandParent = new URLClassLoader(new URL[] {url1}, null); URLClassLoader parent = new URLClassLoader(new URL[] {url2}, grandParent); assertEquals( ImmutableMap.of(url1.toURI(), grandParent, url2.toURI(), parent), ClassPath.getClassPathEntries(new ClassLoader(parent) {})); } public void testClassPathEntries_notURLClassLoader_withGrandParent() throws Exception { URL url = new URL("file:/a"); URLClassLoader grandParent = new URLClassLoader(new URL[] {url}, null); ClassLoader parent = new ClassLoader(grandParent) {}; assertEquals( ImmutableMap.of(url.toURI(), grandParent), ClassPath.getClassPathEntries(new ClassLoader(parent) {})); } public void testScan_classPathCycle() throws IOException { File jarFile = File.createTempFile("with_circular_class_path", ".jar"); try { writeSelfReferencingJarFile(jarFile, "test.txt"); ClassPath.Scanner scanner = new ClassPath.Scanner(); scanner.scan(jarFile.toURI(), ClassPathTest.class.getClassLoader()); assertEquals(1, scanner.getResources().size()); } finally { jarFile.delete(); } } public void testScanFromFile_fileNotExists() throws IOException { ClassLoader classLoader = ClassPathTest.class.getClassLoader(); ClassPath.Scanner scanner = new ClassPath.Scanner(); scanner.scanFrom(new File("no/such/file/anywhere"), classLoader); ASSERT.that(scanner.getResources()).isEmpty(); } public void testScanFromFile_notJarFile() throws IOException { ClassLoader classLoader = ClassPathTest.class.getClassLoader(); File notJar = File.createTempFile("not_a_jar", "txt"); ClassPath.Scanner scanner = new ClassPath.Scanner(); try { scanner.scanFrom(notJar, classLoader); } finally { notJar.delete(); } ASSERT.that(scanner.getResources()).isEmpty(); } public void testGetClassPathEntry() throws URISyntaxException { assertEquals(URI.create("file:/usr/test/dep.jar"), ClassPath.Scanner.getClassPathEntry( new File("/home/build/outer.jar"), "file:/usr/test/dep.jar")); assertEquals(URI.create("file:/home/build/a.jar"), ClassPath.Scanner.getClassPathEntry(new File("/home/build/outer.jar"), "a.jar")); assertEquals(URI.create("file:/home/build/x/y/z"), ClassPath.Scanner.getClassPathEntry(new File("/home/build/outer.jar"), "x/y/z")); assertEquals(URI.create("file:/home/build/x/y/z.jar"), ClassPath.Scanner.getClassPathEntry(new File("/home/build/outer.jar"), "x/y/z.jar")); } public void testGetClassPathFromManifest_nullManifest() { ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(new File("some.jar"), null)).isEmpty(); } public void testGetClassPathFromManifest_noClassPath() throws IOException { File jarFile = new File("base.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest(""))) .isEmpty(); } public void testGetClassPathFromManifest_emptyClassPath() throws IOException { File jarFile = new File("base.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifestClasspath(""))) .isEmpty(); } public void testGetClassPathFromManifest_badClassPath() throws IOException { File jarFile = new File("base.jar"); Manifest manifest = manifestClasspath("an_invalid^path"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .isEmpty(); } public void testGetClassPathFromManifest_relativeDirectory() throws IOException { File jarFile = new File("base/some.jar"); // with/relative/directory is the Class-Path value in the mf file. Manifest manifest = manifestClasspath("with/relative/dir"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("base/with/relative/dir").toURI()).inOrder(); } public void testGetClassPathFromManifest_relativeJar() throws IOException { File jarFile = new File("base/some.jar"); // with/relative/directory is the Class-Path value in the mf file. Manifest manifest = manifestClasspath("with/relative.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("base/with/relative.jar").toURI()).inOrder(); } public void testGetClassPathFromManifest_jarInCurrentDirectory() throws IOException { File jarFile = new File("base/some.jar"); // with/relative/directory is the Class-Path value in the mf file. Manifest manifest = manifestClasspath("current.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("base/current.jar").toURI()).inOrder(); } public void testGetClassPathFromManifest_absoluteDirectory() throws IOException { File jarFile = new File("base/some.jar"); Manifest manifest = manifestClasspath("file:/with/absolute/dir"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("/with/absolute/dir").toURI()).inOrder(); } public void testGetClassPathFromManifest_absoluteJar() throws IOException { File jarFile = new File("base/some.jar"); Manifest manifest = manifestClasspath("file:/with/absolute.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("/with/absolute.jar").toURI()).inOrder(); } public void testGetClassPathFromManifest_multiplePaths() throws IOException { File jarFile = new File("base/some.jar"); Manifest manifest = manifestClasspath("file:/with/absolute.jar relative.jar relative/dir"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly( new File("/with/absolute.jar").toURI(), new File("base/relative.jar").toURI(), new File("base/relative/dir").toURI()) .inOrder(); } public void testGetClassPathFromManifest_leadingBlanks() throws IOException { File jarFile = new File("base/some.jar"); Manifest manifest = manifestClasspath(" relative.jar"); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("base/relative.jar").toURI()).inOrder(); } public void testGetClassPathFromManifest_trailingBlanks() throws IOException { File jarFile = new File("base/some.jar"); Manifest manifest = manifestClasspath("relative.jar "); ASSERT.that(ClassPath.Scanner.getClassPathFromManifest(jarFile, manifest)) .has().exactly(new File("base/relative.jar").toURI()).inOrder(); } public void testGetClassName() { assertEquals("abc.d.Abc", ClassPath.getClassName("abc/d/Abc.class")); } public void testResourceInfo_of() { assertEquals(ClassInfo.class, resourceInfo(ClassPathTest.class).getClass()); assertEquals(ClassInfo.class, resourceInfo(ClassPath.class).getClass()); assertEquals(ClassInfo.class, resourceInfo(Nested.class).getClass()); } public void testGetSimpleName() { assertEquals("Foo", new ClassInfo("Foo.class", getClass().getClassLoader()).getSimpleName()); assertEquals("Foo", new ClassInfo("a/b/Foo.class", getClass().getClassLoader()).getSimpleName()); assertEquals("Foo", new ClassInfo("a/b/Bar$Foo.class", getClass().getClassLoader()).getSimpleName()); assertEquals("", new ClassInfo("a/b/Bar$1.class", getClass().getClassLoader()).getSimpleName()); assertEquals("Foo", new ClassInfo("a/b/Bar$Foo.class", getClass().getClassLoader()).getSimpleName()); assertEquals("", new ClassInfo("a/b/Bar$1.class", getClass().getClassLoader()).getSimpleName()); assertEquals("Local", new ClassInfo("a/b/Bar$1Local.class", getClass().getClassLoader()).getSimpleName()); } public void testGetPackageName() { assertEquals("", new ClassInfo("Foo.class", getClass().getClassLoader()).getPackageName()); assertEquals("a.b", new ClassInfo("a/b/Foo.class", getClass().getClassLoader()).getPackageName()); } private static class Nested {} public void testNulls() throws IOException { new NullPointerTester().testAllPublicStaticMethods(ClassPath.class); new NullPointerTester() .testAllPublicInstanceMethods(ClassPath.from(getClass().getClassLoader())); } private static ClassPath.ClassInfo findClass( Iterable<ClassPath.ClassInfo> classes, Class<?> cls) { for (ClassPath.ClassInfo classInfo : classes) { if (classInfo.getName().equals(cls.getName())) { return classInfo; } } throw new AssertionError("failed to find " + cls); } private static ResourceInfo resourceInfo(Class<?> cls) { return ResourceInfo.of(cls.getName().replace('.', '/') + ".class", cls.getClassLoader()); } private static ClassInfo classInfo(Class<?> cls) { return classInfo(cls, cls.getClassLoader()); } private static ClassInfo classInfo(Class<?> cls, ClassLoader classLoader) { return new ClassInfo(cls.getName().replace('.', '/') + ".class", classLoader); } private static Manifest manifestClasspath(String classpath) throws IOException { return manifest("Class-Path: " + classpath + "\n"); } private static void writeSelfReferencingJarFile(File jarFile, String... entries) throws IOException { Manifest manifest = new Manifest(); // Without version, the manifest is silently ignored. Ugh! manifest.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0"); manifest.getMainAttributes().put(Attributes.Name.CLASS_PATH, jarFile.getName()); Closer closer = Closer.create(); try { FileOutputStream fileOut = closer.register(new FileOutputStream(jarFile)); JarOutputStream jarOut = closer.register(new JarOutputStream(fileOut)); for (String entry : entries) { jarOut.putNextEntry(new ZipEntry(entry)); Resources.copy(ClassPathTest.class.getResource(entry), jarOut); jarOut.closeEntry(); } } catch (Throwable e) { throw closer.rethrow(e); } finally { closer.close(); } } private static Manifest manifest(String content) throws IOException { InputStream in = new ByteArrayInputStream(content.getBytes(Charsets.US_ASCII.name())); Manifest manifest = new Manifest(); manifest.read(in); return manifest; } }