CORDA-1498 - serialization multiple transform bug (#3391)

* CORDA-1498: serialization multiple transform bug (#3216)

* Fix issue when evolving enums with transformation chains

* Regenerate test data for deserializeWithRename test and unignore

* Further tweaks / remove debugging

* Formatting tweaks

* Address review comments

* Remove debug

* Add classname to serialization tranform exceptions

* Use direct node links instead of indexes to improve readability

* More readability tweaks

* More readability improvements

* rename require to requireThat to resolve conflict with kotlin libraries

* Add logging of error message

* Change requireThat helper to inline function

* remove unneeded toString

* Further tweaks

* Change NotSerializableException to more generic IOException

* Make exception context clearer

# Conflicts:
#	node-api/src/test/kotlin/net/corda/nodeapi/internal/serialization/amqp/EnumEvolvabilityTests.kt
#	serialization/src/test/resources/net/corda/serialization/internal/amqp/EnumEvolveTests.deserializeWithRename.1.C
#	serialization/src/test/resources/net/corda/serialization/internal/amqp/EnumEvolveTests.deserializeWithRename.2.C
#	serialization/src/test/resources/net/corda/serialization/internal/amqp/EnumEvolveTests.deserializeWithRename.3.C

* Fix merge conflicts

* Fix broken test

* Revert changes to serialized classes
This commit is contained in:
Anthony Keenan 2018-06-22 15:53:18 +01:00 committed by Katelyn Baker
parent 371031dd3b
commit 5be8c9a102
4 changed files with 154 additions and 37 deletions

View File

@ -0,0 +1,14 @@
package net.corda.nodeapi.internal.serialization
import java.io.IOException
import java.io.NotSerializableException
class NotSerializableDetailedException(classname: String?, val reason: String) : NotSerializableException(classname) {
override fun toString(): String {
return "Unable to serialize/deserialize $message: $reason"
}
}
// This exception is thrown when serialization isn't possible but at the point the exception
// is thrown the classname isn't known. It's caught and rethrown as a [NotSerializableDetailedException]
class NotSerializableWithReasonException(message: String?): IOException(message)

View File

@ -4,9 +4,9 @@ import net.corda.core.internal.uncheckedCast
import net.corda.core.serialization.CordaSerializationTransformEnumDefault import net.corda.core.serialization.CordaSerializationTransformEnumDefault
import net.corda.core.serialization.CordaSerializationTransformEnumDefaults import net.corda.core.serialization.CordaSerializationTransformEnumDefaults
import net.corda.core.serialization.CordaSerializationTransformRename import net.corda.core.serialization.CordaSerializationTransformRename
import net.corda.nodeapi.internal.serialization.NotSerializableWithReasonException
import org.apache.qpid.proton.amqp.DescribedType import org.apache.qpid.proton.amqp.DescribedType
import org.apache.qpid.proton.codec.DescribedTypeConstructor import org.apache.qpid.proton.codec.DescribedTypeConstructor
import java.io.NotSerializableException
/** /**
* Enumerated type that represents each transform that can be applied to a class. Used as the key type in * Enumerated type that represents each transform that can be applied to a class. Used as the key type in
@ -45,25 +45,12 @@ enum class TransformTypes(val build: (Annotation) -> Transform) : DescribedType
*/ */
override fun validate(list: List<Transform>, constants: Map<String, Int>) { override fun validate(list: List<Transform>, constants: Map<String, Int>) {
uncheckedCast<List<Transform>, List<EnumDefaultSchemaTransform>>(list).forEach { uncheckedCast<List<Transform>, List<EnumDefaultSchemaTransform>>(list).forEach {
if (!constants.contains(it.new)) { requireThat(constants.contains(it.new)) {"Unknown enum constant ${it.new}"}
throw NotSerializableException("Unknown enum constant ${it.new}") requireThat(constants.contains(it.old)) { "Enum extension defaults must be to a valid constant: ${it.new} -> ${it.old}. ${it.old} " +
} "doesn't exist in constant set $constants" }
requireThat(it.old != it.new) { "Enum extension ${it.new} cannot default to itself" }
if (!constants.contains(it.old)) { requireThat(constants[it.old]!! < constants[it.new]!!) { "Enum extensions must default to older constants. ${it.new}[${constants[it.new]}] " +
throw NotSerializableException( "defaults to ${it.old}[${constants[it.old]}] which is greater" }
"Enum extension defaults must be to a valid constant: ${it.new} -> ${it.old}. ${it.old} " +
"doesn't exist in constant set $constants")
}
if (it.old == it.new) {
throw NotSerializableException("Enum extension ${it.new} cannot default to itself")
}
if (constants[it.old]!! >= constants[it.new]!!) {
throw NotSerializableException(
"Enum extensions must default to older constants. ${it.new}[${constants[it.new]}] " +
"defaults to ${it.old}[${constants[it.old]}] which is greater")
}
} }
} }
}, },
@ -81,21 +68,56 @@ enum class TransformTypes(val build: (Annotation) -> Transform) : DescribedType
* @param constants The list of enum constants on the type the transforms are being applied to * @param constants The list of enum constants on the type the transforms are being applied to
*/ */
override fun validate(list: List<Transform>, constants: Map<String, Int>) { override fun validate(list: List<Transform>, constants: Map<String, Int>) {
object : Any() { data class Node(val transform: RenameSchemaTransform, var next: Node?, var prev: Node?, var visitedBy: Node? = null) {
val from: MutableSet<String> = mutableSetOf() fun visit(visitedBy: Node) {
val to: MutableSet<String> = mutableSetOf() this.visitedBy = visitedBy
}.apply { }
@Suppress("UNCHECKED_CAST") (list as List<RenameSchemaTransform>).forEach { rename -> val visited get() = visitedBy != null
if (rename.to in this.to || rename.from in this.from) { }
throw NotSerializableException("Cyclic renames are not allowed (${rename.to})")
}
this.to.add(rename.from) val graph = mutableListOf<Node>()
this.from.add(rename.to) // Keep two maps of forward links and back links in order to build the graph in one pass
val forwardLinks = hashMapOf<String, Node>()
val reverseLinks = hashMapOf<String, Node>()
// build a dependency graph
val transforms: List<RenameSchemaTransform> = uncheckedCast(list)
transforms.forEach { rename ->
requireThat(!forwardLinks.contains(rename.from)) { "There are multiple transformations from ${rename.from}, which is not allowed" }
requireThat(!reverseLinks.contains(rename.to)) { "There are multiple transformations to ${rename.to}, which is not allowed" }
val node = Node(rename, forwardLinks[rename.to], reverseLinks[rename.from])
graph.add(node)
node.next?.prev = node
node.prev?.next = node
forwardLinks[rename.from] = node
reverseLinks[rename.to] = node
}
// Check that every property in the current type is at the end of a renaming chain, if it is in one
constants.keys.forEach {
requireThat(reverseLinks[it]?.next == null) { "$it is specified as a previously evolved type, but it also exists in the current type" }
}
// Check for cyclic dependencies
graph.forEach {
if (it.visited) return@forEach
// Find an unvisited node
var currentNode = it
currentNode.visit(it)
while (currentNode.next != null) {
currentNode = currentNode.next!!
if (currentNode.visited) {
requireThat(currentNode.visitedBy != it) { "Cyclic renames are not allowed (${currentNode.transform.from})" }
// we have found the start of another non-cyclic chain of dependencies
// if they were cyclic we would have gone round in a loop and already thrown
break
}
currentNode.visit(it)
} }
} }
} }
} }
// Transform used to test the unknown handler, leave this at as the final constant, uncomment // Transform used to test the unknown handler, leave this at as the final constant, uncomment
// when regenerating test cases - if Java had a pre-processor this would be much neater // when regenerating test cases - if Java had a pre-processor this would be much neater
// //
@ -118,9 +140,7 @@ enum class TransformTypes(val build: (Annotation) -> Transform) : DescribedType
override fun newInstance(obj: Any?): TransformTypes { override fun newInstance(obj: Any?): TransformTypes {
val describedType = obj as DescribedType val describedType = obj as DescribedType
if (describedType.descriptor != DESCRIPTOR) { requireThat(describedType.descriptor == DESCRIPTOR) { "Unexpected descriptor ${describedType.descriptor}." }
throw NotSerializableException("Unexpected descriptor ${describedType.descriptor}.")
}
return try { return try {
values()[describedType.described as Int] values()[describedType.described as Int]
@ -130,5 +150,11 @@ enum class TransformTypes(val build: (Annotation) -> Transform) : DescribedType
} }
override fun getTypeClass(): Class<*> = TransformTypes::class.java override fun getTypeClass(): Class<*> = TransformTypes::class.java
protected inline fun requireThat(expr: Boolean, errorMessage: () -> String) {
if (!expr) {
throw NotSerializableWithReasonException(errorMessage())
}
}
} }
} }

