jdk-24/test/langtools/tools/javac/classfiles/attributes/annotations/RuntimeParameterAnnotationsTestBase.java
Adam Sotona 2b00ac0d02 8308753: Class-File API transition to Preview
Reviewed-by: ihse, mchung, vromero
2023-12-04 07:07:57 +00:00

153 lines
7.1 KiB
Java

/*
* Copyright (c) 2015, 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.lang.classfile.*;
import java.lang.classfile.attribute.*;
import javax.tools.JavaFileObject;
import java.io.IOException;
import java.lang.annotation.RetentionPolicy;
import java.util.*;
public abstract class RuntimeParameterAnnotationsTestBase extends AnnotationsTestBase {
@Override
public void test(TestCase testCase, Map<String, ? extends JavaFileObject> classes)
throws IOException {
for (Map.Entry<String, ? extends JavaFileObject> entry : classes.entrySet()) {
ClassModel classFile = readClassFile(classes.get(entry.getKey()));
Set<String> foundMethods = new HashSet<>();
String className = classFile.thisClass().name().stringValue();
TestCase.TestClassInfo testClassInfo = testCase.classes.get(className);
for (MethodModel method : classFile.methods()) {
String methodName = method.methodName().stringValue() +
method.methodTypeSymbol().displayDescriptor();
methodName = methodName.substring(0, methodName.indexOf(")") + 1);
if (methodName.startsWith("<init>")) {
methodName = methodName.replace("<init>", className);
}
foundMethods.add(methodName);
echo("Testing method : " + methodName);
TestCase.TestMethodInfo testMethod = testClassInfo.getTestMethodInfo(methodName);
if (testMethod == null) {
continue;
}
testAttributes(testMethod, classFile, method);
}
checkContains(foundMethods, testClassInfo.methods.keySet(), "Methods in " + className);
}
}
protected void testAttributes(
TestCase.TestMethodInfo testMethod,
ClassModel classFile,
MethodModel method) {
List<Map<String, Annotation>> actualInvisible = collectAnnotations(
classFile,
testMethod,
method,
Attributes.RUNTIME_INVISIBLE_PARAMETER_ANNOTATIONS);
List<Map<String, Annotation>> actualVisible = collectAnnotations(
classFile,
testMethod,
method,
Attributes.RUNTIME_VISIBLE_PARAMETER_ANNOTATIONS);
List<TestCase.TestParameterInfo> parameters = testMethod.parameters;
for (int i = 0; i < parameters.size(); ++i) {
TestCase.TestParameterInfo parameter = parameters.get(i);
checkEquals(actualInvisible.get(i).keySet(), parameter.getRuntimeInvisibleAnnotations(),
"RuntimeInvisibleParameterAnnotations");
checkEquals(actualVisible.get(i).keySet(), parameter.getRuntimeVisibleAnnotations(),
"RuntimeVisibleParameterAnnotations");
}
for (int i = 0; i < parameters.size(); ++i) {
TestCase.TestParameterInfo parameter = parameters.get(i);
for (TestAnnotationInfo expectedAnnotation : parameter.annotations.values()) {
RetentionPolicy policy = getRetentionPolicy(expectedAnnotation.annotationName);
if (policy == RetentionPolicy.SOURCE) {
continue;
}
printf("Testing: isVisible: %s %s%n", policy.toString(), expectedAnnotation.annotationName);
Annotation actualAnnotation =
(policy == RetentionPolicy.RUNTIME
? actualVisible
: actualInvisible)
.get(i).get(expectedAnnotation.annotationName);
if (checkNotNull(actualAnnotation, "Annotation is found : "
+ expectedAnnotation.annotationName)) {
expectedAnnotation.testAnnotation(this, classFile,
actualAnnotation);
}
}
}
}
private <T extends Attribute<T>> List<Map<String, Annotation>> collectAnnotations(
ClassModel classFile,
TestCase.TestMethodInfo testMethod,
MethodModel method,
AttributeMapper<T> attribute) {
Object attr = method.findAttribute(attribute).orElse(null);
List<Map<String, Annotation>> actualAnnotations = new ArrayList<>();
RetentionPolicy policy = getRetentionPolicy(attribute.name());
if (testMethod.isParameterAnnotated(policy)) {
if (!checkNotNull(attr, "Attribute " + attribute.name() + " must not be null")) {
testMethod.parameters.forEach($ -> actualAnnotations.add(new HashMap<>()));
return actualAnnotations;
}
List<List<Annotation>> annotationList;
switch (attr) {
case RuntimeVisibleParameterAnnotationsAttribute pAnnots -> {
annotationList = pAnnots.parameterAnnotations();
}
case RuntimeInvisibleParameterAnnotationsAttribute pAnnots -> {
annotationList = pAnnots.parameterAnnotations();
}
default -> throw new AssertionError();
}
for (List<Annotation> anns: annotationList) {
Map<String, Annotation> annotations = new HashMap<>();
for (Annotation ann: anns) {
String name = ann.classSymbol().displayName();
annotations.put(name, ann);
}
actualAnnotations.add(annotations);
}
checkEquals(countNumberOfAttributes(method.attributes(),
getRetentionPolicy(attribute.name()) == RetentionPolicy.RUNTIME
? RuntimeVisibleParameterAnnotationsAttribute.class
: RuntimeInvisibleParameterAnnotationsAttribute.class),
1L,
String.format("Number of %s", attribute.name()));
} else {
checkNull(attr, String.format("%s should be null", attribute.name()));
testMethod.parameters.forEach($ -> actualAnnotations.add(new HashMap<>()));
}
return actualAnnotations;
}
}