-
Notifications
You must be signed in to change notification settings - Fork 629
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: Add a way to reorder elements during encoding #2722
base: dev
Are you sure you want to change the base?
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A good starting point, but some issues, including a lack of proper support for nested structs (or a nested struct to be wrapped in a value class).
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
private val mapElementIndex: (SerialDescriptor, Int) -> Int, | ||
) : CompositeEncoder { | ||
// Each time we encode a field, if the next expected schema field index is not the good one, it is buffered until it's the time to encode it | ||
private var bufferedCalls = Array<BufferedCall?>(structureDescriptor.elementsCount) { null } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This cache means you need to create an instance at least per structure descriptor, but more conveniently for every instance. In such case beginStructure
can just become private val delegateEncoder: CompositeEncoder
.
Although it may also make sense to have this retained (and used for subStructures, although in such case the element index would be an important parameter)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For the first part, I suppose you were talking about the supplier that should just be the encoder (no supplier), and I agree.
But for the second part, I don't understand, the buffer is just for the corresponding descriptor, that should not be shared with sub-structure: we are reordering one level of a given descriptor, but not the sub structures.
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
override fun beginStructure(descriptor: SerialDescriptor): CompositeEncoder { | ||
invalidCall("beginStructure") | ||
} | ||
|
||
override fun encodeInline(descriptor: SerialDescriptor): Encoder { | ||
invalidCall("encodeInline") | ||
} | ||
|
||
private fun invalidCall(methodName: String): Nothing { | ||
// This method is normally called by encodeSerializableValue or encodeNullableSerializableValue that is buffered, so we should never go here during buffering as it will be delegated to the concrete CompositeEncoder | ||
throw UnsupportedOperationException("$methodName should not be called when reordering fields, as it should be done by encodeSerializableValue or encodeNullableSerializableValue") | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is incorrect. It is valid to have an inline/value class have a struct member. That would just result in a nested reordering encoder (delay-created with the result of calling beginStructure
).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The beginStructure is called by the serializer from inside the encodeSerializableValue
, so it should never be called directly AFAIK 🤔
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What I mean is that you have an encoder on which beginStructure
is called. In the implementation it will create/call the reordering composite encoder (and return it), and perhaps also to the delegate encoder (to be passed to the reordering encoder).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok I got it. Is it such an expected standard use case to encode inlined structures like this encoder.encodeInlineElement(desc, idx).beginStructure(...)
? Shouldn't it be expected to be like the serializer plugin (encoder.encodeInlineElement(desc, idx).encodeSerializableValue(...)
) where structures are always encoded through encodeSerializableValue
or encodeNullableSerializableValue
to allow interception and support nullability ?
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
override fun encodeInlineElement(descriptor: SerialDescriptor, index: Int): Encoder { | ||
return BufferingInlineEncoder(descriptor, index) | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note that the implementation of this should be equivalent to:
thisEncoder. encodeSerializableElement(MyData. serializer. descriptor, 0, MyInt. serializer(), myInt)
Including the support of nested structs (nested structs always go through encodeSerializableElement
)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, this is working like this after reordering, or maybe I misunderstood
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What about:
@Serializable data class Outer(val a: Middle, val b: Int)
@Serializable value class Middle(val i: Inner)
@Serializable value class Inner(val c: String, d: Boolean)
The way I read it, this at least can be implemented using encodeInlineElement(d, 0)
on the composite encoder for outer. Which would then mean that the inner value is serialized by calling encodeSerializableValue()
on that (BufferingInlineEncoder
) encoder.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Test has been successful with your example, so I still don't get your point https://github.com/Kotlin/kotlinx.serialization/pull/2722/files#r1648164815
the inner value is serialized by calling encodeSerializableValue() on that (BufferingInlineEncoder) encoder
No, here are the steps:
encodeInlineElement(d, 0)
-> returnsBufferingInlineEncoder
with descriptor ofMiddle
and index 0, but does not buffer anything for the moment (this is the role ofBufferingInlineEncoder
)BufferingInlineEncoder.encodeSerializableValue
is called, so nothing more is done, no nested call is done even forInner
thanks tobufferEncoding
bufferingcompositeEncoderDelegate.encodeInlineElement(descriptor, elementIndex).encodeSerializableValue(serializer, value)
- then when
endStructure
is called, replays the elements encoding, so theInner
will be reached out only at this moment
core/commonMain/src/kotlinx/serialization/encoding/ReorderingCompositeEncoder.kt
Outdated
Show resolved
Hide resolved
First, I did not cover everything as I wanted first to agree on the main idea of buffering the calls. It seems that it's ok for you 😄 Secondly, if it's ok for you, I'll focus on the buffering itself, and then refine the public API and namings. I've did the current work quickly to show you the idea. To finish, thanks for the quick answer 🚀 I'll make some tests to test the different use cases |
@Chuckame The buffering is basically correct (I've implemented this for XML for quite some time now) |
core/commonTest/src/kotlinx/serialization/encoding/ReorderingCompositeEncoderTest.kt
Outdated
Show resolved
Hide resolved
e8bb6b8
to
ef214ec
Compare
@pdvrieze I've tried with all the possible combinations, and value classes serializers are not directly using Let's imagine that this bug doesn't exist, and /**
* [...]
*
* Calling [Encoder.beginStructure] on returned instance leads to an unspecified behavior and, in general, is prohibited.
*/
public fun encodeInline(descriptor: SerialDescriptor): Encoder So I would say that this use case shouldn't be handled to follow the docs, as this |
ef214ec
to
bb9f94e
Compare
bb9f94e
to
e067afd
Compare
Closes #2668