View File

@ -2,6 +2,10 @@ package net.corda.nodeapi.internal.serialization.amqp
import net.corda.core.serialization.CordaSerializationTransformEnumDefault import net.corda.core.serialization.CordaSerializationTransformEnumDefault
import net.corda.core.serialization.CordaSerializationTransformRename import net.corda.core.serialization.CordaSerializationTransformRename
import net.corda.core.utilities.contextLogger
import net.corda.core.utilities.trace
import net.corda.nodeapi.internal.serialization.NotSerializableDetailedException
import net.corda.nodeapi.internal.serialization.NotSerializableWithReasonException
import org.apache.qpid.proton.amqp.DescribedType import org.apache.qpid.proton.amqp.DescribedType
import org.apache.qpid.proton.codec.DescribedTypeConstructor import org.apache.qpid.proton.codec.DescribedTypeConstructor
import java.io.NotSerializableException import java.io.NotSerializableException
@ -191,6 +195,7 @@ class RenameSchemaTransform(val from: String, val to: String) : Transform() {
data class TransformsSchema(val types: Map<String, EnumMap<TransformTypes, MutableList<Transform>>>) : DescribedType { data class TransformsSchema(val types: Map<String, EnumMap<TransformTypes, MutableList<Transform>>>) : DescribedType {
companion object : DescribedTypeConstructor<TransformsSchema> { companion object : DescribedTypeConstructor<TransformsSchema> {
val DESCRIPTOR = AMQPDescriptorRegistry.TRANSFORM_SCHEMA.amqpDescriptor val DESCRIPTOR = AMQPDescriptorRegistry.TRANSFORM_SCHEMA.amqpDescriptor
private val logger = contextLogger()
/** /**
* Takes a class name and either returns a cached instance of the TransformSet for it or, on a cache miss, * Takes a class name and either returns a cached instance of the TransformSet for it or, on a cache miss,
@ -240,10 +245,17 @@ data class TransformsSchema(val types: Map<String, EnumMap<TransformTypes, Mutab
type: String, type: String,
sf: SerializerFactory, sf: SerializerFactory,
map: MutableMap<String, EnumMap<TransformTypes, MutableList<Transform>>>) { map: MutableMap<String, EnumMap<TransformTypes, MutableList<Transform>>>) {
get(type, sf).apply { try {
if (isNotEmpty()) { get(type, sf).apply {
map[type] = this if (isNotEmpty()) {
map[type] = this
}
} }
} catch (e: NotSerializableWithReasonException) {
val message = "Error running transforms for $type: ${e.message}"
logger.error(message)
logger.trace { e.toString() }
throw NotSerializableDetailedException(type, e.message ?: "")
} }
} }

View File

@ -1,8 +1,11 @@
package net.corda.nodeapi.internal.serialization.amqp package net.corda.nodeapi.internal.serialization.amqp
import net.corda.core.serialization.* import net.corda.core.serialization.*
import net.corda.nodeapi.internal.serialization.NotSerializableDetailedException
import net.corda.testing.common.internal.ProjectStructure.projectRootDir import net.corda.testing.common.internal.ProjectStructure.projectRootDir
import org.assertj.core.api.Assertions import org.assertj.core.api.Assertions
import org.assertj.core.api.Assertions.assertThat
import org.assertj.core.api.Assertions.assertThatThrownBy
import org.junit.Test import org.junit.Test
import java.io.File import java.io.File
import java.io.NotSerializableException import java.io.NotSerializableException
@ -441,6 +444,69 @@ class EnumEvolvabilityTests {
assertTrue(envelope.transformsSchema.types[WithUnknownTest::class.java.name]!!.containsKey(TransformTypes.Unknown)) assertTrue(envelope.transformsSchema.types[WithUnknownTest::class.java.name]!!.containsKey(TransformTypes.Unknown))
} }
//
// In this test we check that multiple transforms of a property are accepted
//
@CordaSerializationTransformRenames(
CordaSerializationTransformRename(from = "A", to = "B"),
CordaSerializationTransformRename(from = "B", to = "C")
)
enum class AcceptMultipleRename { C }
@Test
fun acceptMultipleRename() {
data class C(val e: AcceptMultipleRename)
val sf = testDefaultFactory()
SerializationOutput(sf).serialize(C(AcceptMultipleRename.C))
}
//
// In this example we will try to rename two different things to the same thing,
// which is not allowed
//
@CordaSerializationTransformRenames(
CordaSerializationTransformRename(from = "D", to = "C"),
CordaSerializationTransformRename(from = "E", to = "C")
)
enum class RejectMultipleRenameTo { A, B, C }
@Test
fun rejectMultipleRenameTo() {
data class C(val e: RejectMultipleRenameTo)
val sf = testDefaultFactory()
assertThatThrownBy {
SerializationOutput(sf).serialize(C(RejectMultipleRenameTo.A))
}.isInstanceOfSatisfying(NotSerializableDetailedException::class.java) { ex ->
assertThat(ex.reason).isEqualToIgnoringCase("There are multiple transformations to C, which is not allowed")
assertThat(ex.message).endsWith(RejectMultipleRenameTo::class.simpleName)
}
}
//
// In this example we will try to rename two different things from the same thing,
// which is not allowed
//
@CordaSerializationTransformRenames(
CordaSerializationTransformRename(from = "D", to = "C"),
CordaSerializationTransformRename(from = "D", to = "B")
)
enum class RejectMultipleRenameFrom { A, B, C }
@Test
fun rejectMultipleRenameFrom() {
data class C(val e: RejectMultipleRenameFrom)
val sf = testDefaultFactory()
assertThatThrownBy {
SerializationOutput(sf).serialize(C(RejectMultipleRenameFrom.A))
}.isInstanceOfSatisfying(NotSerializableDetailedException::class.java) { ex ->
assertThat(ex.reason).isEqualToIgnoringCase("There are multiple transformations from D, which is not allowed")
assertThat(ex.message).endsWith(RejectMultipleRenameFrom::class.simpleName)
}
}
// //
// In this example we will have attempted to rename D back to C // In this example we will have attempted to rename D back to C
// //
@ -533,5 +599,4 @@ class EnumEvolvabilityTests {
SerializationOutput(sf).serialize(C(RejectBadDefaultToSelf.D)) SerializationOutput(sf).serialize(C(RejectBadDefaultToSelf.D))
}.isInstanceOf(NotSerializableException::class.java) }.isInstanceOf(NotSerializableException::class.java)
} }
} }