jdk-24/test/jdk/sun/reflect/ReflectionFactory/ReflectionFactoryTest.java

847 lines
32 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2016, 2024, 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.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectInputStream;
import java.io.ObjectOutput;
import java.io.ObjectOutputStream;
import java.io.ObjectStreamClass;
import java.io.ObjectStreamException;
import java.io.ObjectStreamField;
import java.io.OptionalDataException;
import java.io.Serial;
import java.io.Serializable;
import java.lang.invoke.MethodHandle;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.Arrays;
import sun.reflect.ReflectionFactory;
import org.testng.Assert;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test;
import org.testng.annotations.DataProvider;
import org.testng.TestNG;
/*
* @test
* @bug 8137058 8164908 8168980 8275137 8333796
* @summary Basic test for the unsupported ReflectionFactory
* @modules jdk.unsupported
* @run testng ReflectionFactoryTest
*/
public class ReflectionFactoryTest {
// Initialized by init()
static ReflectionFactory factory;
@DataProvider(name = "ClassConstructors")
static Object[][] classConstructors() {
return new Object[][] {
{Object.class},
{Foo.class},
{Bar.class},
};
}
@BeforeClass
static void init() {
factory = ReflectionFactory.getReflectionFactory();
}
/**
* Test that the correct Constructor is selected and run.
* @param type type of object to create
* @throws NoSuchMethodException - error
* @throws InstantiationException - error
* @throws IllegalAccessException - error
* @throws InvocationTargetException - error
*/
@Test(dataProvider="ClassConstructors")
static void testConstructor(Class<?> type)
throws InstantiationException, IllegalAccessException, InvocationTargetException
{
@SuppressWarnings("unchecked")
Constructor<?> c = factory.newConstructorForSerialization(type);
Object o = c.newInstance();
Assert.assertEquals(o.getClass(), type, "Instance is wrong type");
if (o instanceof Foo) {
Foo foo = (Foo)o;
foo.check();
}
}
@DataProvider(name = "NonSerialConstructors")
static Object[][] constructors() throws NoSuchMethodException {
return new Object[][] {
{Foo.class, Object.class.getDeclaredConstructor()},
{Foo.class, Foo.class.getDeclaredConstructor()},
{Baz.class, Object.class.getDeclaredConstructor()},
{Baz.class, Foo.class.getDeclaredConstructor()},
{Baz.class, Baz.class.getDeclaredConstructor()}
};
}
/**
* Tests that the given Constructor, in the hierarchy, is run.
*/
@Test(dataProvider="NonSerialConstructors")
static void testNonSerializableConstructor(Class<?> cl,
Constructor<?> constructorToCall)
throws ReflectiveOperationException
{
@SuppressWarnings("unchecked")
Constructor<?> c = factory.newConstructorForSerialization(cl,
constructorToCall);
Object o = c.newInstance();
Assert.assertEquals(o.getClass(), cl, "Instance is wrong type");
int expectedFoo = 0;
int expectedBaz = 0;
if (constructorToCall.getName().equals("ReflectionFactoryTest$Foo")) {
expectedFoo = 1;
} else if (constructorToCall.getName().equals("ReflectionFactoryTest$Baz")) {
expectedFoo = 1;
expectedBaz = 4;
}
Assert.assertEquals(((Foo)o).foo(), expectedFoo);
if (o instanceof Baz b) {
Assert.assertEquals(b.baz(), expectedBaz);
}
}
@Test(expectedExceptions = UnsupportedOperationException.class)
static void testConstructorNotSuperClass() throws ReflectiveOperationException {
factory.newConstructorForSerialization(Bar.class, Baz.class.getDeclaredConstructor());
}
static class Foo {
private int foo;
public Foo() {
this.foo = 1;
}
public String toString() {
return "foo: " + foo;
}
public void check() {
int expectedFoo = 1;
Assert.assertEquals(foo, expectedFoo, "foo() constructor not run");
}
public int foo() { return foo; }
}
static class Bar extends Foo implements Serializable {
private int bar;
public Bar() {
this.bar = 1;
}
public String toString() {
return super.toString() + ", bar: " + bar;
}
public void check() {
super.check();
int expectedBar = 0;
Assert.assertEquals(bar, expectedBar, "bar() constructor not run");
}
}
static class Baz extends Foo {
private final int baz;
public Baz() { this.baz = 4; }
public int baz() { return baz; }
}
/**
* Tests that newConstructorForExternalization returns the constructor and it can be called.
* @throws NoSuchMethodException - error
* @throws InstantiationException - error
* @throws IllegalAccessException - error
* @throws InvocationTargetException - error
*/
@Test
static void newConstructorForExternalization()
throws InstantiationException, IllegalAccessException, InvocationTargetException {
Constructor<?> cons = factory.newConstructorForExternalization(Ext.class);
Ext ext = (Ext)cons.newInstance();
Assert.assertEquals(ext.ext, 1, "Constructor not run");
}
static class Ext implements Externalizable {
private static final long serialVersionUID = 1L;
int ext;
public Ext() {
ext = 1;
}
@Override
public void writeExternal(ObjectOutput out) throws IOException {}
@Override
public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {}
}
@Test
static void testReadWriteObjectForSerialization() throws Throwable {
MethodHandle readObjectMethod = factory.readObjectForSerialization(Ser.class);
Assert.assertNotNull(readObjectMethod, "readObjectMethod not found");
MethodHandle readObjectNoDataMethod = factory.readObjectNoDataForSerialization(Ser.class);
Assert.assertNotNull(readObjectNoDataMethod, "readObjectNoDataMethod not found");
MethodHandle writeObjectMethod = factory.writeObjectForSerialization(Ser.class);
Assert.assertNotNull(writeObjectMethod, "writeObjectMethod not found");
MethodHandle readResolveMethod = factory.readResolveForSerialization(Ser.class);
Assert.assertNotNull(readResolveMethod, "readResolveMethod not found");
MethodHandle writeReplaceMethod = factory.writeReplaceForSerialization(Ser.class);
Assert.assertNotNull(writeReplaceMethod, "writeReplaceMethod not found");
byte[] data = null;
try (ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(baos)) {
Ser ser = new Ser();
writeReplaceMethod.invoke(ser);
Assert.assertTrue(ser.writeReplaceCalled, "writeReplace not called");
Assert.assertFalse(ser.writeObjectCalled, "writeObject should not have been called");
writeObjectMethod.invoke(ser, oos);
Assert.assertTrue(ser.writeReplaceCalled, "writeReplace should have been called");
Assert.assertTrue(ser.writeObjectCalled, "writeObject not called");
oos.flush();
data = baos.toByteArray();
}
try (ByteArrayInputStream bais = new ByteArrayInputStream(data);
ObjectInputStream ois = new ObjectInputStream(bais)) {
Ser ser2 = new Ser();
readObjectMethod.invoke(ser2, ois);
Assert.assertTrue(ser2.readObjectCalled, "readObject not called");
Assert.assertFalse(ser2.readObjectNoDataCalled, "readObjectNoData should not be called");
Assert.assertFalse(ser2.readResolveCalled, "readResolve should not be called");
readObjectNoDataMethod.invoke(ser2);
Assert.assertTrue(ser2.readObjectCalled, "readObject should have been called");
Assert.assertTrue(ser2.readObjectNoDataCalled, "readObjectNoData not called");
Assert.assertFalse(ser2.readResolveCalled, "readResolve should not be called");
readResolveMethod.invoke(ser2);
Assert.assertTrue(ser2.readObjectCalled, "readObject should have been called");
Assert.assertTrue(ser2.readObjectNoDataCalled, "readObjectNoData not called");
Assert.assertTrue(ser2.readResolveCalled, "readResolve not called");
}
}
@Test
static void hasStaticInitializer() {
boolean actual = factory.hasStaticInitializerForSerialization(Ser.class);
Assert.assertTrue(actual, "hasStaticInitializerForSerialization is wrong");
}
static class Ser implements Serializable {
private static final long serialVersionUID = 2L;
static {
// Define a static class initialization method
}
boolean readObjectCalled = false;
boolean readObjectNoDataCalled = false;
boolean writeObjectCalled = false;
boolean readResolveCalled = false;
boolean writeReplaceCalled = false;
public Ser() {}
private void readObject(ObjectInputStream ois) throws IOException, ClassNotFoundException {
Assert.assertFalse(writeObjectCalled, "readObject called too many times");
readObjectCalled = ois.readBoolean();
}
private void readObjectNoData() throws ObjectStreamException {
Assert.assertFalse(readObjectNoDataCalled, "readObjectNoData called too many times");
readObjectNoDataCalled = true;
}
private void writeObject(ObjectOutputStream oos) throws IOException {
Assert.assertFalse(writeObjectCalled, "writeObject called too many times");
writeObjectCalled = true;
oos.writeBoolean(writeObjectCalled);
}
private Object writeReplace() throws ObjectStreamException {
Assert.assertFalse(writeReplaceCalled, "writeReplace called too many times");
writeReplaceCalled = true;
return this;
}
private Object readResolve() throws ObjectStreamException {
Assert.assertFalse(readResolveCalled, "readResolve called too many times");
readResolveCalled = true;
return this;
}
}
/**
* Tests the constructor of OptionalDataExceptions.
*/
@Test
static void newOptionalDataException() {
OptionalDataException ode = factory.newOptionalDataExceptionForSerialization(true);
Assert.assertTrue(ode.eof, "eof wrong");
ode = factory.newOptionalDataExceptionForSerialization(false);
Assert.assertFalse(ode.eof, "eof wrong");
}
private static final String[] names = {
"boolean_",
"final_boolean",
"byte_",
"final_byte",
"char_",
"final_char",
"short_",
"final_short",
"int_",
"final_int",
"long_",
"final_long",
"float_",
"final_float",
"double_",
"final_double",
"str",
"final_str",
"writeFields",
};
// test that the generated read/write objects are working properly
@Test
static void testDefaultReadWriteObject() throws Throwable {
Ser2 ser = new Ser2((byte) 0x33, (short) 0x2244, (char) 0x5342, 0x05382716, 0xf035a73b09113bacL, 1234f, 3456.0, true, new Ser3(0x004917aa));
ser.byte_ = (byte) 0x44;
ser.short_ = (short) 0x3355;
ser.char_ = (char) 0x6593;
ser.int_ = 0x4928a299;
ser.long_ = 0x24aa19883f4b9138L;
ser.float_ = 4321f;
ser.double_ = 6543.0;
ser.boolean_ = false;
ser.ser = new Ser3(0x70b030a0);
// first, ensure that each field gets written
MethodHandle writeObject = factory.defaultWriteObjectForSerialization(Ser2.class);
Assert.assertNotNull(writeObject, "writeObject not created");
boolean[] called = new boolean[19];
@SuppressWarnings("removal")
ObjectOutputStream oos = new ObjectOutputStream() {
protected void writeObjectOverride(final Object obj) throws IOException {
throw new IOException("Wrong method called");
}
public void defaultWriteObject() throws IOException {
throw new IOException("Wrong method called");
}
public void writeFields() {
called[18] = true;
}
public PutField putFields() {
return new PutField() {
public void put(final String name, final boolean val) {
switch (name) {
case "boolean_" -> {
Assert.assertFalse(val);
called[0] = true;
}
case "final_boolean" -> {
Assert.assertTrue(val);
called[1] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final byte val) {
switch (name) {
case "byte_" -> {
Assert.assertEquals(val, (byte) 0x44);
called[2] = true;
}
case "final_byte" -> {
Assert.assertEquals(val, (byte) 0x33);
called[3] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final char val) {
switch (name) {
case "char_" -> {
Assert.assertEquals(val, (char) 0x6593);
called[4] = true;
}
case "final_char" -> {
Assert.assertEquals(val, (char) 0x5342);
called[5] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final short val) {
switch (name) {
case "short_" -> {
Assert.assertEquals(val, (short) 0x3355);
called[6] = true;
}
case "final_short" -> {
Assert.assertEquals(val, (short) 0x2244);
called[7] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final int val) {
switch (name) {
case "int_" -> {
Assert.assertEquals(val, 0x4928a299);
called[8] = true;
}
case "final_int" -> {
Assert.assertEquals(val, 0x05382716);
called[9] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final long val) {
switch (name) {
case "long_" -> {
Assert.assertEquals(val, 0x24aa19883f4b9138L);
called[10] = true;
}
case "final_long" -> {
Assert.assertEquals(val, 0xf035a73b09113bacL);
called[11] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final float val) {
switch (name) {
case "float_" -> {
Assert.assertEquals(val, 4321f);
called[12] = true;
}
case "final_float" -> {
Assert.assertEquals(val, 1234f);
called[13] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final double val) {
switch (name) {
case "double_" -> {
Assert.assertEquals(val, 6543.0);
called[14] = true;
}
case "final_double" -> {
Assert.assertEquals(val, 3456.0);
called[15] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
public void put(final String name, final Object val) {
switch (name) {
case "ser" -> {
Assert.assertEquals(val, new Ser3(0x70b030a0));
called[16] = true;
}
case "final_ser" -> {
Assert.assertEquals(val, new Ser3(0x004917aa));
called[17] = true;
}
default -> throw new Error("Unexpected field " + name);
}
}
@SuppressWarnings("removal")
public void write(final ObjectOutput out) throws IOException {
throw new IOException("Wrong method called");
}
};
}
};
writeObject.invokeExact(ser, oos);
for (int i = 0; i < 19; i ++) {
Assert.assertTrue(called[i], names[i]);
}
// now, test the read side
MethodHandle readObject = factory.defaultReadObjectForSerialization(Ser2.class);
Assert.assertNotNull(readObject, "readObject not created");
@SuppressWarnings("removal")
ObjectInputStream ois = new ObjectInputStream() {
protected Object readObjectOverride() throws IOException {
throw new IOException("Wrong method called");
}
public GetField readFields() {
return new GetField() {
public ObjectStreamClass getObjectStreamClass() {
throw new Error("Wrong method called");
}
public boolean defaulted(final String name) throws IOException {
throw new IOException("Wrong method called");
}
public boolean get(final String name, final boolean val) {
return switch (name) {
case "boolean_" -> {
called[0] = true;
yield true;
}
case "final_boolean" -> {
called[1] = true;
yield true;
}
default -> throw new Error("Unexpected field " + name);
};
}
public byte get(final String name, final byte val) {
return switch (name) {
case "byte_" -> {
called[2] = true;
yield (byte) 0x11;
}
case "final_byte" -> {
called[3] = true;
yield (byte) 0x9f;
}
default -> throw new Error("Unexpected field " + name);
};
}
public char get(final String name, final char val) {
return switch (name) {
case "char_" -> {
called[4] = true;
yield (char) 0x59a2;
}
case "final_char" -> {
called[5] = true;
yield (char) 0xe0d0;
}
default -> throw new Error("Unexpected field " + name);
};
}
public short get(final String name, final short val) {
return switch (name) {
case "short_" -> {
called[6] = true;
yield (short) 0x0917;
}
case "final_short" -> {
called[7] = true;
yield (short) 0x110e;
}
default -> throw new Error("Unexpected field " + name);
};
}
public int get(final String name, final int val) {
return switch (name) {
case "int_" -> {
called[8] = true;
yield 0xd0244e19;
}
case "final_int" -> {
called[9] = true;
yield 0x011004da;
}
default -> throw new Error("Unexpected field " + name);
};
}
public long get(final String name, final long val) {
return switch (name) {
case "long_" -> {
called[10] = true;
yield 0x0b8101d84aa31711L;
}
case "final_long" -> {
called[11] = true;
yield 0x30558aa7189ed821L;
}
default -> throw new Error("Unexpected field " + name);
};
}
public float get(final String name, final float val) {
return switch (name) {
case "float_" -> {
called[12] = true;
yield 0x5.01923ap18f;
}
case "final_float" -> {
called[13] = true;
yield 0x0.882afap1f;
}
default -> throw new Error("Unexpected field " + name);
};
}
public double get(final String name, final double val) {
return switch (name) {
case "double_" -> {
called[14] = true;
yield 0x9.4a8fp6;
}
case "final_double" -> {
called[15] = true;
yield 0xf.881a8p4;
}
default -> throw new Error("Unexpected field " + name);
};
}
public Object get(final String name, final Object val) {
return switch (name) {
case "ser" -> {
called[16] = true;
yield new Ser3(0x44cc55dd);
}
case "final_ser" -> {
called[17] = true;
yield new Ser3(0x9a8b7c6d);
}
default -> throw new Error("Unexpected field " + name);
};
}
};
}
};
// all the same methods, except for `writeFields`
Arrays.fill(called, false);
Constructor<?> ctor = factory.newConstructorForSerialization(Ser2.class, Object.class.getDeclaredConstructor());
ser = (Ser2) ctor.newInstance();
readObject.invokeExact(ser, ois);
// excluding "writeFields", so 18 instead of 19
for (int i = 0; i < 18; i ++) {
Assert.assertTrue(called[i], names[i]);
}
Assert.assertEquals(ser.byte_, (byte)0x11);
Assert.assertEquals(ser.final_byte, (byte)0x9f);
Assert.assertEquals(ser.char_, (char)0x59a2);
Assert.assertEquals(ser.final_char, (char)0xe0d0);
Assert.assertEquals(ser.short_, (short)0x0917);
Assert.assertEquals(ser.final_short, (short)0x110e);
Assert.assertEquals(ser.int_, 0xd0244e19);
Assert.assertEquals(ser.final_int, 0x011004da);
Assert.assertEquals(ser.long_, 0x0b8101d84aa31711L);
Assert.assertEquals(ser.final_long, 0x30558aa7189ed821L);
Assert.assertEquals(ser.float_, 0x5.01923ap18f);
Assert.assertEquals(ser.final_float, 0x0.882afap1f);
Assert.assertEquals(ser.double_, 0x9.4a8fp6);
Assert.assertEquals(ser.final_double, 0xf.881a8p4);
Assert.assertEquals(ser.ser, new Ser3(0x44cc55dd));
Assert.assertEquals(ser.final_ser, new Ser3(0x9a8b7c6d));
}
static class Ser2 implements Serializable {
@Serial
private static final long serialVersionUID = -2852896623833548574L;
byte byte_;
short short_;
char char_;
int int_;
long long_;
float float_;
double double_;
boolean boolean_;
Ser3 ser;
final byte final_byte;
final short final_short;
final char final_char;
final int final_int;
final long final_long;
final float final_float;
final double final_double;
final boolean final_boolean;
final Ser3 final_ser;
Ser2(final byte final_byte, final short final_short, final char final_char, final int final_int,
final long final_long, final float final_float, final double final_double,
final boolean final_boolean, final Ser3 final_ser) {
this.final_byte = final_byte;
this.final_short = final_short;
this.final_char = final_char;
this.final_int = final_int;
this.final_long = final_long;
this.final_float = final_float;
this.final_double = final_double;
this.final_boolean = final_boolean;
this.final_ser = final_ser;
}
}
static class Ser3 implements Serializable {
@Serial
private static final long serialVersionUID = -1234752876749422678L;
@Serial
private static final ObjectStreamField[] serialPersistentFields = {
new ObjectStreamField("value", int.class)
};
final int value;
Ser3(final int value) {
this.value = value;
}
public boolean equals(final Object obj) {
return obj instanceof Ser3 s && value == s.value;
}
public int hashCode() {
return value;
}
}
static class SerInvalidFields implements Serializable {
// this is deliberately wrong
@SuppressWarnings({"unused", "serial"})
@Serial
private static final String serialPersistentFields = "Oops!";
@Serial
private static final long serialVersionUID = -8090960816811629489L;
}
static class Ext1 implements Externalizable {
@Serial
private static final long serialVersionUID = 7109990719266285013L;
public void writeExternal(final ObjectOutput objectOutput) {
}
public void readExternal(final ObjectInput objectInput) {
}
}
static class Ext2 implements Externalizable {
public void writeExternal(final ObjectOutput objectOutput) {
}
public void readExternal(final ObjectInput objectInput) {
}
}
record Rec1(int hello, boolean world) implements Serializable {
@Serial
private static final long serialVersionUID = 12349876L;
}
enum Enum1 {
hello,
world,
;
private static final long serialVersionUID = 1020304050L;
}
interface Proxy1 {
void hello();
}
static class SerialPersistentFields implements Serializable {
@Serial
private static final long serialVersionUID = -4947917866973382882L;
@Serial
private static final ObjectStreamField[] serialPersistentFields = {
new ObjectStreamField("array1", Object[].class),
new ObjectStreamField("nonExistent", String.class)
};
private int int1;
private Object[] array1;
}
// Check our simple accessors
@Test
static void testAccessors() {
Assert.assertEquals(factory.serialPersistentFields(Ser3.class), Ser3.serialPersistentFields);
Assert.assertNotSame(factory.serialPersistentFields(Ser3.class), Ser3.serialPersistentFields);
Assert.assertNull(factory.serialPersistentFields(SerInvalidFields.class));
}
// Ensure that classes with serialPersistentFields do not allow default setting/getting
@Test
static void testDisallowed() {
Assert.assertNull(factory.defaultWriteObjectForSerialization(SerialPersistentFields.class));
Assert.assertNull(factory.defaultReadObjectForSerialization(SerialPersistentFields.class));
}
// Main can be used to run the tests from the command line with only testng.jar.
@SuppressWarnings("raw_types")
@Test(enabled = false)
public static void main(String[] args) {
Class<?>[] testclass = {ReflectionFactoryTest.class};
TestNG testng = new TestNG();
testng.setTestClasses(testclass);
testng.run();
}
}