1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
|
/*
* Copyright (c) 2024, 2025, 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 8350704
* @summary Test behaviors with various bad EnclosingMethod attribute
* @library /test/lib
* @run junit BadEnclosingMethodTest
*/
import jdk.test.lib.ByteCodeLoader;
import org.junit.jupiter.api.Test;
import java.lang.classfile.ClassFile;
import java.lang.classfile.attribute.EnclosingMethodAttribute;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Map;
import java.util.Optional;
import static java.lang.constant.ConstantDescs.INIT_NAME;
import static org.junit.jupiter.api.Assertions.*;
class BadEnclosingMethodTest {
private static Path classPath(String className) {
return Path.of(System.getProperty("test.classes"), className + ".class");
}
/**
* Loads a test class that is transformed from the Enclosed local class in
* the Encloser::work method. This local class has its EnclosingMethod
* attribute transformed to the specific name and type, which may be malformed
* strings.
*
* @param name the new enclosing method name, may be malformed
* @param type the new enclosing method type, may be malformed
* @return the loaded test class, for reflective inspection
*/
private Class<?> loadTestClass(String name, String type) throws Exception {
var outerName = "Encloser";
var className = outerName + "$1Enclosed";
var cf = ClassFile.of();
var cm = cf.parse(classPath(className));
var bytes = cf.transformClass(cm, (cb, ce) -> {
if (ce instanceof EnclosingMethodAttribute em) {
var cp = cb.constantPool();
var enclosingMethodName = cp.utf8Entry(name);
var enclosingMethodType = cp.utf8Entry(type); // a malformed method type
cb.with(EnclosingMethodAttribute.of(em.enclosingClass(), Optional.of(cp.nameAndTypeEntry(
enclosingMethodName, enclosingMethodType
))));
} else {
cb.with(ce);
}
});
var map = Map.of(
outerName, Files.readAllBytes(classPath(outerName)),
className, bytes
);
return new ByteCodeLoader(map, BadEnclosingMethodTest.class.getClassLoader())
.loadClass(className);
}
/**
* Test reflection behaviors when the EnclosingMethod attribute's type is
* an invalid string.
*/
@Test
void testMalformedTypes() throws Exception {
assertThrows(ClassFormatError.class, () -> loadTestClass("methodName", "(L[;)V"));
assertThrows(ClassFormatError.class, () -> loadTestClass(INIT_NAME, "(L[;)V"));
}
/**
* Test reflective behaviors when the EnclosingMethod attribute's type is
* valid, but refers to a class or interface that cannot be found.
*/
@Test
void testAbsentMethods() throws Exception {
var absentMethodType = loadTestClass("methodName", "(Ldoes/not/Exist;)V");
var ex = assertThrows(TypeNotPresentException.class,
absentMethodType::getEnclosingMethod);
assertEquals("does.not.Exist", ex.typeName());
var absentConstructorType = loadTestClass(INIT_NAME, "(Ldoes/not/Exist;)V");
ex = assertThrows(TypeNotPresentException.class,
absentConstructorType::getEnclosingConstructor);
assertEquals("does.not.Exist", ex.typeName());
}
}
class Encloser {
private static void work() {
class Enclosed {
}
}
}
|