Files
apache-tomcat-8.5.51/test/javax/el/TestImportHandlerStandardPackages.java
2024-11-30 19:03:49 +08:00

180 lines
7.5 KiB
Java

/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 javax.el;
import java.io.File;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.net.URI;
import java.net.URL;
import java.util.Enumeration;
import java.util.Map;
import java.util.Set;
import org.junit.Assert;
import org.junit.Test;
import org.apache.tomcat.util.compat.JreCompat;
public class TestImportHandlerStandardPackages {
@Test
public void testClassListsAreComplete() throws Exception {
// Use reflection to get hold of the internal Map
Class<?> clazz = ImportHandler.class;
Field f = clazz.getDeclaredField("standardPackages");
f.setAccessible(true);
Object obj = f.get(null);
Assert.assertTrue("Not Map", obj instanceof Map);
@SuppressWarnings("unchecked")
Map<String,Set<String>> standardPackageName = (Map<String, Set<String>>) obj;
for (Map.Entry<String,Set<String>> entry : standardPackageName.entrySet()) {
checkPackageClassList(entry.getKey(), entry.getValue());
}
}
private void checkPackageClassList(String packageName, Set<String> classNames) throws Exception {
if ("java.lang".equals(packageName)) {
// The code below is designed to run on Java 9 so skip this check
// if running on Java 8. The test has previously been run with Java
// 9 (and later) so it is not necessary that this is executed on
// every test run. The intention is that it will catch new classes
// when the tests are run on a newer JRE.
// The latest version of the JRE where this test is known to pass is
// - OpenJDK 14 EA 27
if (!JreCompat.isJre9Available()) {
return;
}
for (String fileName : getJavaBaseClasses()) {
if (!fileName.startsWith("java/lang/") || // Class not in java.lang
fileName.lastIndexOf('/') != 9 || // Class no in sub-package
!fileName.endsWith(".class")) { // Exclude non-class resources
continue;
}
// Extract class name
String className = fileName.substring(10, fileName.length() - 6);
Class<?> clazz = Class.forName("java.lang." + className);
if (!Modifier.isPublic(clazz.getModifiers())) {
// Exclude non-public classes
continue;
}
className = className.replace('$', '.');
if (classNames.contains(className)) {
// Already listed
continue;
}
// Skip public inner classes of non-public classes
if (className.startsWith("FdLibm.") ||
className.startsWith("LiveStackFrame.") ||
className.startsWith("WeakPairMap.")) {
continue;
}
// Anything left at this point is an error
Assert.fail(fileName);
}
} else {
// When this test runs, the class loader will be loading resources
// from a directory for each of these packages.
ClassLoader cl = Thread.currentThread().getContextClassLoader();
String path = packageName.replace('.', '/');
Enumeration<URL> resources = cl.getResources(path);
while (resources.hasMoreElements()) {
URL resource = resources.nextElement();
URI uri = resource.toURI();
// Gump includes some JARs on classpath - skip them
if (!"file".equals(uri.getScheme())) {
continue;
}
File dir = new File(uri);
String[] files = dir.list();
Assert.assertNotNull(files);
for (String file : files) {
if (!file.endsWith(".class")) {
// Skip non-class resoucres
continue;
}
if (file.startsWith("Test")) {
// Skip test resources
continue;
}
if (file.matches(".*\\$[0-9]?\\.class")) {
// Skip anonymous inner classes
continue;
}
String name = file.substring(0, file.length() - 6);
name = name.replace('$', '.');
if (classNames.contains(name)) {
// Skip classes already known
continue;
}
File f = new File (dir, file);
if (!f.isFile()) {
// Skip directories
continue;
}
Class<?> clazz = Class.forName(packageName + "." + name);
if (!Modifier.isPublic(clazz.getModifiers())) {
// Skip non-public classes
continue;
}
// There should be nothing left unless we missed something
Assert.fail(packageName + "." + name);
}
}
}
}
private static String[] getJavaBaseClasses() throws Exception {
// While this code is only used on Java 9 and later, it needs to compile
// with Java 8 so use reflection for now.
Class<?> clazzModuleFinder = Class.forName("java.lang.module.ModuleFinder");
Class<?> clazzModuleReference = Class.forName("java.lang.module.ModuleReference");
Class<?> clazzOptional = Class.forName("java.util.Optional");
Class<?> clazzModuleReader = Class.forName("java.lang.module.ModuleReader");
Class<?> clazzStream = Class.forName("java.util.stream.Stream");
// Returns ModuleFinder
Object mf = clazzModuleFinder.getMethod("ofSystem").invoke(null);
// Returns Optional containing a ModuleReference
Object optMRef = clazzModuleFinder.getMethod("find", String.class).invoke(mf, "java.base");
// Extract the ModuleReference
Object mRef = clazzOptional.getMethod("get").invoke(optMRef);
// Returns ModuleReader
Object mr = clazzModuleReference.getMethod("open").invoke(mRef);
// Returns a Stream of class names
Object stream = clazzModuleReader.getMethod("list").invoke(mr);
// Convert to an array
Object[] names = (Object[]) clazzStream.getMethod("toArray").invoke(stream);
// Cast
String [] result = new String[names.length];
for (int i = 0; i < names.length; i++) {
result[i] = (String) names[i];
}
return result;
}
}