mirror of
https://github.com/openjdk/jdk.git
synced 2026-01-28 03:58:21 +00:00
8370839: Tests to verify peculiar Proxy dispatching behaviors
Reviewed-by: jvernee
This commit is contained in:
parent
b6ba1ac9aa
commit
5f42c77085
86
test/jdk/java/lang/reflect/Proxy/BridgeMethodsTest.java
Normal file
86
test/jdk/java/lang/reflect/Proxy/BridgeMethodsTest.java
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import java.lang.reflect.Proxy;
|
||||||
|
import java.lang.reflect.UndeclaredThrowableException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8370839
|
||||||
|
* @summary Behavior of bridge methods in interfaces
|
||||||
|
* @run junit BridgeMethodsTest
|
||||||
|
*/
|
||||||
|
public class BridgeMethodsTest {
|
||||||
|
|
||||||
|
interface StringCallable extends Callable<String> {
|
||||||
|
@Override
|
||||||
|
String call(); // throws no exception
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testExceptionTypes() throws Throwable {
|
||||||
|
class MyException extends Exception {}
|
||||||
|
// This proxy has two distinct methods, even though the
|
||||||
|
// Java language would treat the first one as overridden:
|
||||||
|
// Object call() throws Exception; - from Callable
|
||||||
|
// String call(); - from StringCallable
|
||||||
|
var instance = Proxy.newProxyInstance(StringCallable.class.getClassLoader(),
|
||||||
|
new Class[] { StringCallable.class }, (_, _, _) -> { throw new MyException(); });
|
||||||
|
// The exception can't be thrown through StringCallable.call which has no throws
|
||||||
|
var undeclared = assertThrows(UndeclaredThrowableException.class, () -> ((StringCallable) instance).call());
|
||||||
|
assertInstanceOf(MyException.class, undeclared.getCause());
|
||||||
|
// But it can be thrown through Callable.call which permits Exception
|
||||||
|
assertThrows(MyException.class, () -> ((Callable<?>) instance).call());
|
||||||
|
}
|
||||||
|
|
||||||
|
interface SpecificConsumer extends Consumer<String> {
|
||||||
|
@Override
|
||||||
|
void accept(String s);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
void testMethodObjects() throws Throwable {
|
||||||
|
List<Method> methods = new ArrayList<>();
|
||||||
|
// This proxy has two distinct methods, even though the
|
||||||
|
// Java language would treat the first one as overridden:
|
||||||
|
// void accept(Object); - from Consumer
|
||||||
|
// void accept(String); - from SpecificConsumer
|
||||||
|
var instance = Proxy.newProxyInstance(SpecificConsumer.class.getClassLoader(),
|
||||||
|
new Class[] { SpecificConsumer.class }, (_, m, _) -> methods.add(m));
|
||||||
|
((Consumer<Object>) instance).accept(null);
|
||||||
|
((SpecificConsumer) instance).accept(null);
|
||||||
|
assertEquals(2, methods.size());
|
||||||
|
// invocation handler gets different method due to covariant parameter types
|
||||||
|
assertNotEquals(methods.getFirst(), methods.getLast());
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2024, 2025, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -21,39 +21,86 @@
|
|||||||
* questions.
|
* questions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
import java.lang.reflect.Proxy;
|
||||||
* @test
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
* @bug 8333854
|
|
||||||
* @summary Test invoking a method in a proxy interface with package-private
|
|
||||||
* classes or interfaces in its method type
|
|
||||||
* @run junit NonPublicMethodTypeTest
|
|
||||||
*/
|
|
||||||
|
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
import java.lang.reflect.Proxy;
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
import static org.junit.jupiter.api.Assertions.assertNotSame;
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8333854 8370839
|
||||||
|
* @summary Behavior of methods whose signature has package-private
|
||||||
|
* class or interfaces but the proxy interface is public
|
||||||
|
* @run junit NonPublicMethodTypeTest
|
||||||
|
*/
|
||||||
|
public class NonPublicMethodTypeTest {
|
||||||
|
// Java language and JVM allow using fields and methods with inaccessible
|
||||||
|
// classes or interfaces in its signature, as long as the field or method
|
||||||
|
// is accessible and its declaring class or interface is accessible.
|
||||||
|
// Such inaccessible classes and interfaces are treated as if an arbitrary
|
||||||
|
// subtype of their accessible types, or an arbitrary supertype of their
|
||||||
|
// accessible subtypes.
|
||||||
|
// java.lang.invoke is stricter - MethodType constant pool entry resolution
|
||||||
|
// for such signatures fail, so they can't be used for MethodHandle or indy.
|
||||||
|
enum Internal { INSTANCE }
|
||||||
|
|
||||||
public final class NonPublicMethodTypeTest {
|
public interface InternalParameter {
|
||||||
interface NonPublicWorker {
|
void call(Internal parameter);
|
||||||
void work();
|
|
||||||
}
|
|
||||||
|
|
||||||
public interface PublicWorkable {
|
|
||||||
void accept(NonPublicWorker worker);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void test() {
|
void testNonPublicParameter() throws Throwable {
|
||||||
PublicWorkable proxy = (PublicWorkable) Proxy.newProxyInstance(
|
// Creation should be successful
|
||||||
NonPublicMethodTypeTest.class.getClassLoader(),
|
// 8333854 - BSM usage fails for looking up such methods
|
||||||
new Class[] {PublicWorkable.class},
|
InternalParameter instance = (InternalParameter) Proxy.newProxyInstance(
|
||||||
(_, _, _) -> null);
|
InternalParameter.class.getClassLoader(),
|
||||||
assertNotSame(NonPublicWorker.class.getPackage(),
|
new Class[] { InternalParameter.class },
|
||||||
proxy.getClass().getPackage(),
|
(_, _, _) -> null);
|
||||||
|
assertNotSame(Internal.class.getPackage(),
|
||||||
|
instance.getClass().getPackage(),
|
||||||
"Proxy class should not be able to access method parameter " +
|
"Proxy class should not be able to access method parameter " +
|
||||||
"NonPublic type's package");
|
"Internal class's package");
|
||||||
proxy.accept(() -> {}); // Call should not fail
|
// Calls should be always successful
|
||||||
|
instance.call(null);
|
||||||
|
instance.call(Internal.INSTANCE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public interface InternalReturn {
|
||||||
|
Internal call();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testNonPublicReturn() throws Throwable {
|
||||||
|
AtomicReference<Internal> returnValue = new AtomicReference<>();
|
||||||
|
// Creation should be successful
|
||||||
|
// A lot of annotation interfaces are implemented by such proxy classes,
|
||||||
|
// due to presence of package-private annotation interface or enum-typed
|
||||||
|
// elements in public annotation interfaces.
|
||||||
|
InternalReturn instance = (InternalReturn) Proxy.newProxyInstance(
|
||||||
|
InternalReturn.class.getClassLoader(),
|
||||||
|
new Class[] { InternalReturn.class },
|
||||||
|
(_, _, _) -> returnValue.get());
|
||||||
|
assertNotSame(Internal.class.getPackage(),
|
||||||
|
instance.getClass().getPackage(),
|
||||||
|
"Proxy class should not be able to access method parameter " +
|
||||||
|
"Internal class's package");
|
||||||
|
|
||||||
|
// The generated call() implementation is as follows:
|
||||||
|
// aload0, getfield Proxy.h, aload 0, getstatic (Method), aconst_null,
|
||||||
|
// invokevirtual InvocationHandler::invoke(Object, Method, Object[])Object,
|
||||||
|
// checkcast Internal.class, areturn
|
||||||
|
// In this bytecode, checkcast Internal.class will fail with a
|
||||||
|
// IllegalAccessError as a result of resolution of Internal.class
|
||||||
|
// if and only if the incoming reference is non-null.
|
||||||
|
|
||||||
|
// checkcast does not perform access check for null
|
||||||
|
returnValue.set(null);
|
||||||
|
instance.call();
|
||||||
|
// checkcast fails - proxy class cannot access the return type
|
||||||
|
// See JDK-8349716
|
||||||
|
returnValue.set(Internal.INSTANCE);
|
||||||
|
assertThrows(IllegalAccessError.class, instance::call);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
190
test/jdk/java/lang/reflect/Proxy/ProtectedObjectMethodsTest.java
Normal file
190
test/jdk/java/lang/reflect/Proxy/ProtectedObjectMethodsTest.java
Normal file
@ -0,0 +1,190 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.lang.invoke.MethodHandle;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.lang.invoke.MethodType;
|
||||||
|
import java.lang.reflect.Proxy;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8370839
|
||||||
|
* @summary Behavior of protected methods in java.lang.Object
|
||||||
|
* @modules java.base/java.lang:+open
|
||||||
|
* @run junit ProtectedObjectMethodsTest
|
||||||
|
*/
|
||||||
|
public class ProtectedObjectMethodsTest {
|
||||||
|
|
||||||
|
static final MethodHandle OBJECT_CLONE;
|
||||||
|
static final MethodHandle OBJECT_FINALIZE;
|
||||||
|
|
||||||
|
static {
|
||||||
|
try {
|
||||||
|
var lookup = MethodHandles.privateLookupIn(Object.class, MethodHandles.lookup());
|
||||||
|
OBJECT_CLONE = lookup.findVirtual(Object.class, "clone", MethodType.methodType(Object.class));
|
||||||
|
OBJECT_FINALIZE = lookup.findVirtual(Object.class, "finalize", MethodType.methodType(void.class));
|
||||||
|
} catch (ReflectiveOperationException ex) {
|
||||||
|
throw new ExceptionInInitializerError(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FakeClone {
|
||||||
|
// This method is not related to Object::clone in the JVM, but most
|
||||||
|
// implementations in the Java Language override Object::clone as
|
||||||
|
// covariant overrides
|
||||||
|
FakeClone clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TrueClone {
|
||||||
|
// This method is identical to Object::clone in the JVM, that calls
|
||||||
|
// to Object::clone can always call an implementation of this method
|
||||||
|
// if it exists
|
||||||
|
Object clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
interface PrimitiveClone {
|
||||||
|
// This method is not related to Object::clone in the JVM, but it can't
|
||||||
|
// be implemented in the Java Language unless using a lambda expression,
|
||||||
|
// due to the Java language covariant override restrictions.
|
||||||
|
int clone();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Does not duplicate with Object::clone so it is not proxied
|
||||||
|
@Test
|
||||||
|
void testDistinctClone() throws Throwable {
|
||||||
|
{
|
||||||
|
// This proxy declares FakeClone clone(); which cannot be
|
||||||
|
// invoked through Object::clone.
|
||||||
|
var fake = (FakeClone) Proxy.newProxyInstance(FakeClone.class.getClassLoader(), new Class[] { FakeClone.class }, (p, _, _) -> p);
|
||||||
|
assertSame(fake, fake.clone());
|
||||||
|
// Verify the default Object::clone behavior (this is not Cloneable)
|
||||||
|
assertThrows(CloneNotSupportedException.class, () -> {
|
||||||
|
var _ = (Object) OBJECT_CLONE.invoke((Object) fake);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// This proxy declares FakeClone clone(); which cannot be
|
||||||
|
// invoked through Object::clone.
|
||||||
|
var fake = (FakeClone) Proxy.newProxyInstance(FakeClone.class.getClassLoader(), new Class[] { FakeClone.class, Cloneable.class }, (p, _, _) -> p);
|
||||||
|
assertSame(fake, fake.clone());
|
||||||
|
// Verify the default Object::clone behavior (this is Cloneable)
|
||||||
|
var fakeClone = (Object) OBJECT_CLONE.invoke((Object) fake);
|
||||||
|
assertNotSame(fake, fakeClone);
|
||||||
|
assertSame(fake.getClass(), fakeClone.getClass());
|
||||||
|
assertSame(fakeClone, ((FakeClone) fakeClone).clone());
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// This proxy declares int clone(); which cannot be
|
||||||
|
// invoked through Object::clone.
|
||||||
|
var instance = (PrimitiveClone) Proxy.newProxyInstance(PrimitiveClone.class.getClassLoader(), new Class[] { PrimitiveClone.class }, (_, _, _) -> 42);
|
||||||
|
assertEquals(42, instance.clone());
|
||||||
|
// Verify the default Object::clone behavior (this is not Cloneable)
|
||||||
|
assertThrows(CloneNotSupportedException.class, () -> {
|
||||||
|
var _ = (Object) OBJECT_CLONE.invoke((Object) instance);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// This proxy declares int clone(); which cannot be
|
||||||
|
// invoked through Object::clone
|
||||||
|
var instance = (PrimitiveClone) Proxy.newProxyInstance(PrimitiveClone.class.getClassLoader(), new Class[] { PrimitiveClone.class, Cloneable.class }, (_, _, _) -> 76);
|
||||||
|
assertEquals(76, instance.clone());
|
||||||
|
// Verify the default Object::clone behavior (this is Cloneable)
|
||||||
|
var clone = (Object) OBJECT_CLONE.invoke((Object) instance);
|
||||||
|
assertNotSame(instance, clone);
|
||||||
|
assertSame(instance.getClass(), clone.getClass());
|
||||||
|
assertEquals(76, ((PrimitiveClone) clone).clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Duplicates with Object::clone so it is proxied
|
||||||
|
@Test
|
||||||
|
void testDuplicateClone() throws Throwable {
|
||||||
|
{
|
||||||
|
// This proxy declares Object clone();, which
|
||||||
|
// accidentally overrides Object::clone
|
||||||
|
var instance = (TrueClone) Proxy.newProxyInstance(TrueClone.class.getClassLoader(), new Class[] { TrueClone.class }, (p, _, _) -> p);
|
||||||
|
assertSame(instance, instance.clone());
|
||||||
|
// Verify Object::clone is overridden
|
||||||
|
assertSame(instance, (Object) OBJECT_CLONE.invoke((Object) instance));
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
// This proxy declares Object clone(); and FakeClone clone();.
|
||||||
|
// They are considered duplicate methods and dispatched equivalently.
|
||||||
|
// Object clone() accidentally overrides Object::clone, so now
|
||||||
|
// FakeClone clone() can be called through Object::clone
|
||||||
|
var instance = Proxy.newProxyInstance(TrueClone.class.getClassLoader(), new Class[] { TrueClone.class, FakeClone.class }, (p, _, _) -> p);
|
||||||
|
assertSame(instance, ((FakeClone) instance).clone());
|
||||||
|
assertSame(instance, ((TrueClone) instance).clone());
|
||||||
|
// Verify Object::clone is overridden
|
||||||
|
assertSame(instance, (Object) OBJECT_CLONE.invoke((Object) instance));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FalseFinalize {
|
||||||
|
// This method is not related to Object::finalize in the JVM, but it can't
|
||||||
|
// be implemented in the Java Language unless using a lambda expression,
|
||||||
|
// due to the Java language covariant override restrictions.
|
||||||
|
int finalize();
|
||||||
|
}
|
||||||
|
|
||||||
|
interface TrueFinalize {
|
||||||
|
// This method is identical to Object::finalize in the JVM, that calls
|
||||||
|
// to Object::finalize can always call an implementation of this method
|
||||||
|
// if it exists
|
||||||
|
void finalize();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testDistinctFinalize() throws Throwable {
|
||||||
|
AtomicInteger invokeCount = new AtomicInteger();
|
||||||
|
// This proxy declares int finalize(), which cannot be
|
||||||
|
// invoked through Object::finalize.
|
||||||
|
var instance = Proxy.newProxyInstance(FalseFinalize.class.getClassLoader(), new Class[] { FalseFinalize.class }, (_, _, _) -> invokeCount.incrementAndGet());
|
||||||
|
// Verify the default Object::finalize behavior
|
||||||
|
OBJECT_FINALIZE.invoke(instance);
|
||||||
|
assertEquals(0, invokeCount.get());
|
||||||
|
assertEquals(1, ((FalseFinalize) instance).finalize());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testDuplicateFinalize() throws Throwable {
|
||||||
|
AtomicInteger invokeCount = new AtomicInteger();
|
||||||
|
// This proxy declares void finalize(), which can be
|
||||||
|
// invoked through Object::finalize.
|
||||||
|
var instance = Proxy.newProxyInstance(TrueFinalize.class.getClassLoader(), new Class[] { TrueFinalize.class }, (_, _, _) -> invokeCount.incrementAndGet());
|
||||||
|
// Verify the overridden Object::finalize behavior
|
||||||
|
OBJECT_FINALIZE.invoke(instance);
|
||||||
|
assertEquals(1, invokeCount.get());
|
||||||
|
((TrueFinalize) instance).finalize();
|
||||||
|
assertEquals(2, invokeCount.get());
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
x
Reference in New Issue
Block a user