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:
Viktor Kolomeyko 2017-08-23 17:37:49 +01:00 committed by GitHub
parent 0c65e5e708
commit d6d7fb52b4
8 changed files with 68 additions and 25 deletions

View File

@ -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)
}
}

View File

@ -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))
} }
} }
} }

View File

@ -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
} }

View File

@ -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) {

View File

@ -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

View File

@ -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.")
}() }()
} }

View File

@ -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)
}

View File

@ -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
} }