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

ADR 023 Protobuf Naming and Versioning #6083

Merged
merged 25 commits into from
Jun 5, 2020
Merged
Show file tree
Hide file tree
Changes from 16 commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
d07ba38
First work on ADR 023
aaronc Apr 27, 2020
c1351d6
WIP on ADR 023
aaronc Apr 27, 2020
ebdf9b3
Fill out descriptions
aaronc Apr 27, 2020
d2f597b
Merge branch 'master' into aaronc/proto-naming
aaronc Apr 27, 2020
ed51ec5
Review test
aaronc Apr 27, 2020
8969b03
Merge remote-tracking branch 'origin/aaronc/proto-naming' into aaronc…
aaronc Apr 27, 2020
967977d
Updates
aaronc Apr 27, 2020
c7b5043
Update docs/architecture/adr-023-protobuf-naming.md
aaronc Apr 28, 2020
165c3b5
Merge branch 'master' into aaronc/proto-naming
aaronc Apr 28, 2020
f30d01f
Merge branch 'master' into aaronc/proto-naming
aaronc Apr 29, 2020
2866b1a
Add ICS name suggestion
aaronc May 7, 2020
e15f036
cosmos.libs -> base, mention alias
aaronc May 14, 2020
55a907c
Merge branch 'master' of github.com:cosmos/cosmos-sdk into aaronc/pro…
aaronc May 14, 2020
ff681d8
fix diff
aaronc May 19, 2020
9f2ea3d
Merge branch 'master' of github.com:cosmos/cosmos-sdk into aaronc/pro…
aaronc May 19, 2020
a2e113e
Merge branch 'master' into aaronc/proto-naming
fedekunze May 21, 2020
b3ade67
Update docs/architecture/adr-023-protobuf-naming.md
alexanderbez Jun 4, 2020
0e31e52
Merge branch 'master' into aaronc/proto-naming
alexanderbez Jun 4, 2020
0bbcd51
Update docs/architecture/adr-023-protobuf-naming.md
aaronc Jun 4, 2020
16721a7
Update version suffixing
aaronc Jun 4, 2020
cbc40ba
Merge branch 'aaronc/proto-naming' of github.com:cosmos/cosmos-sdk in…
aaronc Jun 4, 2020
13d6adf
Merge branch 'master' of github.com:cosmos/cosmos-sdk into aaronc/pro…
aaronc Jun 4, 2020
e4941ae
Describe RPC req/res type names
aaronc Jun 4, 2020
f07daab
Retain Msg prefix
aaronc Jun 5, 2020
66087f7
Merge branch 'master' into aaronc/proto-naming
aaronc Jun 5, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions docs/architecture/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -48,3 +48,4 @@ Please add a entry below in your Pull Request for an ADR.
- [ADR 020: Protocol Buffer Transaction Encoding](./adr-020-protobuf-transaction-encoding.md)
- [ADR 021: Protocol Buffer Query Encoding](./adr-021-protobuf-query-encoding.md)
- [ADR 022: Custom baseapp panic handling](./adr-022-custom-panic-handling.md)
- [ADR 023: Protocol Buffer Naming and Versioning](./adr-023-protobuf-naming.md)
238 changes: 238 additions & 0 deletions docs/architecture/adr-023-protobuf-naming.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,238 @@
# ADR 023: Protocol Buffer Naming and Versioning Conventions

## Changelog

- 2020 April 27: Initial Draft

## Status

Proposed

## Context

