Merge PR #6111: Update ADR 020 to use Any
This commit is contained in:
parent
513c5f30c2
commit
32f82a93b8
|
@ -5,6 +5,7 @@
|
||||||
- 2020 March 06: Initial Draft
|
- 2020 March 06: Initial Draft
|
||||||
- 2020 March 12: API Updates
|
- 2020 March 12: API Updates
|
||||||
- 2020 April 13: Added details on interface `oneof` handling
|
- 2020 April 13: Added details on interface `oneof` handling
|
||||||
|
- 2020 April 30: Switch to `Any`
|
||||||
|
|
||||||
## Status
|
## Status
|
||||||
|
|
||||||
|
@ -24,76 +25,240 @@ With this in mind, we will tackle the migration path via two main areas, txs and
|
||||||
querying. However, this ADR solely focuses on transactions. Querying should be
|
querying. However, this ADR solely focuses on transactions. Querying should be
|
||||||
addressed in a future ADR, but it should build off of these proposals.
|
addressed in a future ADR, but it should build off of these proposals.
|
||||||
|
|
||||||
|
Based on detailed discussions ([\#6030](https://github.com/cosmos/cosmos-sdk/issues/6030)
|
||||||
|
and [\#6078](https://github.com/cosmos/cosmos-sdk/issues/6078)), the original
|
||||||
|
design for transactions was changed substantially from an `oneof` /JSON-signing
|
||||||
|
approach to the approach described below.
|
||||||
|
|
||||||
## Decision
|
## Decision
|
||||||
|
|
||||||
### Transactions
|
### Transactions
|
||||||
|
|
||||||
Since the messages that an application is known and allowed to handle are specific
|
Since interface values are encoded with `google.protobuf.Any` in state (see [ADR 019](adr-019-protobuf-state-encoding.md)),
|
||||||
to the application itself, so must the transactions be specific to the application
|
`sdk.Msg`s are encoding with `Any` in transactions.
|
||||||
itself. Similar to how we described in [ADR 019](./adr-019-protobuf-state-encoding.md),
|
|
||||||
the concrete types will be defined at the application level via Protobuf `oneof`.
|
One of the main goals of using `Any` to encode interface values is to have a
|
||||||
|
core set of types which is reused by apps so that
|
||||||
|
clients can safely be compatible with as many chains as possible.
|
||||||
|
|
||||||
|
It is one of the goals of this specification to provide a flexible cross-chain transaction
|
||||||
|
format that can serve a wide variety of use cases without breaking client
|
||||||
|
compatibility.
|
||||||
|
|
||||||
The application will define a single canonical `Message` Protobuf message
|
In order to facilitate signing, transactions are separated into `TxBody`,
|
||||||
with a single `oneof` that implements the SDK's `Msg` interface.
|
which will be re-used by `SignDoc` below, and `signatures`:
|
||||||
|
|
||||||
Example:
|
```proto
|
||||||
|
// types/types.proto
|
||||||
|
package cosmos_sdk.v1;
|
||||||
|
|
||||||
```protobuf
|
message Tx {
|
||||||
// app/codec/codec.proto
|
TxBody body = 1;
|
||||||
|
AuthInfo auth_info = 2;
|
||||||
|
repeated bytes signatures = 3;
|
||||||
|
}
|
||||||
|
|
||||||
message Message {
|
message TxBody {
|
||||||
option (cosmos_proto.interface_type) = "github.com/cosmos/cosmos-sdk/types.Msg";
|
repeated google.protobuf.Any messages = 1;
|
||||||
|
string memo = 2;
|
||||||
|
int64 timeout_height = 3;
|
||||||
|
repeated google.protobuf.Any extension_options = 1023;
|
||||||
|
}
|
||||||
|
|
||||||
oneof sum {
|
message AuthInfo {
|
||||||
cosmos_sdk.x.bank.v1.MsgSend msg_send = 1;
|
repeated SignerInfo signer_infos = 1;
|
||||||
cosmos_sdk.x.bank.v1.MsgMultiSend msg_multi_send = 2;
|
// The first signer is the primary signer and the one which pays the fee
|
||||||
cosmos_sdk.x.crisis.v1.MsgVerifyInvariant msg_verify_invariant = 3;
|
Fee fee = 2;
|
||||||
// ...
|
}
|
||||||
}
|
|
||||||
|
message SignerInfo {
|
||||||
|
// PublicKey key is optional for accounts that already exist in state
|
||||||
|
PublicKey public_key = 1;
|
||||||
|
// ModeInfo describes the signing mode of the signer and is a nested
|
||||||
|
// structure to support nested multisig pubkey's
|
||||||
|
ModeInfo mode_info = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message ModeInfo {
|
||||||
|
oneof sum {
|
||||||
|
Single single = 1;
|
||||||
|
Multi multi = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Single is the mode info for a single signer. It is structured as a message
|
||||||
|
// to allow for additional fields such as locale for SIGN_MODE_TEXTUAL in the future
|
||||||
|
message Single {
|
||||||
|
SignMode mode = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Multi is the mode info for a multisig public key
|
||||||
|
message Multi {
|
||||||
|
// bitarray specifies which keys within the multisig are signing
|
||||||
|
CompactBitArray bitarray = 1;
|
||||||
|
// mode_infos is the corresponding modes of the signers of the multisig
|
||||||
|
// which could include nested multisig public keys
|
||||||
|
repeated ModeInfo mode_infos = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum SignMode {
|
||||||
|
SIGN_MODE_UNSPECIFIED = 0;
|
||||||
|
|
||||||
|
SIGN_MODE_DIRECT = 1;
|
||||||
|
|
||||||
|
SIGN_MODE_TEXTUAL = 2;
|
||||||
|
|
||||||
|
SIGN_MODE_LEGACY_AMINO_JSON = 127;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Because an application needs to define it's unique `Message` Protobuf message, it
|
As will be discussed below, in order to include as much of the `Tx` as possible
|
||||||
will by proxy have to define a `Transaction` Protobuf message that encapsulates this
|
in the `SignDoc`, `SignerInfo` is separated from signatures so that only the
|
||||||
`Message` type. The `Transaction` message type must implement the SDK's `Tx` interface.
|
raw signatures themselves live outside of what is signed over.
|
||||||
|
|
||||||
Example:
|
Because we are aiming for a flexible, extensible cross-chain transaction
|
||||||
|
format, new transaction processing options should be added to `TxBody` as soon
|
||||||
|
those use cases are discovered, even if they can't be implemented yet.
|
||||||
|
|
||||||
```protobuf
|
Because there is coordination overhead in this, `TxBody` includes an
|
||||||
// app/codec/codec.proto
|
`extension_options` field which can be used for any transaction processing
|
||||||
|
options that are not already covered. App developers should, nevertheless,
|
||||||
message Transaction {
|
attempt to upstream important improvements to `Tx`.
|
||||||
cosmos_sdk.x.auth.v1.StdTxBase base = 1;
|
|
||||||
repeated Message msgs = 2;
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Note, the `Transaction` type includes `StdTxBase` which will be defined by the SDK
|
|
||||||
and includes all the core field members that are common across all transaction types.
|
|
||||||
Developers do not have to include `StdTxBase` if they wish, so it is meant to be
|
|
||||||
used as an auxiliary type.
|
|
||||||
|
|
||||||
### Signing
|
### Signing
|
||||||
|
|
||||||
Signing of a `Transaction` must be canonical across clients and binaries. In order
|
All of the signing modes below aim to provide the following guarantees:
|
||||||
to provide canonical representation of a `Transaction` to sign over, clients must
|
|
||||||
obey the following rules:
|
|
||||||
|
|
||||||
- Encode `SignDoc` (see below) via [Protobuf's canonical JSON encoding](https://developers.google.com/protocol-buffers/docs/proto3#json).
|
* **No Malleability**: `TxBody` and `AuthInfo` cannot change once the transaction
|
||||||
- Default must be stripped from the output!
|
is signed
|
||||||
- JSON keys adhere to their Proto-defined field names.
|
* **Predictable Gas**: if I am signing a transaction where I am paying a fee,
|
||||||
- Generate canonical JSON to sign via the [JSON Canonical Form Spec](https://gibson042.github.io/canonicaljson-spec/).
|
the final gas is fully dependent on what I am signing
|
||||||
- This spec should be trivial to interpret and implement in any language.
|
|
||||||
|
|
||||||
```Protobuf
|
These guarantees give the maximum amount confidence to message signers that
|
||||||
// app/codec/codec.proto
|
manipulation of `Tx`s by intermediaries can't result in any meaningful changes.
|
||||||
|
|
||||||
|
#### `SIGN_MODE_DIRECT`
|
||||||
|
|
||||||
|
The "direct" signing behavior is to sign the raw `TxBody` bytes as broadcast over
|
||||||
|
the wire. This has the advantages of:
|
||||||
|
|
||||||
|
* requiring the minimum additional client capabilities beyond a standard protocol
|
||||||
|
buffers implementation
|
||||||
|
* leaving effectively zero holes for transaction malleability (i.e. there are no
|
||||||
|
subtle differences between the signing and encoding formats which could
|
||||||
|
potentially be exploited by an attacker)
|
||||||
|
|
||||||
|
Signatures are structured using the `SignDoc` below which reuses `TxBody` and
|
||||||
|
`AuthInfo` and only adds the fields which are needed for signatures:
|
||||||
|
|
||||||
|
```proto
|
||||||
|
// types/types.proto
|
||||||
message SignDoc {
|
message SignDoc {
|
||||||
StdSignDocBase base = 1;
|
TxBody body = 1;
|
||||||
repeated Message msgs = 2;
|
AuthInfo auth_info = 2;
|
||||||
|
string chain_id = 3;
|
||||||
|
uint64 account_number = 4;
|
||||||
|
// account_sequence starts at 1 rather than 0 to avoid the case where
|
||||||
|
// the default 0 value must be omitted in protobuf serialization
|
||||||
|
uint64 account_sequence = 5;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
In order to sign in the default mode, clients take the following steps:
|
||||||
|
|
||||||
|
1. Encode `SignDoc`. (The only requirement of the underlying protobuf
|
||||||
|
implementation is that fields are serialized in order).
|
||||||
|
2. Sign the encoded `SignDoc` bytes
|
||||||
|
3. Build and broadcast `Tx`. (The underlying protobuf implementation must encode
|
||||||
|
`TxBody` and `AuthInfo` with the same binary representation as encoded in
|
||||||
|
`SignDoc`. If this is a problem for clients, the "raw" types described under
|
||||||
|
verification can be used for signing as well.)
|
||||||
|
|
||||||
|
Signature verification is based on comparing the raw `TxBody` and `AuthInfo`
|
||||||
|
bytes encoded in `Tx` not based on any ["canonicalization"](https://github.com/regen-network/canonical-proto3)
|
||||||
|
algorithm which creates added complexity for clients in addition to preventing
|
||||||
|
some forms of upgradeability (to be addressed later in this document).
|
||||||
|
|
||||||
|
Signature verifiers should use a special set of "raw" types to perform this
|
||||||
|
binary signature verification rather than attempting to re-encode protobuf
|
||||||
|
documents which could result in a different binary encoding:
|
||||||
|
|
||||||
|
```proto
|
||||||
|
message TxRaw {
|
||||||
|
bytes body_bytes = 1;
|
||||||
|
repeated bytes signatures = 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
message SignDocRaw {
|
||||||
|
bytes body_bytes = 1;
|
||||||
|
string chain_id = 2;
|
||||||
|
uint64 account_number = 3;
|
||||||
|
uint64 account_sequence = 4;
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### `SIGN_MODE_LEGACY_AMINO`
|
||||||
|
|
||||||
|
In order to support legacy wallets and exchanges, Amino JSON will be temporarily
|
||||||
|
supported transaction signing. Once wallets and exchanges have had a
|
||||||
|
chance to upgrade to protobuf based signing, this option will be disabled. In
|
||||||
|
the meantime, it is foreseen that disabling the current Amino signing would cause
|
||||||
|
too much breakage to be feasible. Note that this is mainly a requirement of the
|
||||||
|
Cosmos Hub and other chains may choose to disable Amino signing immediately.
|
||||||
|
|
||||||
|
Legacy clients will be able to sign a transaction using the current Amino
|
||||||
|
JSON format and have it encoded to protobuf using the REST `/tx/encode`
|
||||||
|
endpoint before broadcasting.
|
||||||
|
|
||||||
|
#### `SIGN_MODE_TEXTUAL`
|
||||||
|
|
||||||
|
As was discussed extensively in [\#6078](https://github.com/cosmos/cosmos-sdk/issues/6078),
|
||||||
|
there is a desire for a human-readable signing encoding, especially for hardware
|
||||||
|
wallets like the [Ledger](https://www.ledger.com) which display
|
||||||
|
transaction contents to users before signing. JSON was an attempt at this but
|
||||||
|
falls short of the ideal.
|
||||||
|
|
||||||
|
`SIGN_MODE_TEXTUAL` is intended as a placeholder for a human-readable
|
||||||
|
encoding which will replace Amino JSON. This new encoding should be even more
|
||||||
|
focused on readability than JSON, possibly based on formatting strings like
|
||||||
|
[MessageFormat](http://userguide.icu-project.org/formatparse/messages).
|
||||||
|
|
||||||
|
In order to ensure that the new human-readable format does not suffer from
|
||||||
|
transaction malleability issues, `SIGN_MODE_TEXTUAL`
|
||||||
|
requires that the _human-readable bytes are concatenated with the raw `SignDoc`_
|
||||||
|
to generate sign bytes.
|
||||||
|
|
||||||
|
Multiple human-readable formats (maybe even localized messages) may be supported
|
||||||
|
by `SIGN_MODE_TEXTUAL` when it is implemented.
|
||||||
|
|
||||||
|
### Unknown Field Filtering
|
||||||
|
|
||||||
|
Unknown fields in protobuf messages should generally be rejected by transaction
|
||||||
|
processors because:
|
||||||
|
|
||||||
|
* important data may be present in the unknown fields, that if ignored, will
|
||||||
|
cause unexpected behavior for clients
|
||||||
|
* they present a malleability vulnerability where attackers can bloat tx size
|
||||||
|
by adding random uninterpreted data to unsigned content (i.e. the master `Tx`,
|
||||||
|
not `TxBody`)
|
||||||
|
|
||||||
|
There are also scenarios where we may choose to safely ignore unknown fields
|
||||||
|
(https://github.com/cosmos/cosmos-sdk/issues/6078#issuecomment-624400188) to
|
||||||
|
provide graceful forwards compatibility with newer clients.
|
||||||
|
|
||||||
|
We propose that field numbers with bit 11 set (for most use cases this is
|
||||||
|
the range of 1024-2047) be considered non-critical fields that can safely be
|
||||||
|
ignored if unknown.
|
||||||
|
|
||||||
|
To handle this we will need a unknown field filter that:
|
||||||
|
* always rejects unknown fields in unsigned content (i.e. top-level `Tx` and
|
||||||
|
unsigned parts of `AuthInfo` if present based on the signing mode)
|
||||||
|
* rejects unknown fields in all messages (including nested `Any`s) other than
|
||||||
|
fields with bit 11 set
|
||||||
|
|
||||||
|
This will likely need to be a custom protobuf parser pass that takes message bytes
|
||||||
|
and `FileDescriptor`s and returns a boolean result.
|
||||||
|
|
||||||
### CLI & REST
|
### CLI & REST
|
||||||
|
|
||||||
Currently, the REST and CLI handlers encode and decode types and txs via Amino
|
Currently, the REST and CLI handlers encode and decode types and txs via Amino
|
||||||
|
@ -135,67 +300,85 @@ Then, each module's client handler will at the minimum accept a `Marshaler` inst
|
||||||
of a concrete Amino codec and a `Generator` along with an `AccountRetriever` so
|
of a concrete Amino codec and a `Generator` along with an `AccountRetriever` so
|
||||||
that account fields can be retrieved for signing.
|
that account fields can be retrieved for signing.
|
||||||
|
|
||||||
#### Interface `oneof` Handling
|
|
||||||
|
|
||||||
If the module needs to work with any `sdk.Msg`s that use interface types, that
|
|
||||||
`sdk.Msg` should be implemented as an interface with getters and setters on the
|
|
||||||
module level and a no-arg constructor function should be passed around to
|
|
||||||
required CLI and REST client commands.
|
|
||||||
|
|
||||||
For example, in `x/gov`, `Content` is an interface type, so `MsgSubmitProposalI`
|
|
||||||
should also be an interface and implement setter methods:
|
|
||||||
|
|
||||||
```go
|
|
||||||
// x/gov/types/msgs.go
|
|
||||||
type MsgSubmitProposalI interface {
|
|
||||||
sdk.Msg
|
|
||||||
|
|
||||||
GetContent() Content
|
|
||||||
// SetContent returns an error if the underlying oneof does not support
|
|
||||||
// the concrete Content passed in
|
|
||||||
SetContent(Content) error
|
|
||||||
|
|
||||||
GetInitialDeposit() sdk.Coins
|
|
||||||
SetInitialDeposit(sdk.Coins)
|
|
||||||
|
|
||||||
GetProposer() sdk.AccAddress
|
|
||||||
SetProposer(sdk.AccAddress)
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Note that the implementation of `MsgSubmitProposalI` can be simplified by
|
|
||||||
using an embedded base struct which implements most of that interface - in this
|
|
||||||
case `MsgSubmitProposalBase`.
|
|
||||||
|
|
||||||
A parameter `ctr func() MsgSubmitProposalI` would then be passed to CLI client
|
|
||||||
methods in order to construct a concrete instance.
|
|
||||||
|
|
||||||
## Future Improvements
|
## Future Improvements
|
||||||
|
|
||||||
Requiring application developers to have to redefine their `Message` Protobuf types
|
### `SIGN_MODE_TEXTUAL` specification
|
||||||
can be extremely tedious and may increase the surface area of bugs by potentially
|
|
||||||
missing one or more messages in the `oneof`.
|
|
||||||
|
|
||||||
To circumvent this, an optional strategy can be taken that has each module define
|
A concrete specification and implementation of `SIGN_MODE_TEXTUAL` is intended
|
||||||
it's own `oneof` and then the application-level `Message` simply imports each module's
|
as a near-term future improvement so that the ledger app and other wallets
|
||||||
`oneof`. However, this requires additional tooling and the use of reflection.
|
can gracefully transition away from Amino JSON.
|
||||||
|
|
||||||
Example:
|
### `SIGN_MODE_DIRECT_AUX`
|
||||||
|
|
||||||
```protobuf
|
(*Documented as option (3) in https://github.com/cosmos/cosmos-sdk/issues/6078#issuecomment-628026933)
|
||||||
// app/codec/codec.proto
|
|
||||||
|
|
||||||
message Message {
|
We could add a mode `SIGN_MODE_DIRECT_AUX`
|
||||||
option (cosmos_proto.interface_type) = "github.com/cosmos/cosmos-sdk/types.Msg";
|
to support scenarios where multiple signatures
|
||||||
|
are being gathered into a single transaction but the message composer does not
|
||||||
|
yet know which signatures will be included in the final transaction. For instance,
|
||||||
|
I may have a 3/5 multisig wallet and want to send a `TxBody` to all 5
|
||||||
|
signers to see who signs first. As soon as I have 3 signatures then I will go
|
||||||
|
ahead and build the full transaction.
|
||||||
|
|
||||||
oneof sum {
|
With `SIGN_MODE_DIRECT`, each signer needs
|
||||||
bank.Msg = 1;
|
to sign the full `AuthInfo` which includes the full list of all signers and
|
||||||
staking.Msg = 2;
|
their signing modes, making the above scenario very hard.
|
||||||
// ...
|
|
||||||
}
|
`SIGN_MODE_DIRECT_AUX` would allow "auxiliary" signers to create their signature
|
||||||
|
using only `TxBody` and their own `PublicKey`. This allows the full list of
|
||||||
|
signers in `AuthInfo` to be delayed until signatures have been collected.
|
||||||
|
|
||||||
|
An "auxiliary" signer is any signer besides the primary signer who is paying
|
||||||
|
the fee. For the primary signer, the full `AuthInfo` is actually needed to calculate gas and fees
|
||||||
|
because that is dependent on how many signers and which key types and signing
|
||||||
|
modes they are using. Auxiliary signers, however, do not need to worry about
|
||||||
|
fees or gas and thus can just sign `TxBody`.
|
||||||
|
|
||||||
|
To generate a signature in `SIGN_MODE_DIRECT_AUX` these steps would be followed:
|
||||||
|
|
||||||
|
1. Encode `SignDocAux` (with the same requirement that fields must be serialized
|
||||||
|
in order):
|
||||||
|
|
||||||
|
```proto
|
||||||
|
// types/types.proto
|
||||||
|
message SignDocAux {
|
||||||
|
bytes body_bytes = 1;
|
||||||
|
// PublicKey is included in SignDocAux :
|
||||||
|
// 1. as a special case for multisig public keys. For multisig public keys,
|
||||||
|
// the signer should use the top-level multisig public key they are signing
|
||||||
|
// against, not their own public key. This is to prevent against a form
|
||||||
|
// of malleability where a signature could be taken out of context of the
|
||||||
|
// multisig key that was intended to be signed for
|
||||||
|
// 2. to guard against scenario where configuration information is encoded
|
||||||
|
// in public keys (it has been proposed) such that two keys can generate
|
||||||
|
// the same signature but have different security properties
|
||||||
|
//
|
||||||
|
// By including it here, the composer of AuthInfo cannot reference the
|
||||||
|
// a public key variant the signer did not intend to use
|
||||||
|
PublicKey public_key = 2;
|
||||||
|
string chain_id = 3;
|
||||||
|
uint64 account_number = 4;
|
||||||
|
// account_sequence starts at 1 rather than 0 to avoid the case where
|
||||||
|
// the default 0 value must be omitted in protobuf serialization
|
||||||
|
uint64 account_sequence = 5;
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
|
2. Sign the encoded `SignDocAux` bytes
|
||||||
|
3. Send their signature and `SignerInfo` to primary signer who will then
|
||||||
|
sign and broadcast the final transaction (with `SIGN_MODE_DIRECT` and `AuthInfo`
|
||||||
|
added) once enough signatures have been collected
|
||||||
|
|
||||||
|
### `SIGN_MODE_DIRECT_RELAXED`
|
||||||
|
|
||||||
|
(*Documented as option (1)(a) in https://github.com/cosmos/cosmos-sdk/issues/6078#issuecomment-628026933*)
|
||||||
|
|
||||||
|
This is a variation of `SIGN_MODE_DIRECT` where multiple signers wouldn't need to
|
||||||
|
coordinate public keys and signing modes in advance. It would involve an alternate
|
||||||
|
`SignDoc` similar to `SignDocAux` above with fee. This could be added in the future
|
||||||
|
if client developers found the burden of collecting public keys and modes in advance
|
||||||
|
too burdensome.
|
||||||
|
|
||||||
## Consequences
|
## Consequences
|
||||||
|
|
||||||
### Positive
|
### Positive
|
||||||
|
@ -203,14 +386,12 @@ message Message {
|
||||||
- Significant performance gains.
|
- Significant performance gains.
|
||||||
- Supports backward and forward type compatibility.
|
- Supports backward and forward type compatibility.
|
||||||
- Better support for cross-language clients.
|
- Better support for cross-language clients.
|
||||||
|
- Multiple signing modes allow for greater protocol evolution
|
||||||
|
|
||||||
### Negative
|
### Negative
|
||||||
|
|
||||||
- Learning curve required to understand and implement Protobuf messages.
|
- `google.protobuf.Any` type URLs increase transaction size although the effect
|
||||||
- Less flexibility in cross-module type registration. We now need to define types
|
may be negligible or compression may be able to mitigate it.
|
||||||
at the application-level.
|
|
||||||
- Client business logic and tx generation become a bit more complex as developers
|
|
||||||
have to define more types and implement more interfaces.
|
|
||||||
|
|
||||||
### Neutral
|
### Neutral
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue