-
Notifications
You must be signed in to change notification settings - Fork 228
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
Serialization Concerns #197
Comments
So true, independent of this repo 😆 |
A common technique I've seen elsewhere for dealing with encoding/decoding of complex types (for example, in Cargo) is separating the "domain" types (i.e. the ideal ones you want to use) from the ones which handle encoding, then having |
In this context: instead of having currently two versions of a type (one for JSON another for amino/binary encoding), we could merge these two into one. Pros:
Cons:
I know we discussed this a while ago but I can't really remember if we were in favour of "merging" serialization types or not. |
I think we were against it for now given the idiosyncracies of the two formats. If we manage to consolidate everything to a standard proto/proto-JSON format for which there is already robust library support, then maybe we can revisit consolidating. Another note, in my understanding, a single Rust struct cannot be encoded two different ways depending on the encoding. I could be wrong but I thought it mapped down to a common intermediate representation so if you have strings encoded one way for proto and another way for JSON you have no choice but to use distinct types ... |
Well if both encodings implement A quick try confirms that this at least compiles: I'm not in favour of consolidating the different structs though. |
For IBC, as we move to proto3 for all queries and messages, I would be strongly in favor of using a single types/ structs derived from the .proto and making sure the canonical encodings in JSON (as per https://developers.google.com/protocol-buffers/docs/proto3#json) are accepted by both Rust and Go impl. This is of course supported in Go. |
@ebuchman brought up that we don't yet have a full story for JSON decoding. It is clear that we can and already use serde for JSON. But there is also the amino style type/value-style encoding for registered types (see this snippet for an example). The closest to this is the Any encoding in proto3 JSON mappings but it has minor differences (e.g. the It should be really easy (no really) to write a decoder/encoder for these though which we can use for all cases (without code duplication). I will draft a little parametrizable struct. @marbar3778 was the plan to replace the amino JSON with vanilla proto JSON mapping? I only familiar on how the binary encoding changes with look like but I didn't look into how JSON will be handled. |
@ebuchman in any case we could use sth like this: tendermint-rs/tendermint/src/amino_types/registered_json.rs Lines 16 to 23 in 720d841
Or without tendermint-rs/tendermint/src/amino_types/registered_json.rs Lines 22 to 28 in 440bc49
Actually much simpler and more idiomatic would be using serde's enum representations: e.g., https://serde.rs/enum-representations.html#adjacently-tagged tendermint-rs/tendermint/src/amino_types/registered_json.rs Lines 30 to 37 in 3c4f23f
|
Assigning greg here as he is already working on serialization improvements (#248 ). |
Note that in tendermint amino JSON encoding will go full "proto mapping" encoding tendermint issue: tendermint/tendermint#4828 This might be relevant to some of the work done in #247 too. cc @marbar3778 |
Particularly relevant for the current JSON work: tendermint/tendermint#4828 (comment) |
I think we've been making progress on this with the recent upgrade to v0.33 and the upcoming upgrade to v0.34 of Tendermint. Maybe we can close this? @greg-szabo ? |
Let's finish the JSON part of 0.34 (sometime next week) and then we can close this. I just want to make sure. ;) |
@greg-szabo should we close this now? Note I opened #567 to track brining back JSON tests. There might be more here but we have a pretty comprehensive domain type system and can follow up with more concrete things as we need them. |
Lets close this for #567 |
Serialization issues have been a constant plague upon us.
At a high level, virtually all our data structures must serialize to/from both Amino and JSON.
The main data structures target JSON, while copies of the data structures in the amino_types module target Amino.
A lot of the difficulties come from peculiarities in Tendermint JSON encoding, especially around empty/missing/nil values. In particular:
null
omitempty
)In certain cases, multiple fields in a struct are all present or all absent together. We should consider making such types enums with Present and Absent variants - the Present variant requires all the fields (except the genuinely Optional ones), and the Absent variant omits all the fields that are omitted together. This is for instance the case with the Block struct - if it's the first block, all the data pertaining to previous blocks is omitted, but if its any other block, it all must be there #101 .
We don't seem to currently have a coherent/standard way to deal with all this complexity. In some cases we use
Option
in structs. In some cases we use custom serialize/deserialize functions. In some cases we use custom types that implement their own serialize/deserialize methods. It would be worth attempting some kind of inventory of all this special casing and try to achieve more consistency and simplification.While we are aiming for compliance with Tendermint Core v0.33, there may be cases where its worth recommending a change to Tendermint Core to make this more manageable.
Related:
block::header::Header
do not match #187The text was updated successfully, but these errors were encountered: