Skip to content
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

Update ADR 020 to address client-side tx UX #6031

Closed
wants to merge 14 commits into from
164 changes: 127 additions & 37 deletions docs/architecture/adr-020-protobuf-transaction-encoding.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
- 2020 March 06: Initial Draft
- 2020 March 12: API Updates
- 2020 April 13: Added details on interface `oneof` handling
- 2020 April 20: Describe improved signing/transaction-submission UX

## Status

Expand All @@ -26,10 +27,10 @@ addressed in a future ADR, but it should build off of these proposals.

## Decision

### Transactions
### Transaction Encoding

Since the messages that an application is known and allowed to handle are specific
to the application itself, so must the transactions be specific to the application
Since the messages that an application knows and is allowed to handle are specific
to the application itself, so must the transaction encoding type be specific to the application
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`.

Expand Down Expand Up @@ -63,7 +64,7 @@ Example:
// app/codec/codec.proto

message Transaction {
cosmos_sdk.x.auth.v1.StdTxBase base = 1;
cosmos_sdk.codec.std.v1.StdTxBase base = 1;
repeated Message msgs = 2;
}
```
Expand All @@ -73,27 +74,59 @@ and includes all the core field members that are common across all transaction t
Developers do not have to include `StdTxBase` if they wish, so it is meant to be
used as an auxiliary type.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What would be the alternative to using StdTxBase? Signatures need to go somewhere. Even slightly changing this type probably has a lot of consequences. Right now the sentence sounds like you can just drop StdTxBase.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is intended to imply that apps can override the std behavior


### Signing
### Signing & Transaction Submission
aaronc marked this conversation as resolved.
Show resolved Hide resolved

Signing of a `Transaction` must be canonical across clients and binaries. In order
to provide canonical representation of a `Transaction` to sign over, clients must
To provide an acceptable user experience for client-side developers, we separate
transaction encoding and signing. While every app defines its own transaction
types for encoding purposes, we define a single standard signing `SignDoc` to be used
across apps that leverages `google.protobuf.Any` to handle interface types. `Any`
aaronc marked this conversation as resolved.
Show resolved Hide resolved
can be used to wrap any protobuf message by encoding both its type URL and value.
We avoid using `Any` during the encoding phase because these URLs can be long
and we want to keep encoded messages small. We are not, however, sending the
`SignDoc` (encoded in JSON) over the wire or storing it on nodes,
therefore using `Any` for signing does not add any encoding overhead.

```proto
// app/codec/codec.proto

message SignDoc {
aaronc marked this conversation as resolved.
Show resolved Hide resolved
StdSignDocBase base = 1;
repeated Message msgs = 2;
}
```

Signing of a `SignDoc` must be canonical across clients and binaries. In order
to provide canonical representation of a `SignDoc` 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).
- Encode `SignDoc` via [Protobuf's canonical JSON encoding](https://developers.google.com/protocol-buffers/docs/proto3#json).
- Default must be stripped from the output!
- JSON keys adhere to their Proto-defined field names.
- Generate canonical JSON to sign via the [JSON Canonical Form Spec](https://gibson042.github.io/canonicaljson-spec/).
- This spec should be trivial to interpret and implement in any language.

```Protobuf
// app/codec/codec.proto
Because signing and encoding are separated and apps will know how to encode
a signing messages, we can provide a gRPC/REST endpoint that allows for generic
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we can provide a gRPC/REST endpoint that allows for generic
app-independent transaction submission. 

I don't understand what this means. Are you referring to a type to be used to generate a JSON blob which when provided to /tx/broadcast will be JSON decoded and then re-encoded using the app-specific tx encoding scheme?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's not json, it's protobuf via gRPC. Basically this AnyTransaction type which is not app dependent

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

And yes, the rpc endpoint would re-encode this in the app level transaction that uses a oneof

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's not json, it's protobuf via gRPC

This makes sense. But in this case, isn't the term "REST" in "we can provide a gRPC/REST endpoint" misleading in this paragraph? I mean RPC and REST are entirely different API concepts and when creating a bridge from REST to gRPC, it would require more explanation.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I still don't follow this @aaronc. I think it can be made more clear or explicit.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will remove the part about REST here and just say gRPC because REST refers to grpc-gateway and that's a separate topic.

app-independent transaction submission. This can be done via a generic
`AnyTransaction` type:

message SignDoc {
StdSignDocBase base = 1;
repeated Message msgs = 2;
```proto
message AnyTransaction {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It's not clear to me what purpose AnyTransaction serves apart from being used during tx signing. How does this fit into tx generation and broadcasting?

Are you stating that module tx endpoints will generate tx JSON blobs using AnyTransaction. Will these JSON blobs then be fed to /tx/broadcast, where they're JSON decoded and then Proto re-encoded?

Please make this very clear.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should be SignDoc+signatures, i.e. a signed transaction in an application agnostic container.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes AnyTransaction can be used for tx broadcasting. It is related to SignDoc but not quite the same. This is basically the same exact design that we currently have with amino. It just involves a re-encoding step to translate Any to the oneof's

cosmos_sdk.codec.std.v1.StdTxBase base = 1;
repeated google.protobuf.Any msgs = 2;
}
```

On the backend, this will be converted to the app-specific `Transaction` type,
with any errors returned for unsupported `Msg`s. Client developers that prefer
to use the app-specific `Transaction` and `Message` `oneof`s can also do this
encoding on their own. This provides additional type safety where needed. There
is, however, no security disadvantage for using the generic RPC transaction
endpoint because all clients regardless of whether they have access to the
app-level proto files will know how to correctly sign transactions. This allows
aaronc marked this conversation as resolved.
Show resolved Hide resolved
for wallets and block explorers to easily add support for new chains, even
discovering them dynamically.

### CLI & REST

Currently, the REST and CLI handlers encode and decode types and txs via Amino
Expand Down Expand Up @@ -137,38 +170,97 @@ 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.
If a module needs to work with `sdk.Msg`s that use interface types, those interface
types should be dealt with using `google.protobuf.Any` for signing and a `oneof`
at the app-level for encoding.

Using `google.protobuf.Any` for signing will allow client libraries to create
reusable code for dealing with interface types that doesn't require regenerating
protobuf client libraries for every chain. Using `oneof`s at the encoding level
aaronc marked this conversation as resolved.
Show resolved Hide resolved
saves space in the Tendermint block store.

Modules should define a generic `sdk.Msg` for interface types at the module
level that uses `Any`. Ex:

```go
// x/gov/types/types.proto
message MsgSubmitAnyProposal {
alexanderbez marked this conversation as resolved.
Show resolved Hide resolved
MsgSubmitProposalBase base = 1;
google.protobuf.Any content = 2;
}
```

For example, in `x/gov`, `Content` is an interface type, so `MsgSubmitProposalI`
should also be an interface and implement setter methods:
Apps should define an app-specific `sdk.Msg` that encodes the concrete types
it supports using an `oneof`:

```go
// x/gov/types/msgs.go
type MsgSubmitProposalI interface {
sdk.Msg
// myapp/types/types.proto
message MsgSubmitProposal {
MsgSubmitProposalBase base = 1;
MyAppContent content = 1;
aaronc marked this conversation as resolved.
Show resolved Hide resolved
}

GetContent() Content
// SetContent returns an error if the underlying oneof does not support
// the concrete Content passed in
SetContent(Content) error
message MyAppContent {
aaronc marked this conversation as resolved.
Show resolved Hide resolved
oneof sum {
TextProposal text = 1;
SomeOtherProposal = 2;
}
}
```

GetInitialDeposit() sdk.Coins
SetInitialDeposit(sdk.Coins)
**Client libraries should always sign transactions using the generic module-level
`sdk.Msg` that uses `Any`.** Convenience gRPC/REST methods will be provided
that allow the generic module-level `Msg` to be used for transaction
submission, converting it to the app-level encoding `Msg` using the `oneof`.

GetProposer() sdk.AccAddress
SetProposer(sdk.AccAddress)
In order to smoothly allow for conversion between _signing_ and _encoding_
`Msg`s, modules that use interface types should implement the following
interface:

```go
type InterfaceMsgEncoder {
GetSigningMsg(sdk.Msg) (sdk.Msg, error)
GetEncodingMsg(sdk.Msg) (sdk.Msg, error)
}
```

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`.
`GetSigningMsg` should type switch over the `sdk.Msg`s that use interfaces and
convert `Msg`s in the encoding format (that use `oneof`)
to the signing version (that uses `Any`). `GetEncodingMsg` should
convert `Msg`s used for signing (using `Any`) to those used for encoding
(using `oneof`). Ex:

```go
// x/gov//module.go
aaronc marked this conversation as resolved.
Show resolved Hide resolved
var _ InterfaceMsgEncoder = AppModule{}

type AppModule struct {
...
newEncodingMsg fn(MsgSubmitProposalBase, Content) (MsgSubmitProposalI, error)
alexanderbez marked this conversation as resolved.
Show resolved Hide resolved
}

func (am AppModule) GetSigningMsg(msg sdk.Msg) (sdk.Msg, error) {
switch msg := msg.(type) {
case MsgSubmitProposalI:
return NewMsgSubmitAnyProposal(msg.GetBase(), msg.GetContent()), nil
default:
return msg, nil
}
}

func (am AppModule) GetEncodingMsg(sdk.Msg) (sdk.Msg, error) {
switch msg := msg.(type) {
case MsgSubmitProposalI:
return am.newEncodingMsg(msg.GetBase(), msg.GetContent())
default:
return msg, nil
}
}
```

A parameter `ctr func() MsgSubmitProposalI` would then be passed to CLI client
methods in order to construct a concrete instance.
Because apps will know how to convert signing `Msg`s to encoding `Msg`s,
signing `Msg`s can and should be used for transaction submission against a
helper function (for CLI methods) or RPC method that does the actually encoding.

## Future Improvements

Expand Down Expand Up @@ -209,8 +301,6 @@ message Message {
- Learning curve required to understand and implement Protobuf messages.
- Less flexibility in cross-module type registration. We now need to define types
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

Expand Down