mirror of
https://github.com/corda/corda.git
synced 2025-01-03 03:36:48 +00:00
Make sure members of type Class
correctly serialized using AMQP (#1314)
* Minor changes and expose the problem with class serialization * Custom serializer for Class * More changes to make TransactionEncumbranceTests pass in AMQP mode
This commit is contained in:
parent
0c65e5e708
commit
d6d7fb52b4
@ -0,0 +1,18 @@
|
|||||||
|
package net.corda.core.serialization
|
||||||
|
|
||||||
|
import net.corda.finance.contracts.CommercialPaper
|
||||||
|
import net.corda.finance.contracts.asset.Cash
|
||||||
|
import net.corda.testing.TestDependencyInjectionBase
|
||||||
|
import org.junit.Test
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
|
||||||
|
class CommandsSerializationTests : TestDependencyInjectionBase() {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun `test cash move serialization`() {
|
||||||
|
val command = Cash.Commands.Move(CommercialPaper::class.java)
|
||||||
|
val copiedCommand = command.serialize().deserialize()
|
||||||
|
|
||||||
|
assertEquals(command, copiedCommand)
|
||||||
|
}
|
||||||
|
}
|
@ -36,6 +36,7 @@ abstract class AbstractAMQPSerializationScheme : SerializationScheme {
|
|||||||
register(net.corda.nodeapi.internal.serialization.amqp.custom.YearMonthSerializer(this))
|
register(net.corda.nodeapi.internal.serialization.amqp.custom.YearMonthSerializer(this))
|
||||||
register(net.corda.nodeapi.internal.serialization.amqp.custom.MonthDaySerializer(this))
|
register(net.corda.nodeapi.internal.serialization.amqp.custom.MonthDaySerializer(this))
|
||||||
register(net.corda.nodeapi.internal.serialization.amqp.custom.PeriodSerializer(this))
|
register(net.corda.nodeapi.internal.serialization.amqp.custom.PeriodSerializer(this))
|
||||||
|
register(net.corda.nodeapi.internal.serialization.amqp.custom.ClassSerializer(this))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -131,10 +131,10 @@ class DeserializationInput(internal val serializerFactory: SerializerFactory) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* TODO: Currently performs rather basic checks aimed in particular at [java.util.List<Command<?>>]
|
* TODO: Currently performs rather basic checks aimed in particular at [java.util.List<Command<?>>] and
|
||||||
|
* [java.lang.Class<? extends net.corda.core.contracts.Contract>]
|
||||||
* In the future tighter control might be needed
|
* In the future tighter control might be needed
|
||||||
*/
|
*/
|
||||||
private fun Type.materiallyEquivalentTo(that: Type): Boolean =
|
private fun Type.materiallyEquivalentTo(that: Type): Boolean =
|
||||||
asClass() == that.asClass() && this is ParameterizedType && that is ParameterizedType &&
|
asClass() == that.asClass() && that is ParameterizedType
|
||||||
actualTypeArguments.size == that.actualTypeArguments.size
|
|
||||||
}
|
}
|
@ -339,6 +339,16 @@ internal fun fingerprintForDescriptors(vararg typeDescriptors: String): String {
|
|||||||
return hasher.hash().asBytes().toBase64()
|
return hasher.hash().asBytes().toBase64()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun Hasher.fingerprintWithCustomSerializerOrElse(factory: SerializerFactory, clazz: Class<*>, declaredType: Type, block: () -> Hasher) : Hasher {
|
||||||
|
// Need to check if a custom serializer is applicable
|
||||||
|
val customSerializer = factory.findCustomSerializer(clazz, declaredType)
|
||||||
|
return if (customSerializer != null) {
|
||||||
|
putUnencodedChars(customSerializer.typeDescriptor)
|
||||||
|
} else {
|
||||||
|
block()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// This method concatentates various elements of the types recursively as unencoded strings into the hasher, effectively
|
// This method concatentates various elements of the types recursively as unencoded strings into the hasher, effectively
|
||||||
// creating a unique string for a type which we then hash in the calling function above.
|
// creating a unique string for a type which we then hash in the calling function above.
|
||||||
private fun fingerprintForType(type: Type, contextType: Type?, alreadySeen: MutableSet<Type>, hasher: Hasher, factory: SerializerFactory): Hasher {
|
private fun fingerprintForType(type: Type, contextType: Type?, alreadySeen: MutableSet<Type>, hasher: Hasher, factory: SerializerFactory): Hasher {
|
||||||
@ -357,9 +367,7 @@ private fun fingerprintForType(type: Type, contextType: Type?, alreadySeen: Muta
|
|||||||
} else if (isCollectionOrMap(type)) {
|
} else if (isCollectionOrMap(type)) {
|
||||||
hasher.putUnencodedChars(type.name)
|
hasher.putUnencodedChars(type.name)
|
||||||
} else {
|
} else {
|
||||||
// Need to check if a custom serializer is applicable
|
hasher.fingerprintWithCustomSerializerOrElse(factory, type, type) {
|
||||||
val customSerializer = factory.findCustomSerializer(type, type)
|
|
||||||
if (customSerializer == null) {
|
|
||||||
if (type.kotlin.objectInstance != null) {
|
if (type.kotlin.objectInstance != null) {
|
||||||
// TODO: name collision is too likely for kotlin objects, we need to introduce some reference
|
// TODO: name collision is too likely for kotlin objects, we need to introduce some reference
|
||||||
// to the CorDapp but maybe reference to the JAR in the short term.
|
// to the CorDapp but maybe reference to the JAR in the short term.
|
||||||
@ -367,8 +375,6 @@ private fun fingerprintForType(type: Type, contextType: Type?, alreadySeen: Muta
|
|||||||
} else {
|
} else {
|
||||||
fingerprintForObject(type, contextType, alreadySeen, hasher, factory)
|
fingerprintForObject(type, contextType, alreadySeen, hasher, factory)
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
hasher.putUnencodedChars(customSerializer.typeDescriptor)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (type is ParameterizedType) {
|
} else if (type is ParameterizedType) {
|
||||||
@ -377,8 +383,10 @@ private fun fingerprintForType(type: Type, contextType: Type?, alreadySeen: Muta
|
|||||||
val startingHash = if (isCollectionOrMap(clazz)) {
|
val startingHash = if (isCollectionOrMap(clazz)) {
|
||||||
hasher.putUnencodedChars(clazz.name)
|
hasher.putUnencodedChars(clazz.name)
|
||||||
} else {
|
} else {
|
||||||
|
hasher.fingerprintWithCustomSerializerOrElse(factory, clazz, type) {
|
||||||
fingerprintForObject(type, type, alreadySeen, hasher, factory)
|
fingerprintForObject(type, type, alreadySeen, hasher, factory)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// ... and concatentate the type data for each parameter type.
|
// ... and concatentate the type data for each parameter type.
|
||||||
type.actualTypeArguments.fold(startingHash) { orig, paramType -> fingerprintForType(paramType, type, alreadySeen, orig, factory) }
|
type.actualTypeArguments.fold(startingHash) { orig, paramType -> fingerprintForType(paramType, type, alreadySeen, orig, factory) }
|
||||||
} else if (type is GenericArrayType) {
|
} else if (type is GenericArrayType) {
|
||||||
|
@ -74,7 +74,8 @@ private fun <T : Any> propertiesForSerializationFromConstructor(kotlinConstructo
|
|||||||
val rc: MutableList<PropertySerializer> = ArrayList(kotlinConstructor.parameters.size)
|
val rc: MutableList<PropertySerializer> = ArrayList(kotlinConstructor.parameters.size)
|
||||||
for (param in kotlinConstructor.parameters) {
|
for (param in kotlinConstructor.parameters) {
|
||||||
val name = param.name ?: throw NotSerializableException("Constructor parameter of $clazz has no name.")
|
val name = param.name ?: throw NotSerializableException("Constructor parameter of $clazz has no name.")
|
||||||
val matchingProperty = properties[name] ?: throw NotSerializableException("No property matching constructor parameter named $name of $clazz." +
|
val matchingProperty = properties[name] ?:
|
||||||
|
throw NotSerializableException("No property matching constructor parameter named $name of $clazz." +
|
||||||
" If using Java, check that you have the -parameters option specified in the Java compiler.")
|
" If using Java, check that you have the -parameters option specified in the Java compiler.")
|
||||||
// Check that the method has a getter in java.
|
// Check that the method has a getter in java.
|
||||||
val getter = matchingProperty.readMethod ?: throw NotSerializableException("Property has no getter method for $name of $clazz." +
|
val getter = matchingProperty.readMethod ?: throw NotSerializableException("Property has no getter method for $name of $clazz." +
|
||||||
@ -163,21 +164,20 @@ private fun resolveTypeVariables(actualType: Type, contextType: Type?): Type {
|
|||||||
}
|
}
|
||||||
|
|
||||||
internal fun Type.asClass(): Class<*>? {
|
internal fun Type.asClass(): Class<*>? {
|
||||||
return if (this is Class<*>) {
|
return when {
|
||||||
this
|
this is Class<*> -> this
|
||||||
} else if (this is ParameterizedType) {
|
this is ParameterizedType -> this.rawType.asClass()
|
||||||
this.rawType.asClass()
|
this is GenericArrayType -> this.genericComponentType.asClass()?.arrayClass()
|
||||||
} else if (this is GenericArrayType) {
|
else -> null
|
||||||
this.genericComponentType.asClass()?.arrayClass()
|
}
|
||||||
} else null
|
|
||||||
}
|
}
|
||||||
|
|
||||||
internal fun Type.asArray(): Type? {
|
internal fun Type.asArray(): Type? {
|
||||||
return if (this is Class<*>) {
|
return when {
|
||||||
this.arrayClass()
|
this is Class<*> -> this.arrayClass()
|
||||||
} else if (this is ParameterizedType) {
|
this is ParameterizedType -> DeserializedGenericArrayType(this)
|
||||||
DeserializedGenericArrayType(this)
|
else -> null
|
||||||
} else null
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal fun Class<*>.arrayClass(): Class<*> = java.lang.reflect.Array.newInstance(this, 0).javaClass
|
internal fun Class<*>.arrayClass(): Class<*> = java.lang.reflect.Array.newInstance(this, 0).javaClass
|
||||||
|
@ -156,7 +156,8 @@ class SerializerFactory(val whitelist: ClassWhitelist, cl : ClassLoader) {
|
|||||||
fun get(typeDescriptor: Any, schema: Schema): AMQPSerializer<Any> {
|
fun get(typeDescriptor: Any, schema: Schema): AMQPSerializer<Any> {
|
||||||
return serializersByDescriptor[typeDescriptor] ?: {
|
return serializersByDescriptor[typeDescriptor] ?: {
|
||||||
processSchema(schema)
|
processSchema(schema)
|
||||||
serializersByDescriptor[typeDescriptor] ?: throw NotSerializableException("Could not find type matching descriptor $typeDescriptor.")
|
serializersByDescriptor[typeDescriptor] ?:
|
||||||
|
throw NotSerializableException("Could not find type matching descriptor $typeDescriptor.")
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,15 @@
|
|||||||
|
package net.corda.nodeapi.internal.serialization.amqp.custom
|
||||||
|
|
||||||
|
import net.corda.nodeapi.internal.serialization.amqp.CustomSerializer
|
||||||
|
import net.corda.nodeapi.internal.serialization.amqp.SerializerFactory
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A serializer for [Class] that uses [ClassProxy] proxy object to write out
|
||||||
|
*/
|
||||||
|
class ClassSerializer(factory: SerializerFactory) : CustomSerializer.Proxy<Class<*>, ClassSerializer.ClassProxy>(Class::class.java, ClassProxy::class.java, factory) {
|
||||||
|
override fun toProxy(obj: Class<*>): ClassProxy = ClassProxy(obj.name)
|
||||||
|
|
||||||
|
override fun fromProxy(proxy: ClassProxy): Class<*> = Class.forName(proxy.className, true, factory.classloader)
|
||||||
|
|
||||||
|
data class ClassProxy(val className: String)
|
||||||
|
}
|
@ -6,11 +6,11 @@ import net.corda.core.identity.Party
|
|||||||
import net.corda.core.transactions.LedgerTransaction
|
import net.corda.core.transactions.LedgerTransaction
|
||||||
import net.corda.core.transactions.TransactionBuilder
|
import net.corda.core.transactions.TransactionBuilder
|
||||||
|
|
||||||
// The dummy contract doesn't do anything useful. It exists for testing purposes.
|
// The dummy contract doesn't do anything useful. It exists for testing purposes, but has to be serializable
|
||||||
|
|
||||||
val DUMMY_PROGRAM_ID = DummyContract()
|
val DUMMY_PROGRAM_ID = DummyContract()
|
||||||
|
|
||||||
data class DummyContract(private val blank: Void? = null) : Contract {
|
data class DummyContract(val blank: Any? = null) : Contract {
|
||||||
interface State : ContractState {
|
interface State : ContractState {
|
||||||
val magicNumber: Int
|
val magicNumber: Int
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user