Protocol Buffers provide a basic [style guide](https://developers.google.com/protocol-buffers/docs/style)
and [Buf](https://buf.build/docs/style-guide) builds upon that. To the
extent possible, we want to follow industry accepted guidelines and wisdom for
the effective usage of protobuf, deviating from those only when there is clear
rationale for our use case.

### Adoption of `Any`

The adoption of `google.protobuf.Any` as the recommended approach for encoding
interface types (as opposed to `oneof`) makes package naming a central part
of the encoding as fully-qualified message names now appear in encoded
transactions.
aaronc marked this conversation as resolved.
Show resolved Hide resolved

### Current Directory Organization

Thus far we have mostly followed [Buf's](https://buf.build) [DEFAULT](https://buf.build/docs/lint-checkers#default)
recommendations, with the minor deviation of disabling [`PACKAGE_DIRECTORY_MATCH`](https://buf.build/docs/lint-checkers#file_layout)
which although being convenient for developing code comes with the warning
from Buf that:

> you will have a very bad time with many Protobuf plugins across various languages if you do not do this

### Adoption of gRPC Queries

In [ADR 021](adr-021-protobuf-query-encoding.md), gRPC was adopted for Protobuf
native queries. The full gRPC service path thus becomes a key part of ABCI query
path. In the future, gRPC queries may be allowed from within persistent scripts
by technologies such as CosmWasm and these query routes would be stored within
script binaries.

## Decision

The goal of this ADR is to provide thoughtful naming conventions that
alexanderbez marked this conversation as resolved.
Show resolved Hide resolved

* encourage a good user experience for when users interact directly with
.proto files and fully-qualified protobuf names
* balance conciseness against the possibility of either over-optimizing (making
names too short and cryptic) or under-optimizing (just accepting bloated names
with lots of redundant information)

These guidelines are meant to act as a style guide for both the SDK and
third-party modules.

As a starting point, we should adopt all of the [DEFAULT](https://buf.build/docs/lint-checkers#default)
checkers in [Buf's](https://buf.build) including [`PACKAGE_DIRECTORY_MATCH`](https://buf.build/docs/lint-checkers#file_layout),
except:
* [PACKAGE_VERSION_SUFFIX](https://buf.build/docs/lint-checkers#package_version_suffix)
* [SERVICE_SUFFIX](https://buf.build/docs/lint-checkers#service_suffix)

Further guidelines to be described below.

### Principles

#### Concise and Descriptive Names

Names should be descriptive enough to convey their meaning and distinguish
them from other names.

Given that we are using fully-qualifed names within
`google.protobuf.Any` as well as within gRPC query routes, we should aim to
keep names concise, without going overboard. The general rule of thumb should
be if a shorter name would convey more or else the same thing, pick the shorter
name.

For instance, `cosmos.bank.Send` (16 bytes) conveys roughly the same information
aaronc marked this conversation as resolved.
Show resolved Hide resolved
as `cosmos_sdk.x.bank.v1.MsgSend` (28 bytes) but is notably shorter.
If we needed version 2 of bank, we could just write `cosmos.bank2.Send`.

Such conciseness makes names both more pleasant to work with and take up less
space within transactions and on the wire.

We should also resist the temptation to over-optimize, by making names
cryptically short with abbreviations. For instance, we shouldn't try to
reduce `cosmos.bank.Send` to `csm.bk.Snd` just to save six bytes.

The goal is to make names **_concise but not cryptic_**.

#### Names are for Clients First

Package and type names should be chosen for the benefit of users, not
necessarily because of legacy concerns related to the go code-base.

#### Plan for Longevity

In the interests of long-term support, we should plan on the names we do
choose to be in usage for a long time, so now is the opportunity to make
the best choices for the future.

### Versioning

#### Don't Allow Breaking Changes in Stable Packages

Always use a breaking change detector such as [Buf](https://buf.build) to prevent
breaking changes in stable (non-alpha or beta) packages. Breaking changes can
break smart contracts/persistent scripts and generally provide a bad UX for
clients. With protobuf, there should usually be ways to extend existing
functionality instead of just breaking it.

#### Use Simple Version Suffixes

Instead of using [Buf's recommended version suffix](https://buf.build/docs/lint-checkers#package_version_suffix),
we can more concisely indicate version 2, 3, etc. of a package with a simple
package suffix rather than a sub-package, ex. `cosmos.bank2.Send` or `cosmos.bank.Send2`.
aaronc marked this conversation as resolved.
Show resolved Hide resolved
Version 1 should obviously be the non-suffixed version, ex. `cosmos.bank.Send`.

#### Use `alpha` or `beta` to Denote Non-stable Packages

[Buf's recommended version suffix](https://buf.build/docs/lint-checkers#package_version_suffix)
(ex. `v1alpha1`) _should_ be used for non-stable packages. These packages should
likely be excluded from breaking change detection and _should_ generally
be blacklisted from usage by smart contracts/persistent scripts to prevent them
from breaking. The SDK _should_ mark any packages as alpha or beta where the
API is likely to change significantly in the near future.

### Package Naming

#### Adopt a short, unique top-level package name

Top-level packages should adopt a short name that is known to not collide with
other names in common usage within the Cosmos ecosystem. In the near future, a
registry should be created to reserve and index top-level package names used
within the Cosmos ecosystem. Because the Cosmos SDK is intended to provide
the top-level types for the Cosmos project, the top-level package name `cosmos`
is recommended for usage within the Cosmos SDK instead of the longer `cosmos_sdk`.
[ICS](https://github.com/cosmos/ics) specifications could consider a
short top-level package like `ics23` based upon the standard number.

#### Limit sub-package depth

Sub-package depth should be increased with caution. Generally a single
sub-package is needed for a module or a library. Even though `x` or `modules`
is used in source code to denote modules, this is often unnecessary for .proto
files as modules are the primary thing sub-packages are used for. Only items which
are known to be used infrequently should have deep sub-package depths.

For the Cosmos SDK, it is recommended that that we simply write `cosmos.bank`,
`cosmos.gov`, etc. rather than `cosmos.x.bank`. In practice, most non-module
types can go straight in the `cosmos` package or we can introduce a
`cosmos.base` package if needed. Note that this naming _will not_ change
go package names, i.e. the `cosmos.bank` protobuf package will still live in
`x/bank`.

### Message Naming

#### Use simple verbs for transaction messages
aaronc marked this conversation as resolved.
Show resolved Hide resolved

This is likely one of the most controversial recommendations here as it will
change legacy go struct names.

Historically all transaction messages in the SDK have been prefixed with `Msg`
which is a shortening of msssage. To begin with "message", not being an action
word, does not clearly indicate the concept of transaction or operation. So
understanding that `Msg` indicates transaction likely requires knowledge of
Cosmos SDK conventions. Once someone understands that `Msg` means transaction,
however, it still is likely redundant information as the type name usually
includes an action verb such as "send", "create" or "submit".

Going forward, for both conciseness and clarity, transaction messages should
simply use a descriptive action verb to indicate that this is a type which
performs an action in a transaction. This should also be made clear in the
documentation.

To maintain compatibility with existing names in go code, an alias can be
introduced for legacy type names.

#### Use simple nouns for state types

Nouns should be used for other types which are used in state, queries and as
arguments to some transaction messages. Ex. `Coin`, `Proposal` and `Delegation`.

### Service and RPC Naming

[ADR 021](adr-021-protobuf-query-encoding.md) specifies that modules should
implement a gRPC query service. We should consider the principle of conciseness
for query service and RPC names as these may be called from persistent script
modules such as CosmWasm. Also, users may use these query paths from tools like
[gRPCurl](https://github.com/fullstorydev/grpcurl). As an example, we can shorten
`/cosmos_sdk.x.bank.v1.QueryService/QueryBalance` to
`/cosmos.bank.Query/Balance` without losing much useful information.

#### Use just `Query` for the query service

Instead of [Buf's default service suffix recommendation](https://github.com/cosmos/cosmos-sdk/pull/6033),
we should simply use the shorter `Query` for query services.

For other types of gRPC services, we should consider sticking with Buf's
default recommendation.

#### Omit `Get` and `Query` from query service RPC names

`Get` and `Query` should be omitted from `Query` service names because they are
redundant in the fully-qualified name. For instance, `/cosmos.bank.Query/QueryBalance`
just says `Query` twice without any new information.

## Future Improvements

A registry of top-level package names should be created to coordinate naming
across the ecosystem, prevent collisions, and also help developers discover
useful schemas. A simple starting point would be a git repository with
community-based governance.

## Consequences

### Positive

* names will be more concise and easier to read and type
* all transactions using `Any` will be at least 12 bytes shorter
(`_sdk.x`, `.v1`, `Msg` will be removed from `sdk.Msg` message names)
* `.proto` file imports will be more standard (without `"third_party/proto"` in
the path)
* code generation will be easier for clients because .proto files will be
in a single `proto/` directory which can be copied rather than scattered
throughout the SDK

### Negative

* some legacy go struct names will change (ex. `MsgSend` vs `Send`), but we can
make it effectively non-API breaking with aliases (i.e. `type MsgSend = Send`).
Note also that legacy Amino type names will not change

### Neutral

* `.proto` files will need to be reorganized and refactored
* some modules may need to be marked as alpha or beta

## References