diff --git a/CHANGELOG.md b/CHANGELOG.md index 425c2cf14..50316b4a7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -245,6 +245,7 @@ IMPROVEMENTS * [cli] [\#2128](https://github.com/cosmos/cosmos-sdk/issues/2128) fixed segfault when exporting directly after `gaiad init` * [cli] [\#1255](https://github.com/cosmos/cosmos-sdk/issues/1255) open KeyBase in read-only mode for query-purpose CLI commands + * [docs] Added commands for querying governance deposits, votes and tally * Gaia * [x/stake] [#2023](https://github.com/cosmos/cosmos-sdk/pull/2023) Terminate iteration loop in `UpdateBondedValidators` and `UpdateBondedValidatorsFull` when the first revoked validator is encountered and perform a sanity check. @@ -289,6 +290,7 @@ BUG FIXES * Gaia * [x/stake] Return correct Tendermint validator update set on `EndBlocker` by not including non previously bonded validators that have zero power. [#2189](https://github.com/cosmos/cosmos-sdk/issues/2189) + * [docs] Fixed light client section links * SDK * [\#1988](https://github.com/cosmos/cosmos-sdk/issues/1988) Make us compile on OpenBSD (disable ledger) [#1988] (https://github.com/cosmos/cosmos-sdk/issues/1988) diff --git a/Gopkg.lock b/Gopkg.lock index 9289fd24b..bcc306edb 100644 --- a/Gopkg.lock +++ b/Gopkg.lock @@ -418,12 +418,12 @@ revision = "e5840949ff4fff0c56f9b6a541e22b63581ea9df" [[projects]] - digest = "1:10b3a599325740c84a7c81f3f3cb2e1fdb70b3ea01b7fa28495567a2519df431" + digest = "1:ad9c4c1a4e7875330b1f62906f2830f043a23edb5db997e3a5ac5d3e6eadf80a" name = "github.com/tendermint/go-amino" packages = ["."] pruneopts = "UT" - revision = "6dcc6ddc143e116455c94b25c1004c99e0d0ca12" - version = "v0.14.0" + revision = "dc14acf9ef15f85828bfbc561ed9dd9d2a284885" + version = "v0.14.1" [[projects]] digest = "1:9f8c4c93658315a795ffd3e0c943d39f78067dd8382b8d7bcfaf6686b92f3978" @@ -434,7 +434,7 @@ version = "v0.11.1" [[projects]] - digest = "1:5b1373b03f39e6f6061cd91f3829100527ebb5f94240c092bf9e5d314b153501" + digest = "1:ba2ba7d6a0853472bdb7a64c4f9c1d5f9cba0eb7aac0b024654104387bf5eb57" name = "github.com/tendermint/tendermint" packages = [ "abci/client", @@ -500,8 +500,8 @@ "version", ] pruneopts = "UT" - revision = "48ab899923c564bbf2fa2f1244c11cb930e28132" - version = "v0.26.1-rc3" + revision = "80d0a362500fea2dd089258319075a54e5d40a2d" + version = "v0.26.1" [[projects]] digest = "1:7886f86064faff6f8d08a3eb0e8c773648ff5a2e27730831e2bfbf07467f6666" diff --git a/Gopkg.toml b/Gopkg.toml index 466fd373f..f8ffcca05 100644 --- a/Gopkg.toml +++ b/Gopkg.toml @@ -28,7 +28,7 @@ [[override]] name = "github.com/tendermint/go-amino" - version = "v0.14.0" + version = "v0.14.1" [[override]] name = "github.com/tendermint/iavl" @@ -36,7 +36,7 @@ [[override]] name = "github.com/tendermint/tendermint" - version = "v0.26.1-rc3" # TODO replace w/ 0.26.1 + version = "v0.26.1" ## deps without releases: diff --git a/Makefile b/Makefile index aaa2fbcf3..35e5b50ef 100644 --- a/Makefile +++ b/Makefile @@ -1,15 +1,14 @@ PACKAGES_NOSIMULATION=$(shell go list ./... | grep -v '/simulation') PACKAGES_SIMTEST=$(shell go list ./... | grep '/simulation') -VERSION := $(shell git describe --tags --long | sed 's/v\(.*\)/\1/') -BUILD_TAGS = netgo ledger +VERSION := $(subst v,,$(shell git describe --tags --long)) +BUILD_TAGS = netgo BUILD_FLAGS = -tags "${BUILD_TAGS}" -ldflags "-X github.com/cosmos/cosmos-sdk/version.Version=${VERSION}" -GCC := $(shell command -v gcc 2> /dev/null) LEDGER_ENABLED ?= true -UNAME_S := $(shell uname -s) GOTOOLS = \ github.com/golang/dep/cmd/dep \ github.com/alecthomas/gometalinter \ github.com/rakyll/statik +GOBIN ?= $(GOPATH)/bin all: get_tools get_vendor_deps install install_examples install_cosmos-sdk-cli test_lint test ######################################## @@ -20,23 +19,30 @@ ci: get_tools get_vendor_deps install test_cover test_lint test ######################################## ### Build/Install -check-ledger: ifeq ($(LEDGER_ENABLED),true) - ifeq ($(UNAME_S),OpenBSD) - $(info "OpenBSD detected, disabling ledger support (https://github.com/cosmos/cosmos-sdk/issues/1988)") -TMP_BUILD_TAGS := $(BUILD_TAGS) -BUILD_TAGS = $(filter-out ledger, $(TMP_BUILD_TAGS)) - else - ifndef GCC - $(error "gcc not installed for ledger support, please install or set LEDGER_ENABLED to false in the Makefile") - endif - endif -else -TMP_BUILD_TAGS := $(BUILD_TAGS) -BUILD_TAGS = $(filter-out ledger, $(TMP_BUILD_TAGS)) + ifeq ($(OS),Windows_NT) + GCCEXE = $(shell where gcc.exe 2> NUL) + ifeq ($(GCCEXE),) + $(error gcc.exe not installed for ledger support, please install or set LEDGER_ENABLED=false) + else + BUILD_TAGS += ledger + endif + else + UNAME_S = $(shell uname -s) + ifeq ($(UNAME_S),OpenBSD) + $(warning OpenBSD detected, disabling ledger support (https://github.com/cosmos/cosmos-sdk/issues/1988)) + else + GCC = $(shell command -v gcc 2> /dev/null) + ifeq ($(GCC),) + $(error gcc not installed for ledger support, please install or set LEDGER_ENABLED=false) + else + BUILD_TAGS += ledger + endif + endif + endif endif -build: check-ledger update_gaia_lite_docs +build: ifeq ($(OS),Windows_NT) go build $(BUILD_FLAGS) -o build/gaiad.exe ./cmd/gaia/cmd/gaiad go build $(BUILD_FLAGS) -o build/gaiacli.exe ./cmd/gaia/cmd/gaiacli @@ -60,15 +66,15 @@ endif build_examples: ifeq ($(OS),Windows_NT) - go build $(BUILD_FLAGS) -o build/basecoind.exe ./examples/basecoin/cmd/basecoind - go build $(BUILD_FLAGS) -o build/basecli.exe ./examples/basecoin/cmd/basecli - go build $(BUILD_FLAGS) -o build/democoind.exe ./examples/democoin/cmd/democoind - go build $(BUILD_FLAGS) -o build/democli.exe ./examples/democoin/cmd/democli + go build $(BUILD_FLAGS) -o build/basecoind.exe ./docs/examples/basecoin/cmd/basecoind + go build $(BUILD_FLAGS) -o build/basecli.exe ./docs/examples/basecoin/cmd/basecli + go build $(BUILD_FLAGS) -o build/democoind.exe ./docs/examples/democoin/cmd/democoind + go build $(BUILD_FLAGS) -o build/democli.exe ./docs/examples/democoin/cmd/democli else - go build $(BUILD_FLAGS) -o build/basecoind ./examples/basecoin/cmd/basecoind - go build $(BUILD_FLAGS) -o build/basecli ./examples/basecoin/cmd/basecli - go build $(BUILD_FLAGS) -o build/democoind ./examples/democoin/cmd/democoind - go build $(BUILD_FLAGS) -o build/democli ./examples/democoin/cmd/democli + go build $(BUILD_FLAGS) -o build/basecoind ./docs/examples/basecoin/cmd/basecoind + go build $(BUILD_FLAGS) -o build/basecli ./docs/examples/basecoin/cmd/basecli + go build $(BUILD_FLAGS) -o build/democoind ./docs/examples/democoin/cmd/democoind + go build $(BUILD_FLAGS) -o build/democli ./docs/examples/democoin/cmd/democli endif install: check-ledger update_gaia_lite_docs @@ -76,10 +82,10 @@ install: check-ledger update_gaia_lite_docs go install $(BUILD_FLAGS) ./cmd/gaia/cmd/gaiacli install_examples: - go install $(BUILD_FLAGS) ./examples/basecoin/cmd/basecoind - go install $(BUILD_FLAGS) ./examples/basecoin/cmd/basecli - go install $(BUILD_FLAGS) ./examples/democoin/cmd/democoind - go install $(BUILD_FLAGS) ./examples/democoin/cmd/democli + go install $(BUILD_FLAGS) ./docs/examples/basecoin/cmd/basecoind + go install $(BUILD_FLAGS) ./docs/examples/basecoin/cmd/basecli + go install $(BUILD_FLAGS) ./docs/examples/democoin/cmd/democoind + go install $(BUILD_FLAGS) ./docs/examples/democoin/cmd/democli install_cosmos-sdk-cli: go install $(BUILD_FLAGS) ./cmd/cosmos-sdk-cli @@ -101,33 +107,36 @@ check_tools: update_tools: @echo "--> Updating tools to correct version" - ./scripts/get_tools.sh + $(MAKE) -C scripts get_tools update_dev_tools: @echo "--> Downloading linters (this may take awhile)" $(GOPATH)/src/github.com/alecthomas/gometalinter/scripts/install.sh -b $(GOBIN) go get -u github.com/tendermint/lint/golint -get_tools: +get_tools: $(GOBIN)/dep $(GOBIN)/gometalinter $(GOBIN)/statik @echo "--> Installing tools" - ./scripts/get_tools.sh + $(MAKE) -C scripts get_tools -get_dev_tools: +$(GOBIN)/%: + $(MAKE) -C scripts $(subst $(GOBIN)/,,$(@)) + +get_dev_tools: get_tools @echo "--> Downloading linters (this may take awhile)" $(GOPATH)/src/github.com/alecthomas/gometalinter/scripts/install.sh -b $(GOBIN) go get github.com/tendermint/lint/golint -get_vendor_deps: +get_vendor_deps: get_tools @echo "--> Generating vendor directory via dep ensure" @rm -rf .vendor-new @dep ensure -v -vendor-only -update_vendor_deps: +update_vendor_deps: get_tools @echo "--> Running dep ensure" @rm -rf .vendor-new @dep ensure -v -draw_deps: +draw_deps: get_tools @# requires brew install graphviz or apt-get install graphviz go get github.com/RobotsAndPencils/goviz @goviz -i github.com/cosmos/cosmos-sdk/cmd/gaia/cmd/gaiad -d 2 | dot -Tpng -o dependency-graph.png @@ -150,8 +159,8 @@ test_cli: @go test -count 1 -p 1 `go list github.com/cosmos/cosmos-sdk/cmd/gaia/cli_test` -tags=cli_test test_examples: - @go test -count 1 -p 1 `go list github.com/cosmos/cosmos-sdk/examples/basecoin/cli_test` -tags=cli_test - @go test -count 1 -p 1 `go list github.com/cosmos/cosmos-sdk/examples/democoin/cli_test` -tags=cli_test + @go test -count 1 -p 1 `go list github.com/cosmos/cosmos-sdk/docs/examples/basecoin/cli_test` -tags=cli_test + @go test -count 1 -p 1 `go list github.com/cosmos/cosmos-sdk/docs/examples/democoin/cli_test` -tags=cli_test test_unit: @VERSION=$(VERSION) go test $(PACKAGES_NOSIMULATION) @@ -173,12 +182,7 @@ test_sim_gaia_fast: test_sim_gaia_import_export: @echo "Running Gaia import/export simulation. This may take several minutes..." - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=4 -v -timeout 24h - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=11 -v -timeout 24h - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=12 -v -timeout 24h - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=13 -v -timeout 24h - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=414 -v -timeout 24h - @go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=50 -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=4142 -v -timeout 24h + @bash scripts/import-export-sim.sh 50 test_sim_gaia_multi_seed: @echo "Running multi-seed Gaia simulation. This may take awhile!" @@ -256,7 +260,7 @@ localnet-stop: # unless there is a reason not to. # https://www.gnu.org/software/make/manual/html_node/Phony-Targets.html .PHONY: build build_cosmos-sdk-cli build_examples install install_examples install_cosmos-sdk-cli install_debug dist \ -check_tools check_dev_tools get_tools get_dev_tools get_vendor_deps draw_deps test test_cli test_unit \ +check_tools check_dev_tools get_dev_tools get_vendor_deps draw_deps test test_cli test_unit \ test_cover test_lint benchmark devdoc_init devdoc devdoc_save devdoc_update \ build-linux build-docker-gaiadnode localnet-start localnet-stop \ format check-ledger test_sim_gaia_nondeterminism test_sim_modules test_sim_gaia_fast \ diff --git a/PENDING.md b/PENDING.md index 141df2a9b..52bf8a7ee 100644 --- a/PENDING.md +++ b/PENDING.md @@ -5,7 +5,9 @@ BREAKING CHANGES * Gaia REST API (`gaiacli advanced rest-server`) * Gaia CLI (`gaiacli`) + * [cli] [\#2728](https://github.com/cosmos/cosmos-sdk/pull/2728) Seperate `tx` and `query` subcommands by module * [cli] [\#2727](https://github.com/cosmos/cosmos-sdk/pull/2727) Fix unbonding command flow + * [cli] [\#2786](https://github.com/cosmos/cosmos-sdk/pull/2786) Fix redelegation command flow * Gaia @@ -18,10 +20,17 @@ BREAKING CHANGES FEATURES * Gaia REST API (`gaiacli advanced rest-server`) + * [gov] [\#2479](https://github.com/cosmos/cosmos-sdk/issues/2479) Added governance parameter + query REST endpoints. * Gaia CLI (`gaiacli`) - + * [gov][cli] [\#2479](https://github.com/cosmos/cosmos-sdk/issues/2479) Added governance + parameter query commands. + * [stake][cli] [\#2027] Add CLI query command for getting all delegations to a specific validator. + * Gaia + * [x/gov] [#2479](https://github.com/cosmos/cosmos-sdk/issues/2479) Implemented querier + for getting governance parameters. * SDK * [simulator] \#2682 MsgEditValidator now looks at the validator's max rate, thus it now succeeds a significant portion of the time @@ -37,11 +46,14 @@ IMPROVEMENTS * [\#2749](https://github.com/cosmos/cosmos-sdk/pull/2749) Add --chain-id flag to gaiad testnet * Gaia + - #2773 Require moniker to be provided on `gaiad init`. + - #2672 [Makefile] Updated for better Windows compatibility and ledger support logic, get_tools was rewritten as a cross-compatible Makefile. * SDK - [x/mock/simulation] [\#2720] major cleanup, introduction of helper objects, reorganization * Tendermint + - #2796 Update to go-amino 0.14.1 BUG FIXES @@ -52,9 +64,11 @@ BUG FIXES * Gaia * [\#2723] Use `cosmosvalcons` Bech32 prefix in `tendermint show-address` - * [\#2742](https://github.com/cosmos/cosmos-sdk/issues/2742) Fix time format of TimeoutCommit override - + * [\#2742](https://github.com/cosmos/cosmos-sdk/issues/2742) Fix time format of TimeoutCommit override + * SDK + + - \#2733 [x/gov, x/mock/simulation] Fix governance simulation, update x/gov import/export * Tendermint * [\#2797](https://github.com/tendermint/tendermint/pull/2797) AddressBook requires addresses to have IDs; Do not crap out immediately after sending pex addrs in seed mode diff --git a/README.md b/README.md index 94b2b19da..b57cd1dff 100644 --- a/README.md +++ b/README.md @@ -17,27 +17,26 @@ It is being used to build `Gaia`, the first implementation of the [Cosmos Hub](h **WARNING**: The SDK has mostly stabilized, but we are still making some breaking changes. -**Note**: Requires [Go 1.10+](https://golang.org/dl/) +**Note**: Requires [Go 1.11+](https://golang.org/dl/) ## Gaia Testnet +To install the binaries, read the [install instructions](./docs/gaia/installation.md) + To join the latest testnet, follow -[the guide](./docs/getting-started/join-testnet.md). +[the guide](./docs/gaia/join-testnet.md). For status updates and genesis files, see the [testnets repo](https://github.com/cosmos/testnets). -## Install - -See the -[install instructions](./docs/getting-started/installation.md). ## Quick Start -See the [Cosmos Docs](https://cosmos.network/docs/) +To learn how the SDK works from a high-level perspective, go to the [SDK Intro](./docs/intro/README.md). -- [Getting started with the SDK](./docs/sdk/core/intro.md) -- [SDK Examples](/examples) +If you want to get started quickly and learn how to build on top of the SDK, please follow the [SDK Application Tutorial](https://github.com/cosmos/sdk-application-tutorial). You can also fork the tutorial's repo to get started building your own Cosmos SDK application. + +For more, please go to the [Cosmos SDK Docs](./docs/README.md) ## Disambiguation diff --git a/client/config.go b/client/config.go index a1d38a016..b76610638 100644 --- a/client/config.go +++ b/client/config.go @@ -3,20 +3,20 @@ package client import ( "bufio" "fmt" + "io/ioutil" + "os" + "path" + "github.com/cosmos/cosmos-sdk/types" "github.com/mitchellh/go-homedir" "github.com/pelletier/go-toml" "github.com/spf13/cobra" - "io/ioutil" - "os" - "path" ) type cliConfig struct { Home string `toml:"home"` ChainID string `toml:"chain_id"` TrustNode bool `toml:"trust_node"` - Encoding string `toml:"encoding"` Output string `toml:"output"` Node string `toml:"node"` Trace bool `toml:"trace"` @@ -41,23 +41,24 @@ func runConfigCmd(cmd *cobra.Command, args []string) error { } stdin := BufferStdin() + gaiaCLIHome, err := handleGaiaCLIHome(home, stdin) if err != nil { return err } + node, err := handleNode(stdin) if err != nil { return err } + trustNode, err := handleTrustNode(stdin) if err != nil { return err } - encoding := "btc" - output := "text" - var chainID string - chainID, err = types.DefaultChainID() + chainID, err := types.DefaultChainID() + if err != nil { fmt.Println("Couldn't populate ChainID, so using an empty one.") } @@ -66,8 +67,7 @@ func runConfigCmd(cmd *cobra.Command, args []string) error { Home: gaiaCLIHome, ChainID: chainID, TrustNode: trustNode, - Encoding: encoding, - Output: output, + Output: "text", Node: node, Trace: false, } diff --git a/client/lcd/lcd_test.go b/client/lcd/lcd_test.go index 9b042832b..1be908e98 100644 --- a/client/lcd/lcd_test.go +++ b/client/lcd/lcd_test.go @@ -539,7 +539,7 @@ func TestBonding(t *testing.T) { require.Equal(t, int64(40), coins.AmountOf(denom).Int64()) - // query validator + // query delegation bond := getDelegation(t, port, addr, operAddrs[0]) require.Equal(t, amt, bond.Shares) @@ -547,6 +547,10 @@ func TestBonding(t *testing.T) { require.Len(t, delegatorDels, 1) require.Equal(t, amt, delegatorDels[0].Shares) + // query all delegations to validator + bonds := getValidatorDelegations(t, port, operAddrs[0]) + require.Len(t, bonds, 2) + bondedValidators := getDelegatorValidators(t, port, addr) require.Len(t, bondedValidators, 1) require.Equal(t, operAddrs[0], bondedValidators[0].OperatorAddr) @@ -731,26 +735,33 @@ func TestProposalsQuery(t *testing.T) { cleanup, _, _, port := InitializeTestLCD(t, 1, []sdk.AccAddress{addrs[0], addrs[1]}) defer cleanup() + depositParam := getDepositParam(t, port) + halfMinDeposit := depositParam.MinDeposit.AmountOf(stakeTypes.DefaultBondDenom).Int64() / 2 + getVotingParam(t, port) + getTallyingParam(t, port) + // Addr1 proposes (and deposits) proposals #1 and #2 - resultTx := doSubmitProposal(t, port, seeds[0], names[0], passwords[0], addrs[0], 5) + resultTx := doSubmitProposal(t, port, seeds[0], names[0], passwords[0], addrs[0], halfMinDeposit) var proposalID1 uint64 cdc.MustUnmarshalBinaryLengthPrefixed(resultTx.DeliverTx.GetData(), &proposalID1) tests.WaitForHeight(resultTx.Height+1, port) - resultTx = doSubmitProposal(t, port, seeds[0], names[0], passwords[0], addrs[0], 5) + + resultTx = doSubmitProposal(t, port, seeds[0], names[0], passwords[0], addrs[0], halfMinDeposit) var proposalID2 uint64 cdc.MustUnmarshalBinaryLengthPrefixed(resultTx.DeliverTx.GetData(), &proposalID2) tests.WaitForHeight(resultTx.Height+1, port) // Addr2 proposes (and deposits) proposals #3 - resultTx = doSubmitProposal(t, port, seeds[1], names[1], passwords[1], addrs[1], 5) + resultTx = doSubmitProposal(t, port, seeds[1], names[1], passwords[1], addrs[1], halfMinDeposit) var proposalID3 uint64 cdc.MustUnmarshalBinaryLengthPrefixed(resultTx.DeliverTx.GetData(), &proposalID3) tests.WaitForHeight(resultTx.Height+1, port) // Addr2 deposits on proposals #2 & #3 - resultTx = doDeposit(t, port, seeds[1], names[1], passwords[1], addrs[1], proposalID2, 5) + resultTx = doDeposit(t, port, seeds[1], names[1], passwords[1], addrs[1], proposalID2, halfMinDeposit) tests.WaitForHeight(resultTx.Height+1, port) - resultTx = doDeposit(t, port, seeds[1], names[1], passwords[1], addrs[1], proposalID3, 5) + + resultTx = doDeposit(t, port, seeds[1], names[1], passwords[1], addrs[1], proposalID3, halfMinDeposit) tests.WaitForHeight(resultTx.Height+1, port) // check deposits match proposal and individual deposits @@ -1207,6 +1218,17 @@ func getValidator(t *testing.T, port string, validatorAddr sdk.ValAddress) stake return validator } +func getValidatorDelegations(t *testing.T, port string, validatorAddr sdk.ValAddress) []stake.Delegation { + res, body := Request(t, port, "GET", fmt.Sprintf("/stake/validators/%s/delegations", validatorAddr.String()), nil) + require.Equal(t, http.StatusOK, res.StatusCode, body) + + var delegations []stake.Delegation + err := cdc.UnmarshalJSON([]byte(body), &delegations) + require.Nil(t, err) + + return delegations +} + func getValidatorUnbondingDelegations(t *testing.T, port string, validatorAddr sdk.ValAddress) []stake.UnbondingDelegation { res, body := Request(t, port, "GET", fmt.Sprintf("/stake/validators/%s/unbonding_delegations", validatorAddr.String()), nil) require.Equal(t, http.StatusOK, res.StatusCode, body) @@ -1231,6 +1253,36 @@ func getValidatorRedelegations(t *testing.T, port string, validatorAddr sdk.ValA // ============= Governance Module ================ +func getDepositParam(t *testing.T, port string) gov.DepositParams { + res, body := Request(t, port, "GET", "/gov/parameters/deposit", nil) + require.Equal(t, http.StatusOK, res.StatusCode, body) + + var depositParams gov.DepositParams + err := cdc.UnmarshalJSON([]byte(body), &depositParams) + require.Nil(t, err) + return depositParams +} + +func getVotingParam(t *testing.T, port string) gov.VotingParams { + res, body := Request(t, port, "GET", "/gov/parameters/voting", nil) + require.Equal(t, http.StatusOK, res.StatusCode, body) + + var votingParams gov.VotingParams + err := cdc.UnmarshalJSON([]byte(body), &votingParams) + require.Nil(t, err) + return votingParams +} + +func getTallyingParam(t *testing.T, port string) gov.TallyParams { + res, body := Request(t, port, "GET", "/gov/parameters/tallying", nil) + require.Equal(t, http.StatusOK, res.StatusCode, body) + + var tallyParams gov.TallyParams + err := cdc.UnmarshalJSON([]byte(body), &tallyParams) + require.Nil(t, err) + return tallyParams +} + func getProposal(t *testing.T, port string, proposalID uint64) gov.Proposal { res, body := Request(t, port, "GET", fmt.Sprintf("/gov/proposals/%d", proposalID), nil) require.Equal(t, http.StatusOK, res.StatusCode, body) diff --git a/client/lcd/swagger-ui/swagger.yaml b/client/lcd/swagger-ui/swagger.yaml index 86b6ec681..920194ffa 100644 --- a/client/lcd/swagger-ui/swagger.yaml +++ b/client/lcd/swagger-ui/swagger.yaml @@ -950,6 +950,30 @@ paths: description: Invalid validator address 500: description: Internal Server Error + /stake/validators/{validatorAddr}/delegations: + parameters: + - in: path + name: validatorAddr + description: Bech32 OperatorAddress of validator + required: true + type: string + get: + summary: Get all delegations from a validator + tags: + - ICS21 + produces: + - application/json + responses: + 200: + description: OK + schema: + type: array + items: + $ref: "#/definitions/Delegation" + 400: + description: Invalid validator address + 500: + description: Internal Server Error /stake/validators/{validatorAddr}/unbonding_delegations: parameters: - in: path @@ -1427,7 +1451,7 @@ paths: /gov/proposals/{proposalId}/votes/{voter}: get: summary: Query vote - description: Query vote information by proposalId and voter address + description: Query vote information by proposal Id and voter address produces: - application/json tags: @@ -1454,6 +1478,83 @@ paths: description: Found no vote 500: description: Internal Server Error + /gov/parameters/deposit: + get: + summary: Query governance deposit parameters + description: Query governance deposit parameters. The max_deposit_period units are in nanoseconds. + produces: + - application/json + tags: + - ICS22 + responses: + 200: + description: OK + schema: + type: object + properties: + min_deposit: + type: array + items: + $ref: "#/definitions/Coin" + max_deposit_period: + type: string + example: "86400000000000" + 400: + description: is not a valid query request path + 404: + description: Found no deposit parameters + 500: + description: Internal Server Error + /gov/parameters/tallying: + get: + summary: Query governance tally parameters + description: Query governance tally parameters + produces: + - application/json + tags: + - ICS22 + responses: + 200: + description: OK + schema: + properties: + threshold: + type: string + example: "0.5000000000" + veto: + type: string + example: "0.3340000000" + governance_penalty: + type: string + example: "0.0100000000" + 400: + description: is not a valid query request path + 404: + description: Found no tally parameters + 500: + description: Internal Server Error + /gov/parameters/voting: + get: + summary: Query governance voting parameters + description: Query governance voting parameters. The voting_period units are in nanoseconds. + produces: + - application/json + tags: + - ICS22 + responses: + 200: + description: OK + schema: + properties: + voting_period: + type: string + example: "86400000000000" + 400: + description: is not a valid query request path + 404: + description: Found no voting parameters + 500: + description: Internal Server Error definitions: CheckTxResult: diff --git a/client/lcd/test_helpers.go b/client/lcd/test_helpers.go index 283550a16..774fd4d6a 100644 --- a/client/lcd/test_helpers.go +++ b/client/lcd/test_helpers.go @@ -4,9 +4,7 @@ import ( "bytes" "encoding/json" "fmt" - "github.com/cosmos/cosmos-sdk/x/stake" - stakeTypes "github.com/cosmos/cosmos-sdk/x/stake/types" - "github.com/tendermint/tendermint/crypto/secp256k1" + "io/ioutil" "net" "net/http" @@ -16,6 +14,9 @@ import ( "strings" "testing" + stakeTypes "github.com/cosmos/cosmos-sdk/x/stake/types" + "github.com/tendermint/tendermint/crypto/secp256k1" + "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/client/keys" gapp "github.com/cosmos/cosmos-sdk/cmd/gaia/app" @@ -25,6 +26,7 @@ import ( "github.com/cosmos/cosmos-sdk/tests" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" + "github.com/cosmos/cosmos-sdk/x/stake" "github.com/spf13/viper" "github.com/stretchr/testify/require" diff --git a/client/rpc/root.go b/client/rpc/root.go index 74c5a69a2..7bdcd8cd8 100644 --- a/client/rpc/root.go +++ b/client/rpc/root.go @@ -22,15 +22,8 @@ func todoNotImplemented(_ *cobra.Command, _ []string) error { return errors.New("todo: Command not yet implemented") } -// AddCommands adds a number of rpc-related subcommands -func AddCommands(cmd *cobra.Command) { - cmd.AddCommand( - initClientCommand(), - statusCommand(), - ) -} - -func initClientCommand() *cobra.Command { +// InitClientCommand initializes client commands +func InitClientCommand() *cobra.Command { cmd := &cobra.Command{ Use: "init", Short: "Initialize light client", diff --git a/client/rpc/status.go b/client/rpc/status.go index 68716f3a0..0f81fc9c6 100644 --- a/client/rpc/status.go +++ b/client/rpc/status.go @@ -14,7 +14,8 @@ import ( ctypes "github.com/tendermint/tendermint/rpc/core/types" ) -func statusCommand() *cobra.Command { +// StatusCommand returns the status of the network +func StatusCommand() *cobra.Command { cmd := &cobra.Command{ Use: "status", Short: "Query remote node for status", diff --git a/client/tx/root.go b/client/tx/root.go index 0f237b596..5cd7eca0b 100644 --- a/client/tx/root.go +++ b/client/tx/root.go @@ -2,20 +2,11 @@ package tx import ( "github.com/gorilla/mux" - "github.com/spf13/cobra" "github.com/cosmos/cosmos-sdk/client/context" "github.com/cosmos/cosmos-sdk/codec" ) -// AddCommands adds a number of tx-query related subcommands -func AddCommands(cmd *cobra.Command, cdc *codec.Codec) { - cmd.AddCommand( - SearchTxCmd(cdc), - QueryTxCmd(cdc), - ) -} - // register REST routes func RegisterRoutes(cliCtx context.CLIContext, r *mux.Router, cdc *codec.Codec) { r.HandleFunc("/txs/{hash}", QueryTxRequestHandlerFn(cdc, cliCtx)).Methods("GET") diff --git a/cmd/cosmos-sdk-cli/cmd/init.go b/cmd/cosmos-sdk-cli/cmd/init.go index e9e9cd8a9..fcf30b972 100644 --- a/cmd/cosmos-sdk-cli/cmd/init.go +++ b/cmd/cosmos-sdk-cli/cmd/init.go @@ -14,7 +14,7 @@ import ( tmversion "github.com/tendermint/tendermint/version" ) -var remoteBasecoinPath = "github.com/cosmos/cosmos-sdk/examples/basecoin" +var remoteBasecoinPath = "github.com/cosmos/cosmos-sdk/docs/examples/basecoin" // Replacer to replace all instances of basecoin/basecli/BasecoinApp to project specific names // Gets initialized when initCmd is executing after getting the project name from user diff --git a/cmd/gaia/app/app.go b/cmd/gaia/app/app.go index 918bfc67d..765d624e9 100644 --- a/cmd/gaia/app/app.go +++ b/cmd/gaia/app/app.go @@ -251,6 +251,8 @@ func (app *GaiaApp) initChainer(ctx sdk.Context, req abci.RequestInitChain) abci gov.InitGenesis(ctx, app.govKeeper, genesisState.GovData) mint.InitGenesis(ctx, app.mintKeeper, genesisState.MintData) distr.InitGenesis(ctx, app.distrKeeper, genesisState.DistrData) + + // validate genesis state err = GaiaValidateGenesisState(genesisState) if err != nil { panic(err) // TODO find a way to do this w/o panics diff --git a/cmd/gaia/cli_test/cli_test.go b/cmd/gaia/cli_test/cli_test.go index 07cbbab78..874c2a29a 100644 --- a/cmd/gaia/cli_test/cli_test.go +++ b/cmd/gaia/cli_test/cli_test.go @@ -237,7 +237,7 @@ func TestGaiaCLICreateValidator(t *testing.T) { initialPool.BondedTokens = initialPool.BondedTokens.Add(sdk.NewDec(100)) // Delegate tx on GaiaAppGenState // create validator - cvStr := fmt.Sprintf("gaiacli tx create-validator %v", flags) + cvStr := fmt.Sprintf("gaiacli tx stake create-validator %v", flags) cvStr += fmt.Sprintf(" --from=%s", "bar") cvStr += fmt.Sprintf(" --pubkey=%s", barCeshPubKey) cvStr += fmt.Sprintf(" --amount=%v", fmt.Sprintf("2%s", stakeTypes.DefaultBondDenom)) @@ -268,12 +268,16 @@ func TestGaiaCLICreateValidator(t *testing.T) { barAcc = executeGetAccount(t, fmt.Sprintf("gaiacli query account %s %v", barAddr, flags)) require.Equal(t, int64(8), barAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64(), "%v", barAcc) - validator := executeGetValidator(t, fmt.Sprintf("gaiacli query validator %s --output=json %v", sdk.ValAddress(barAddr), flags)) + validator := executeGetValidator(t, fmt.Sprintf("gaiacli query stake validator %s --output=json %v", sdk.ValAddress(barAddr), flags)) require.Equal(t, validator.OperatorAddr, sdk.ValAddress(barAddr)) require.True(sdk.DecEq(t, sdk.NewDec(2), validator.Tokens)) + validatorDelegations := executeGetValidatorDelegations(t, fmt.Sprintf("gaiacli query stake delegations-to %s --output=json %v", sdk.ValAddress(barAddr), flags)) + require.Len(t, validatorDelegations, 1) + require.NotZero(t, validatorDelegations[0].Shares) + // unbond a single share - unbondStr := fmt.Sprintf("gaiacli tx unbond begin %v", flags) + unbondStr := fmt.Sprintf("gaiacli tx stake unbond begin %v", flags) unbondStr += fmt.Sprintf(" --from=%s", "bar") unbondStr += fmt.Sprintf(" --validator=%s", sdk.ValAddress(barAddr)) unbondStr += fmt.Sprintf(" --shares-amount=%v", "1") @@ -286,19 +290,19 @@ func TestGaiaCLICreateValidator(t *testing.T) { barAcc = executeGetAccount(t, fmt.Sprintf("gaiacli query account %v %v", barCech, flags)) require.Equal(t, int64(9), barAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64(), "%v", barAcc) */ - validator = executeGetValidator(t, fmt.Sprintf("gaiacli query validator %s --output=json %v", sdk.ValAddress(barAddr), flags)) + validator = executeGetValidator(t, fmt.Sprintf("gaiacli query stake validator %s --output=json %v", sdk.ValAddress(barAddr), flags)) require.Equal(t, "1.0000000000", validator.Tokens.String()) validatorUbds := executeGetValidatorUnbondingDelegations(t, - fmt.Sprintf("gaiacli query unbonding-delegations-from %s --output=json %v", + fmt.Sprintf("gaiacli query stake unbonding-delegations-from %s --output=json %v", sdk.ValAddress(barAddr), flags)) require.Len(t, validatorUbds, 1) require.Equal(t, "1", validatorUbds[0].Balance.Amount.String()) - params := executeGetParams(t, fmt.Sprintf("gaiacli query parameters --output=json %v", flags)) + params := executeGetParams(t, fmt.Sprintf("gaiacli query stake parameters --output=json %v", flags)) require.True(t, defaultParams.Equal(params)) - pool := executeGetPool(t, fmt.Sprintf("gaiacli query pool --output=json %v", flags)) + pool := executeGetPool(t, fmt.Sprintf("gaiacli query stake pool --output=json %v", flags)) require.Equal(t, initialPool.BondedTokens, pool.BondedTokens) } @@ -313,16 +317,20 @@ func TestGaiaCLISubmitProposal(t *testing.T) { tests.WaitForTMStart(port) tests.WaitForNextNBlocksTM(2, port) + executeGetDepositParam(t, fmt.Sprintf("gaiacli query gov param deposit %v", flags)) + executeGetVotingParam(t, fmt.Sprintf("gaiacli query gov param voting %v", flags)) + executeGetTallyingParam(t, fmt.Sprintf("gaiacli query gov param tallying %v", flags)) + fooAddr, _ := executeGetAddrPK(t, fmt.Sprintf("gaiacli keys show foo --output=json --home=%s", gaiacliHome)) fooAcc := executeGetAccount(t, fmt.Sprintf("gaiacli query account %s %v", fooAddr, flags)) require.Equal(t, int64(50), fooAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64()) - proposalsQuery, _ := tests.ExecuteT(t, fmt.Sprintf("gaiacli query proposals %v", flags), "") + proposalsQuery, _ := tests.ExecuteT(t, fmt.Sprintf("gaiacli query gov proposals %v", flags), "") require.Equal(t, "No matching proposals found", proposalsQuery) // submit a test proposal - spStr := fmt.Sprintf("gaiacli tx submit-proposal %v", flags) + spStr := fmt.Sprintf("gaiacli tx gov submit-proposal %v", flags) spStr += fmt.Sprintf(" --from=%s", "foo") spStr += fmt.Sprintf(" --deposit=%s", fmt.Sprintf("5%s", stakeTypes.DefaultBondDenom)) spStr += fmt.Sprintf(" --type=%s", "Text") @@ -349,19 +357,19 @@ func TestGaiaCLISubmitProposal(t *testing.T) { fooAcc = executeGetAccount(t, fmt.Sprintf("gaiacli query account %s %v", fooAddr, flags)) require.Equal(t, int64(45), fooAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64()) - proposal1 := executeGetProposal(t, fmt.Sprintf("gaiacli query proposal --proposal-id=1 --output=json %v", flags)) + proposal1 := executeGetProposal(t, fmt.Sprintf("gaiacli query gov proposal --proposal-id=1 --output=json %v", flags)) require.Equal(t, uint64(1), proposal1.GetProposalID()) require.Equal(t, gov.StatusDepositPeriod, proposal1.GetStatus()) - proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query proposals %v", flags), "") + proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query gov proposals %v", flags), "") require.Equal(t, " 1 - Test", proposalsQuery) deposit := executeGetDeposit(t, - fmt.Sprintf("gaiacli query deposit --proposal-id=1 --depositer=%s --output=json %v", + fmt.Sprintf("gaiacli query gov deposit --proposal-id=1 --depositer=%s --output=json %v", fooAddr, flags)) require.Equal(t, int64(5), deposit.Amount.AmountOf(stakeTypes.DefaultBondDenom).Int64()) - depositStr := fmt.Sprintf("gaiacli tx deposit %v", flags) + depositStr := fmt.Sprintf("gaiacli tx gov deposit %v", flags) depositStr += fmt.Sprintf(" --from=%s", "foo") depositStr += fmt.Sprintf(" --deposit=%s", fmt.Sprintf("10%s", stakeTypes.DefaultBondDenom)) depositStr += fmt.Sprintf(" --proposal-id=%s", "1") @@ -381,22 +389,23 @@ func TestGaiaCLISubmitProposal(t *testing.T) { // test query deposit deposits := executeGetDeposits(t, - fmt.Sprintf("gaiacli query deposits --proposal-id=1 --output=json %v", flags)) + fmt.Sprintf("gaiacli query gov deposits --proposal-id=1 --output=json %v", flags)) require.Len(t, deposits, 1) require.Equal(t, int64(15), deposits[0].Amount.AmountOf(stakeTypes.DefaultBondDenom).Int64()) deposit = executeGetDeposit(t, - fmt.Sprintf("gaiacli query deposit --proposal-id=1 --depositer=%s --output=json %v", + fmt.Sprintf("gaiacli query gov deposit --proposal-id=1 --depositer=%s --output=json %v", fooAddr, flags)) require.Equal(t, int64(15), deposit.Amount.AmountOf(stakeTypes.DefaultBondDenom).Int64()) fooAcc = executeGetAccount(t, fmt.Sprintf("gaiacli query account %s %v", fooAddr, flags)) + require.Equal(t, int64(35), fooAcc.GetCoins().AmountOf(stakeTypes.DefaultBondDenom).Int64()) - proposal1 = executeGetProposal(t, fmt.Sprintf("gaiacli query proposal --proposal-id=1 --output=json %v", flags)) + proposal1 = executeGetProposal(t, fmt.Sprintf("gaiacli query gov proposal --proposal-id=1 --output=json %v", flags)) require.Equal(t, uint64(1), proposal1.GetProposalID()) require.Equal(t, gov.StatusVotingPeriod, proposal1.GetStatus()) - voteStr := fmt.Sprintf("gaiacli tx vote %v", flags) + voteStr := fmt.Sprintf("gaiacli tx gov vote %v", flags) voteStr += fmt.Sprintf(" --from=%s", "foo") voteStr += fmt.Sprintf(" --proposal-id=%s", "1") voteStr += fmt.Sprintf(" --option=%s", "Yes") @@ -414,23 +423,23 @@ func TestGaiaCLISubmitProposal(t *testing.T) { executeWrite(t, voteStr, app.DefaultKeyPass) tests.WaitForNextNBlocksTM(2, port) - vote := executeGetVote(t, fmt.Sprintf("gaiacli query vote --proposal-id=1 --voter=%s --output=json %v", fooAddr, flags)) + vote := executeGetVote(t, fmt.Sprintf("gaiacli query gov vote --proposal-id=1 --voter=%s --output=json %v", fooAddr, flags)) require.Equal(t, uint64(1), vote.ProposalID) require.Equal(t, gov.OptionYes, vote.Option) - votes := executeGetVotes(t, fmt.Sprintf("gaiacli query votes --proposal-id=1 --output=json %v", flags)) + votes := executeGetVotes(t, fmt.Sprintf("gaiacli query gov votes --proposal-id=1 --output=json %v", flags)) require.Len(t, votes, 1) require.Equal(t, uint64(1), votes[0].ProposalID) require.Equal(t, gov.OptionYes, votes[0].Option) - proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query proposals --status=DepositPeriod %v", flags), "") + proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query gov proposals --status=DepositPeriod %v", flags), "") require.Equal(t, "No matching proposals found", proposalsQuery) - proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query proposals --status=VotingPeriod %v", flags), "") + proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query gov proposals --status=VotingPeriod %v", flags), "") require.Equal(t, " 1 - Test", proposalsQuery) // submit a second test proposal - spStr = fmt.Sprintf("gaiacli tx submit-proposal %v", flags) + spStr = fmt.Sprintf("gaiacli tx gov submit-proposal %v", flags) spStr += fmt.Sprintf(" --from=%s", "foo") spStr += fmt.Sprintf(" --deposit=%s", fmt.Sprintf("5%s", stakeTypes.DefaultBondDenom)) spStr += fmt.Sprintf(" --type=%s", "Text") @@ -440,7 +449,7 @@ func TestGaiaCLISubmitProposal(t *testing.T) { executeWrite(t, spStr, app.DefaultKeyPass) tests.WaitForNextNBlocksTM(2, port) - proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query proposals --limit=1 %v", flags), "") + proposalsQuery, _ = tests.ExecuteT(t, fmt.Sprintf("gaiacli query gov proposals --limit=1 %v", flags), "") require.Equal(t, " 2 - Apples", proposalsQuery) } @@ -553,7 +562,6 @@ func TestGaiaCLIConfig(t *testing.T) { config, err := ioutil.ReadFile(path.Join(gaiacliHome, "config", "config.toml")) require.NoError(t, err) expectedConfig := fmt.Sprintf(`chain_id = "%s" -encoding = "btc" home = "%s" node = "%s" output = "text" @@ -572,7 +580,6 @@ trust_node = true // ensure it works without an initialized gaiad state expectedConfig = fmt.Sprintf(`chain_id = "" -encoding = "btc" home = "%s" node = "%s" output = "text" @@ -751,6 +758,15 @@ func executeGetValidatorRedelegations(t *testing.T, cmdStr string) []stake.Redel return reds } +func executeGetValidatorDelegations(t *testing.T, cmdStr string) []stake.Delegation { + out, _ := tests.ExecuteT(t, cmdStr, "") + var delegations []stake.Delegation + cdc := app.MakeCodec() + err := cdc.UnmarshalJSON([]byte(out), &delegations) + require.NoError(t, err, "out %v\n, err %v", out, err) + return delegations +} + func executeGetPool(t *testing.T, cmdStr string) stake.Pool { out, _ := tests.ExecuteT(t, cmdStr, "") var pool stake.Pool @@ -772,6 +788,33 @@ func executeGetParams(t *testing.T, cmdStr string) stake.Params { //___________________________________________________________________________________ // gov +func executeGetDepositParam(t *testing.T, cmdStr string) gov.DepositParams { + out, _ := tests.ExecuteT(t, cmdStr, "") + var depositParam gov.DepositParams + cdc := app.MakeCodec() + err := cdc.UnmarshalJSON([]byte(out), &depositParam) + require.NoError(t, err, "out %v\n, err %v", out, err) + return depositParam +} + +func executeGetVotingParam(t *testing.T, cmdStr string) gov.VotingParams { + out, _ := tests.ExecuteT(t, cmdStr, "") + var votingParam gov.VotingParams + cdc := app.MakeCodec() + err := cdc.UnmarshalJSON([]byte(out), &votingParam) + require.NoError(t, err, "out %v\n, err %v", out, err) + return votingParam +} + +func executeGetTallyingParam(t *testing.T, cmdStr string) gov.TallyParams { + out, _ := tests.ExecuteT(t, cmdStr, "") + var tallyingParam gov.TallyParams + cdc := app.MakeCodec() + err := cdc.UnmarshalJSON([]byte(out), &tallyingParam) + require.NoError(t, err, "out %v\n, err %v", out, err) + return tallyingParam +} + func executeGetProposal(t *testing.T, cmdStr string) gov.Proposal { out, _ := tests.ExecuteT(t, cmdStr, "") var proposal gov.Proposal diff --git a/cmd/gaia/cmd/gaiacli/main.go b/cmd/gaia/cmd/gaiacli/main.go index 212a67021..de99b0fc8 100644 --- a/cmd/gaia/cmd/gaiacli/main.go +++ b/cmd/gaia/cmd/gaiacli/main.go @@ -13,18 +13,10 @@ import ( "github.com/cosmos/cosmos-sdk/client/keys" "github.com/cosmos/cosmos-sdk/client/lcd" "github.com/cosmos/cosmos-sdk/client/rpc" - "github.com/cosmos/cosmos-sdk/client/tx" "github.com/cosmos/cosmos-sdk/cmd/gaia/app" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/version" - authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" - bankcmd "github.com/cosmos/cosmos-sdk/x/bank/client/cli" - distrcmd "github.com/cosmos/cosmos-sdk/x/distribution/client/cli" - govcmd "github.com/cosmos/cosmos-sdk/x/gov/client/cli" - slashingcmd "github.com/cosmos/cosmos-sdk/x/slashing/client/cli" - stakecmd "github.com/cosmos/cosmos-sdk/x/stake/client/cli" - _ "github.com/cosmos/cosmos-sdk/client/lcd/statik" ) @@ -40,7 +32,7 @@ const ( var ( rootCmd = &cobra.Command{ Use: "gaiacli", - Short: "Gaia light-client", + Short: "Command line interface for interacting with gaiad", } ) @@ -57,90 +49,23 @@ func main() { // TODO: setup keybase, viper object, etc. to be passed into // the below functions and eliminate global vars, like we do // with the cdc - rootCmd.AddCommand(client.ConfigCmd()) - // add standard rpc commands - rpc.AddCommands(rootCmd) - - //Add query commands - queryCmd := &cobra.Command{ - Use: "query", - Aliases: []string{"q"}, - Short: "Querying subcommands", - } - queryCmd.AddCommand( - rpc.BlockCommand(), - rpc.ValidatorCommand(), - ) - tx.AddCommands(queryCmd, cdc) - queryCmd.AddCommand(client.LineBreak) - queryCmd.AddCommand(client.GetCommands( - authcmd.GetAccountCmd(storeAcc, cdc, authcmd.GetAccountDecoder(cdc)), - stakecmd.GetCmdQueryDelegation(storeStake, cdc), - stakecmd.GetCmdQueryDelegations(storeStake, cdc), - stakecmd.GetCmdQueryUnbondingDelegation(storeStake, cdc), - stakecmd.GetCmdQueryUnbondingDelegations(storeStake, cdc), - stakecmd.GetCmdQueryRedelegation(storeStake, cdc), - stakecmd.GetCmdQueryRedelegations(storeStake, cdc), - stakecmd.GetCmdQueryValidator(storeStake, cdc), - stakecmd.GetCmdQueryValidators(storeStake, cdc), - stakecmd.GetCmdQueryValidatorUnbondingDelegations(queryRouteStake, cdc), - stakecmd.GetCmdQueryValidatorRedelegations(queryRouteStake, cdc), - stakecmd.GetCmdQueryParams(storeStake, cdc), - stakecmd.GetCmdQueryPool(storeStake, cdc), - govcmd.GetCmdQueryProposal(storeGov, cdc), - govcmd.GetCmdQueryProposals(storeGov, cdc), - govcmd.GetCmdQueryVote(storeGov, cdc), - govcmd.GetCmdQueryVotes(storeGov, cdc), - govcmd.GetCmdQueryDeposit(storeGov, cdc), - govcmd.GetCmdQueryDeposits(storeGov, cdc), - slashingcmd.GetCmdQuerySigningInfo(storeSlashing, cdc), - )...) - - //Add query commands - txCmd := &cobra.Command{ - Use: "tx", - Short: "Transactions subcommands", - } - - //Add auth and bank commands - txCmd.AddCommand( - client.PostCommands( - bankcmd.GetBroadcastCommand(cdc), - authcmd.GetSignCommand(cdc, authcmd.GetAccountDecoder(cdc)), - )...) - txCmd.AddCommand(client.LineBreak) - - txCmd.AddCommand( - client.PostCommands( - stakecmd.GetCmdCreateValidator(cdc), - stakecmd.GetCmdEditValidator(cdc), - stakecmd.GetCmdDelegate(cdc), - stakecmd.GetCmdRedelegate(storeStake, cdc), - stakecmd.GetCmdUnbond(storeStake, cdc), - distrcmd.GetCmdWithdrawRewards(cdc), - distrcmd.GetCmdSetWithdrawAddr(cdc), - govcmd.GetCmdDeposit(cdc), - bankcmd.SendTxCmd(cdc), - govcmd.GetCmdSubmitProposal(cdc), - slashingcmd.GetCmdUnjail(cdc), - govcmd.GetCmdVote(cdc), - )...) + // Construct Root Command rootCmd.AddCommand( - queryCmd, - txCmd, + rpc.InitClientCommand(), + rpc.StatusCommand(), + client.ConfigCmd(), + queryCmd(cdc), + txCmd(cdc), + client.LineBreak, lcd.ServeCommand(cdc), client.LineBreak, - ) - - // add proxy, version and key info - rootCmd.AddCommand( keys.Commands(), client.LineBreak, version.VersionCmd, ) - // prepare and add flags + // Add flags and prefix all env exposed with GA executor := cli.PrepareMainCmd(rootCmd, "GA", app.DefaultCLIHome) err := initConfig(rootCmd) if err != nil { diff --git a/cmd/gaia/cmd/gaiacli/query.go b/cmd/gaia/cmd/gaiacli/query.go new file mode 100644 index 000000000..3a806d36c --- /dev/null +++ b/cmd/gaia/cmd/gaiacli/query.go @@ -0,0 +1,83 @@ +package main + +import ( + "github.com/cosmos/cosmos-sdk/client" + "github.com/cosmos/cosmos-sdk/client/rpc" + "github.com/cosmos/cosmos-sdk/client/tx" + "github.com/spf13/cobra" + + authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" + govcmd "github.com/cosmos/cosmos-sdk/x/gov/client/cli" + slashingcmd "github.com/cosmos/cosmos-sdk/x/slashing/client/cli" + stakecmd "github.com/cosmos/cosmos-sdk/x/stake/client/cli" + amino "github.com/tendermint/go-amino" +) + +func queryCmd(cdc *amino.Codec) *cobra.Command { + //Add query commands + queryCmd := &cobra.Command{ + Use: "query", + Aliases: []string{"q"}, + Short: "Querying subcommands", + } + + // Group staking queries under a subcommand + stakeQueryCmd := &cobra.Command{ + Use: "stake", + Short: "Querying commands for the staking module", + } + + stakeQueryCmd.AddCommand(client.GetCommands( + stakecmd.GetCmdQueryDelegation(storeStake, cdc), + stakecmd.GetCmdQueryDelegations(storeStake, cdc), + stakecmd.GetCmdQueryUnbondingDelegation(storeStake, cdc), + stakecmd.GetCmdQueryUnbondingDelegations(storeStake, cdc), + stakecmd.GetCmdQueryRedelegation(storeStake, cdc), + stakecmd.GetCmdQueryRedelegations(storeStake, cdc), + stakecmd.GetCmdQueryValidator(storeStake, cdc), + stakecmd.GetCmdQueryValidators(storeStake, cdc), + stakecmd.GetCmdQueryValidatorDelegations(storeStake, cdc), + stakecmd.GetCmdQueryValidatorUnbondingDelegations(queryRouteStake, cdc), + stakecmd.GetCmdQueryValidatorRedelegations(queryRouteStake, cdc), + stakecmd.GetCmdQueryParams(storeStake, cdc), + stakecmd.GetCmdQueryPool(storeStake, cdc))...) + + // Group gov queries under a subcommand + govQueryCmd := &cobra.Command{ + Use: "gov", + Short: "Querying commands for the governance module", + } + + govQueryCmd.AddCommand(client.GetCommands( + govcmd.GetCmdQueryProposal(storeGov, cdc), + govcmd.GetCmdQueryProposals(storeGov, cdc), + govcmd.GetCmdQueryVote(storeGov, cdc), + govcmd.GetCmdQueryVotes(storeGov, cdc), + govcmd.GetCmdQueryParams(storeGov, cdc), + govcmd.GetCmdQueryDeposit(storeGov, cdc), + govcmd.GetCmdQueryDeposits(storeGov, cdc))...) + + // Group slashing queries under a subcommand + slashingQueryCmd := &cobra.Command{ + Use: "slashing", + Short: "Querying commands for the slashing module", + } + + slashingQueryCmd.AddCommand(client.GetCommands( + slashingcmd.GetCmdQuerySigningInfo(storeSlashing, cdc))...) + + // Query commcmmand structure + queryCmd.AddCommand( + rpc.BlockCommand(), + rpc.ValidatorCommand(), + tx.SearchTxCmd(cdc), + tx.QueryTxCmd(cdc), + client.LineBreak, + client.GetCommands(authcmd.GetAccountCmd(storeAcc, cdc, authcmd.GetAccountDecoder(cdc)))[0], + stakeQueryCmd, + govQueryCmd, + slashingQueryCmd, + ) + + return queryCmd +} diff --git a/cmd/gaia/cmd/gaiacli/tx.go b/cmd/gaia/cmd/gaiacli/tx.go new file mode 100644 index 000000000..fa0abc4ad --- /dev/null +++ b/cmd/gaia/cmd/gaiacli/tx.go @@ -0,0 +1,83 @@ +package main + +import ( + "github.com/cosmos/cosmos-sdk/client" + "github.com/spf13/cobra" + + authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" + bankcmd "github.com/cosmos/cosmos-sdk/x/bank/client/cli" + distrcmd "github.com/cosmos/cosmos-sdk/x/distribution/client/cli" + govcmd "github.com/cosmos/cosmos-sdk/x/gov/client/cli" + slashingcmd "github.com/cosmos/cosmos-sdk/x/slashing/client/cli" + stakecmd "github.com/cosmos/cosmos-sdk/x/stake/client/cli" + amino "github.com/tendermint/go-amino" +) + +func txCmd(cdc *amino.Codec) *cobra.Command { + //Add transaction generation commands + txCmd := &cobra.Command{ + Use: "tx", + Short: "Transactions subcommands", + } + + stakeTxCmd := &cobra.Command{ + Use: "stake", + Short: "Staking transaction subcommands", + } + + stakeTxCmd.AddCommand(client.PostCommands( + stakecmd.GetCmdCreateValidator(cdc), + stakecmd.GetCmdEditValidator(cdc), + stakecmd.GetCmdDelegate(cdc), + stakecmd.GetCmdRedelegate(storeStake, cdc), + stakecmd.GetCmdUnbond(storeStake, cdc), + )...) + + distTxCmd := &cobra.Command{ + Use: "dist", + Short: "Distribution transactions subcommands", + } + + distTxCmd.AddCommand(client.PostCommands( + distrcmd.GetCmdWithdrawRewards(cdc), + distrcmd.GetCmdSetWithdrawAddr(cdc), + )...) + + govTxCmd := &cobra.Command{ + Use: "gov", + Short: "Governance transactions subcommands", + } + + govTxCmd.AddCommand(client.PostCommands( + govcmd.GetCmdDeposit(cdc), + govcmd.GetCmdVote(cdc), + govcmd.GetCmdSubmitProposal(cdc), + )...) + + slashingTxCmd := &cobra.Command{ + Use: "slashing", + Short: "Slashing transactions subcommands", + } + + slashingTxCmd.AddCommand(client.PostCommands( + slashingcmd.GetCmdUnjail(cdc), + )...) + + txCmd.AddCommand( + //Add auth and bank commands + client.PostCommands( + bankcmd.SendTxCmd(cdc), + bankcmd.GetBroadcastCommand(cdc), + authcmd.GetSignCommand(cdc, authcmd.GetAccountDecoder(cdc)), + )...) + + txCmd.AddCommand( + client.LineBreak, + stakeTxCmd, + distTxCmd, + govTxCmd, + slashingTxCmd, + ) + + return txCmd +} diff --git a/cmd/gaia/init/init.go b/cmd/gaia/init/init.go index a297caee2..1f12da6bd 100644 --- a/cmd/gaia/init/init.go +++ b/cmd/gaia/init/init.go @@ -51,25 +51,27 @@ func InitCmd(ctx *server.Context, cdc *codec.Codec, appInit server.AppInit) *cob RunE: func(_ *cobra.Command, _ []string) error { config := ctx.Config config.SetRoot(viper.GetString(cli.HomeFlag)) + chainID := viper.GetString(client.FlagChainID) if chainID == "" { chainID = fmt.Sprintf("test-chain-%v", common.RandStr(6)) } + nodeID, _, err := InitializeNodeValidatorFiles(config) if err != nil { return err } - if viper.GetString(flagMoniker) != "" { - config.Moniker = viper.GetString(flagMoniker) - } + config.Moniker = viper.GetString(flagMoniker) var appState json.RawMessage genFile := config.GenesisFile() + if appState, err = initializeEmptyGenesis(cdc, genFile, chainID, viper.GetBool(flagOverwrite)); err != nil { return err } + if err = ExportGenesisFile(genFile, chainID, nil, appState); err != nil { return err } @@ -91,5 +93,7 @@ func InitCmd(ctx *server.Context, cdc *codec.Codec, appInit server.AppInit) *cob cmd.Flags().BoolP(flagOverwrite, "o", false, "overwrite the genesis.json file") cmd.Flags().String(client.FlagChainID, "", "genesis file chain-id, if left blank will be randomly created") cmd.Flags().String(flagMoniker, "", "set the validator's moniker") + cmd.MarkFlagRequired(flagMoniker) + return cmd } diff --git a/cmd/gaia/init/init_test.go b/cmd/gaia/init/init_test.go index bd5274c26..d1d9b9ce2 100644 --- a/cmd/gaia/init/init_test.go +++ b/cmd/gaia/init/init_test.go @@ -2,15 +2,16 @@ package init import ( "bytes" - "github.com/cosmos/cosmos-sdk/client" - "github.com/cosmos/cosmos-sdk/cmd/gaia/app" - "github.com/tendermint/tendermint/libs/cli" "io" "io/ioutil" "os" "testing" "time" + "github.com/cosmos/cosmos-sdk/client" + "github.com/cosmos/cosmos-sdk/cmd/gaia/app" + "github.com/tendermint/tendermint/libs/cli" + "github.com/cosmos/cosmos-sdk/server" "github.com/cosmos/cosmos-sdk/server/mock" "github.com/stretchr/testify/require" @@ -28,12 +29,16 @@ func TestInitCmd(t *testing.T) { logger := log.NewNopLogger() cfg, err := tcmd.ParseConfig() require.Nil(t, err) + ctx := server.NewContext(cfg, logger) cdc := app.MakeCodec() appInit := server.AppInit{ AppGenState: mock.AppGenState, } cmd := InitCmd(ctx, cdc, appInit) + + viper.Set(flagMoniker, "gaianode-test") + err = cmd.RunE(nil, nil) require.NoError(t, err) } @@ -53,14 +58,19 @@ func setupClientHome(t *testing.T) func() { func TestEmptyState(t *testing.T) { defer server.SetupViper(t)() defer setupClientHome(t)() + logger := log.NewNopLogger() cfg, err := tcmd.ParseConfig() require.Nil(t, err) + ctx := server.NewContext(cfg, logger) cdc := app.MakeCodec() appInit := server.AppInit{ AppGenState: mock.AppGenStateEmpty, } + + viper.Set(flagMoniker, "gaianode-test") + cmd := InitCmd(ctx, cdc, appInit) err = cmd.RunE(nil, nil) require.NoError(t, err) @@ -69,6 +79,7 @@ func TestEmptyState(t *testing.T) { r, w, _ := os.Pipe() os.Stdout = w cmd = server.ExportCmd(ctx, cdc, nil) + err = cmd.RunE(nil, nil) require.NoError(t, err) diff --git a/docs/.vuepress/config.js b/docs/.vuepress/config.js index 4746b3dd4..0774208ab 100644 --- a/docs/.vuepress/config.js +++ b/docs/.vuepress/config.js @@ -12,105 +12,39 @@ module.exports = { nav: [{ text: "Back to Cosmos", link: "https://cosmos.network" }], sidebar: [ { - title: "Introduction", + title: "Overview", collapsable: false, children: [ - "/introduction/cosmos-hub", - "/introduction/tendermint-cosmos", - "/introduction/tendermint" + "/intro/README", + "/intro/sdk-app-architecture", + "/intro/ocap" ] }, { - title: "Getting Started", + title: "Gaia", collapsable: false, children: [ - "/getting-started/voyager", - "/getting-started/installation", - "/getting-started/join-testnet", - "/getting-started/networks" - ] - }, - { - title: "Cosmos SDK", - collapsable: false, - children: [ - ["/sdk/overview", "Overview"], - ["/sdk/core/intro", "Core"], - "/sdk/core/app1", - "/sdk/core/app2", - "/sdk/core/app3", - "/sdk/core/app4", - "/sdk/core/app5", - // "/sdk/modules", - "/sdk/clients" - ] - }, - // { - // title: "Specifications", - // collapsable: false, - // children: [ - // ["/specs/overview", "Overview"], - // "/specs/governance", - // "/specs/ibc", - // "/specs/staking", - // "/specs/icts", - // ] - // }, - { - title: "SDK by Examples - Simple Governance", - collapsable: false, - children: [ - ["/sdk/sdk-by-examples/simple-governance/intro", "Intro"], - "/sdk/sdk-by-examples/simple-governance/setup-and-design", - "/sdk/sdk-by-examples/simple-governance/app-init", - "/sdk/sdk-by-examples/simple-governance/simple-gov-module", - "/sdk/sdk-by-examples/simple-governance/bridging-it-all", - "/sdk/sdk-by-examples/simple-governance/running-the-application" - ] - }, - { - title: "Light Client", - collapsable: false, - children: [ - "/lite/", - "/lite/getting_started" - ] - }, - { - title: "Lotion JS", - collapsable: false, - children: [ - ["/lotion/overview", "Overview"] - ] - }, - { - title: "Validators", - collapsable: false, - children: [ - ["/validators/overview", "Overview"], - ["/validators/security", "Security"], - ["/validators/validator-setup", "Validator Setup"], - "/validators/validator-faq" + "/gaia/installation", + "/gaia/join-testnet", + "/gaia/networks", + "/gaia/validators/overview", + "/gaia/validators/security", + "/gaia/validators/validator-faq", + "/gaia/validators/validator-setup", + "/gaia/ledger" ] }, { title: "Clients", collapsable: false, children: [ - ["/clients/service-providers", "Service Providers"] - ] - }, - { - title: "Resources", - collapsable: false, - children: [ - // ["/resources/faq" "General"], - "/resources/delegator-faq", - ["/resources/whitepaper", "Whitepaper - English"], - ["/resources/whitepaper-ko", "Whitepaper - 한국어"], - ["/resources/whitepaper-zh-CN", "Whitepaper - 中文"], - ["/resources/whitepaper-pt", "Whitepaper - Português"] - ] + "/clients/README", + "/lite/", // this renders the readme + "/lite/getting_started", + "/lite/specification", + "/clients/cli", + "/clients/service-providers" + ] } ] } diff --git a/docs/DOCS_README.md b/docs/DOCS_README.md index 48a31c2ea..a27f093b4 100644 --- a/docs/DOCS_README.md +++ b/docs/DOCS_README.md @@ -1,4 +1,8 @@ -# Docs Build Workflow +## Updating the docs + +If you want to open a PR on the Cosmos SDK to update the documentation, please follow the guidelines in the [`CONTRIBUTING.md`](https://github.com/cosmos/cosmos-sdk/tree/master/CONTRIBUTING.md) + +## Docs Build Workflow The documentation for the Cosmos SDK is hosted at: @@ -15,34 +19,34 @@ Besides, gaia-lite API docs are also provided by gaia-lite. The default API docs https://localhost:1317/swagger-ui/ ``` -## How It Works +### How It Works There is a Jenkins job listening for changes in the `/docs` directory, on both the `master` and `develop` branches. Any updates to files in this directory on those branches will automatically trigger a website deployment. Under the hood, a private website repository has make targets consumed by a standard Jenkins task. -## README +### README The [README.md](./README.md) is also the landing page for the documentation on the website. During the Jenkins build, the current commit is added to the bottom of the README. -## Config.js +### Config.js The [config.js](./.vuepress/config.js) generates the sidebar and Table of Contents on the website docs. Note the use of relative links and the omission of file extensions. Additional features are available to improve the look of the sidebar. -## Links +### Links **NOTE:** Strongly consider the existing links - both within this directory and to the website docs - when moving or deleting files. Relative links should be used nearly everywhere, having discovered and weighed the following: -### Relative +#### Relative Where is the other file, relative to the current one? @@ -50,7 +54,7 @@ Where is the other file, relative to the current one? - confusing / annoying to have things like: `../../../../myfile.md` - requires more updates when files are re-shuffled -### Absolute +#### Absolute Where is the other file, given the root of the repo? @@ -58,12 +62,12 @@ Where is the other file, given the root of the repo? - this is much nicer: `/docs/hereitis/myfile.md` - if you move that file around, the links inside it are preserved (but not to it, of course) -### Full +#### Full The full GitHub URL to a file or directory. Used occasionally when it makes sense to send users to the GitHub. -## Building Locally +### Building Locally To build and serve the documentation locally, run: @@ -94,12 +98,12 @@ python -m SimpleHTTPServer 8080 then navigate to localhost:8080 in your browser. -## Consistency +### Consistency Because the build processes are identical (as is the information contained herein), this file should be kept in sync as much as possible with its [counterpart in the Tendermint Core repo](https://github.com/tendermint/tendermint/blob/develop/docs/DOCS_README.md). -## Update and Build the RPC docs +### Update and Build the RPC docs 1. Execute the following command at the root directory to install the swagger-ui generate tool. ``` diff --git a/docs/README.md b/docs/README.md index 13b831f39..fbdf89797 100644 --- a/docs/README.md +++ b/docs/README.md @@ -1,24 +1,38 @@ -# Welcome to the Cosmos Docs! +# Welcome to the SDK Docs! -![cosmonaut reading the cosmos docs in space](./graphics/cosmos-docs.jpg) +Welcome to the SDK docs! -Cosmos is a decentralized network of independent parallel blockchains, each powered by classical BFT consensus algorithms like Tendermint. -The first blockchain in the Cosmos Network is the Cosmos Hub, whose native token is the Atom. Cosmos is a permission-less network, meaning that anybody can build a blockchain on it. +![cosmonaut reading the cosmos docs in space](./cosmos-docs.jpg) -Cosmos can interoperate with multiple other applications and cryptocurrencies. By creating a new zone, you can plug any blockchain system into the Cosmos hub and pass tokens back and forth between those zones, without the need for an intermediary. +## Learn the SDK -## Quick Start +### SDK Intro -- [Getting started with the SDK](./sdk/core/intro.md) -- [SDK Examples](../examples) -- [Join the testnet](./getting-started/join-testnet.md#run-a-full-node) +If you are a newcomer and would like to learn more about the Cosmos SDK, this **[intro](./intro/README.md)** is a good starting place. + +### SDK tutorial + +If you like to learn by doing, you can follow the **[SDK application tutorial](https://github.com/cosmos/sdk-application-tutorial)**. It showcases how to build an SDK-based blockchain from scratch, and teaches you about the basic principles the SDK in the process. + +## Use the SDK + +The following sections contain the information you need if you want to build a fully-functional SDK-based blockchain: + +>*NOTE*: We are currently working on improving the docs. Some info might be missing. If that is the case, try the Cosmos [Forum](https://forum.cosmos.network). Failing that, [open an issue](https://github.com/cosmos/cosmos-sdk/issues/new). + +- [Introduction](./intro/README.md): Contains introductory high-level material on the Cosmos SDK. +- [Gaia](./gaia/README.md): Contains all documentation related to the gaia application (current name for the Cosmos-Hub). +- [Clients](./clients/README.md): Documentation about SDK clients like the SDK Command-Line interface and the SDK Light-client. +- [Specifications](./spec/README.md): Contains SDK and modules specifications. + +If you are reading this on the Cosmos Website, please know that you can find more information on [github](https://github.com/cosmos/cosmos-sdk/tree/develop/docs). Also if you find any issues with the documentation please [*open a Pull Request*](https://github.com/cosmos/cosmos-sdk/compare?expand=1), or at least [*open an Issue*](https://github.com/cosmos/cosmos-sdk/issues/new) to update the docs! + +## Join the public testnet for the Cosmos Hub + +To install the latest version of the `gaia` application and join the public testnet, **click [here](./gaia/README.md#join-the-cosmos-hub-public-testnet)** ## Edit the Documentation See [this file](./DOCS_README.md) for details of the build process and considerations when making changes. - -## Version - -This documentation is built from the following commit: diff --git a/docs/clients/node.md b/docs/_attic/node.md similarity index 100% rename from docs/clients/node.md rename to docs/_attic/node.md diff --git a/docs/sdk/core/app1.md b/docs/_attic/sdk/core/app1.md similarity index 100% rename from docs/sdk/core/app1.md rename to docs/_attic/sdk/core/app1.md diff --git a/docs/sdk/core/app2.md b/docs/_attic/sdk/core/app2.md similarity index 100% rename from docs/sdk/core/app2.md rename to docs/_attic/sdk/core/app2.md diff --git a/docs/sdk/core/app3.md b/docs/_attic/sdk/core/app3.md similarity index 100% rename from docs/sdk/core/app3.md rename to docs/_attic/sdk/core/app3.md diff --git a/docs/sdk/core/app4.md b/docs/_attic/sdk/core/app4.md similarity index 100% rename from docs/sdk/core/app4.md rename to docs/_attic/sdk/core/app4.md diff --git a/docs/sdk/core/app5.md b/docs/_attic/sdk/core/app5.md similarity index 100% rename from docs/sdk/core/app5.md rename to docs/_attic/sdk/core/app5.md diff --git a/docs/sdk/core/examples/app1.go b/docs/_attic/sdk/core/examples/app1.go similarity index 100% rename from docs/sdk/core/examples/app1.go rename to docs/_attic/sdk/core/examples/app1.go diff --git a/docs/sdk/core/examples/app2.go b/docs/_attic/sdk/core/examples/app2.go similarity index 100% rename from docs/sdk/core/examples/app2.go rename to docs/_attic/sdk/core/examples/app2.go diff --git a/docs/sdk/core/examples/app2_test.go b/docs/_attic/sdk/core/examples/app2_test.go similarity index 100% rename from docs/sdk/core/examples/app2_test.go rename to docs/_attic/sdk/core/examples/app2_test.go diff --git a/docs/sdk/core/examples/app3.go b/docs/_attic/sdk/core/examples/app3.go similarity index 100% rename from docs/sdk/core/examples/app3.go rename to docs/_attic/sdk/core/examples/app3.go diff --git a/docs/sdk/core/examples/app4.go b/docs/_attic/sdk/core/examples/app4.go similarity index 100% rename from docs/sdk/core/examples/app4.go rename to docs/_attic/sdk/core/examples/app4.go diff --git a/docs/sdk/core/examples/app4_test.go b/docs/_attic/sdk/core/examples/app4_test.go similarity index 100% rename from docs/sdk/core/examples/app4_test.go rename to docs/_attic/sdk/core/examples/app4_test.go diff --git a/docs/sdk/core/intro.md b/docs/_attic/sdk/core/intro.md similarity index 100% rename from docs/sdk/core/intro.md rename to docs/_attic/sdk/core/intro.md diff --git a/docs/sdk/core/multistore.md b/docs/_attic/sdk/core/multistore.md similarity index 100% rename from docs/sdk/core/multistore.md rename to docs/_attic/sdk/core/multistore.md diff --git a/docs/sdk/sdk-by-examples/intro.md b/docs/_attic/sdk/sdk-by-examples/intro.md similarity index 100% rename from docs/sdk/sdk-by-examples/intro.md rename to docs/_attic/sdk/sdk-by-examples/intro.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/app-init.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/app-init.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/app-init.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/app-init.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/bridging-it-all.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/bridging-it-all.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/bridging-it-all.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/bridging-it-all.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/intro.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/intro.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/intro.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/intro.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/running-the-application.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/running-the-application.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/running-the-application.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/running-the-application.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/setup-and-design.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/setup-and-design.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/setup-and-design.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/setup-and-design.md diff --git a/docs/sdk/sdk-by-examples/simple-governance/simple-gov-module.md b/docs/_attic/sdk/sdk-by-examples/simple-governance/simple-gov-module.md similarity index 100% rename from docs/sdk/sdk-by-examples/simple-governance/simple-gov-module.md rename to docs/_attic/sdk/sdk-by-examples/simple-governance/simple-gov-module.md diff --git a/docs/architecture/adr-002-docs-structure.md b/docs/architecture/adr-002-docs-structure.md new file mode 100644 index 000000000..4d80ad711 --- /dev/null +++ b/docs/architecture/adr-002-docs-structure.md @@ -0,0 +1,94 @@ +# ADR 002: SDK Documentation Structure + +## Context + +There is a need for a scalable structure of the SDK documentation. Current documentation includes a lot of non-related SDK material, is difficult to maintain and hard to follow as a user. + +Ideally, we would have: +- All docs related to dev frameworks or tools live in their respective github repos (sdk repo would contain sdk docs, hub repo would contain hub docs, lotion repo would contain lotion docs, etc.) +- All other docs (faqs, whitepaper, high-level material about Cosmos) would live on the website. + +## Decision + +Re-structure the `/docs` folder of the SDK github repo as follows: + +``` +docs/ +├── README +├── intro/ +├── gaia/ +│ ├── overview.md +│ ├── install.md +│ ├── join-testnet.md +│ ├── validator-node.md +│ ├── validator-faq.md +│ └── delegator-faq.md +├── reference/ +│ ├── baseapp +│ ├── types +│ ├── store +│ ├── server +│ ├── modules/ +│ │ ├── keeper +│ │ ├── handler +│ │ ├── cli +│ ├── gas +│ └── commands +├── examples/ +│ ├── basecoin/ +│ └── democoin/ +├── clients/ +│ ├── lite/ +│ ├── service-providers +├── spec/ +└── architecture/ +``` + +The files in each sub-folders do not matter and will likely change. What matters is the sectioning: + +- `README`: Landing page of the docs. +- `into`: Introductory material. Goal is to have a short explainer of the SDK and then channel people to the resource they need. The [sdk-tutorial](https://github.com/cosmos/sdk-application-tutorial/) will be highlighted, as well as the `godocs`. +- `gaia`: Contains all docs related to the `gaia` application. Will later be renamed to `cosmos-hub` or `chub` and probably moved to its own repository. +- `reference`: Contains high-level explanations of the abstractions of the SDK. It does not contain specific code implementation and does not need to be updated often. **It is not an API specification of the interfaces**. API spec is the `godoc`. +- `examples`: Contain a couple examples of sdk application like `basecoin` and `democoin`. Developers need to maintain them up-to-date and make sure they compile as the SDK gets upgraded. +- `clients`: Contains specs and info about the various SDK clients. +- `spec`: Contains specs of modules, and others. +- `architecture`: Contains architecture-related docs like the present one. + +Website docs sidebar will only include the following sections: + +- `README` +- `intro` +- `gaia` +- `reference` +- `clients` + +`architecture` and `examples` need not be displayed on the website. As for `modules`, we might need to think about creating a modules manager, but this is out of scope for this document. + +## Status + +Proposed + +## Consequences + +### Positive + +- Much clearer organisation of the SDK docs. +- The `/docs` folder now only contains SDK and gaia related material. Later, it will only contain SDK related material. +- Developers only have to update `/docs` folder when they open a PR (and not `/examples` for example). +- Easier for developers to find what they need to update in the docs thanks to reworked architecture. +- Cleaner vuepress build for website docs. +- Will help build an executable doc (cf https://github.com/cosmos/cosmos-sdk/issues/2611) + +### Neutral + +- We need to move a bunch of deprecated stuff to `/_attic` folder. +- We need to integrate content in `docs/sdk/docs/core` in `reference`. +- We need to move all the content that currently lives in `docs` and does not fit in new structure (like `lotion`, intro material, whitepaper) to the website repository. +- Update `DOCS_README.md` + +## References + +- https://github.com/cosmos/cosmos-sdk/issues/1460 +- https://github.com/cosmos/cosmos-sdk/pull/2695 +- https://github.com/cosmos/cosmos-sdk/issues/2611 \ No newline at end of file diff --git a/docs/architecture/decision-records/README.md b/docs/architecture/decision-records/README.md deleted file mode 100644 index 7c669566d..000000000 --- a/docs/architecture/decision-records/README.md +++ /dev/null @@ -1,22 +0,0 @@ -# Architecture Decision Records (ADR) - -This is a location to record all high-level architecture decisions in the cosmos-sdk project. - -You can read more about the ADR concept in this [blog post](https://product.reverb.com/documenting-architecture-decisions-the-reverb-way-a3563bb24bd0#.78xhdix6t). - -An ADR should provide: - -- Context on the relevant goals and the current state -- Proposed changes to achieve the goals -- Summary of pros and cons -- References -- Changelog - -Note the distinction between an ADR and a spec. The ADR provides the context, intuition, reasoning, and -justification for a change in architecture, or for the architecture of something -new. The spec is much more compressed and streamlined summary of everything as -it stands today. - -If recorded decisions turned out to be lacking, convene a discussion, record the new decisions here, and then modify the code to match. - -Note the context/background should be written in the present tense. diff --git a/docs/architecture/decision-records/adr-001-message-counter.md b/docs/architecture/decision-records/adr-001-message-counter.md deleted file mode 100644 index 0e9f25990..000000000 --- a/docs/architecture/decision-records/adr-001-message-counter.md +++ /dev/null @@ -1,54 +0,0 @@ -# ADR 001: Global Message Counter - -## Context - -There is a desire for modules to have a concept of orderings between messages. - -One such example is in staking, we currently use an "intra bond tx counter" and -bond height. -The purpose these two serve is to providing an ordering for validators with equal stake, -for usage in the power-ranking of validators. -We can't use address here, as that would create a bad incentive to grind -addresses that optimized the sort function, which lowers the private key's -security. -Instead we order by whose transaction appeared first, as tracked by bondHeight -and intra bond tx counter. - -This logic however should not be unique to staking. -It is very conceivable that many modules in the future will want to be able to -know the ordering of messages / objects after they were initially created. - -## Decision - -Create a global message counter field of type int64. -Note that with int64's, there is no fear of overflow under normal use, -as it is only getting incremented by one, -and thus has a space of 9 quintillion values to go through. - -This counter must be persisted in state, but can just be read and written on -begin/end block respectively. -This field will get incremented upon every DeliverTx, -regardless if the transaction succeeds or not. -It must also be incremented within the check state for CheckTx. -The global message ordering field should be set within the context -so that modules can access it. - -## Corollary - Intra block ordering -In the event that there is desire to just have an intra block msg counter, -this can easily be derived from the global message counter. -Simply subtract current counter from first global message counter in the block. -Thus the relevant module could easily implement this. - -## Status -Proposed - -## Consequences - -### Positive -* Moves message ordering out of the set of things staking must keep track of -* Abstracts the logic well so other modules can use it - -### Negative -* Another thing to implement prelaunch. (Though this should be easy to implement) - -### Neutral diff --git a/docs/architecture/decision-records/adr-template.md b/docs/architecture/decision-records/adr-template.md deleted file mode 100644 index 6153a9d1e..000000000 --- a/docs/architecture/decision-records/adr-template.md +++ /dev/null @@ -1,32 +0,0 @@ -# ADR {ADR-NUMBER}: {TITLE} - -## Changelog -* {date}: {changelog} - -## Context -> This section contains all the context one needs to understand the current state, and why there is a problem. It should be as succinct as possible and introduce the high level idea behind the solution. - -## Decision -> This section explains all of the details of the proposed solution, including implementation details. -It should also describe affects / corollary items that may need to be changed as a part of this. -If the proposed change will be large, please also indicate a way to do the change to maximize ease of review. -(e.g. the optimal split of things to do between separate PR's) - -## Status -> A decision may be "proposed" if it hasn't been agreed upon yet, or "accepted" once it is agreed upon. If a later ADR changes or reverses a decision, it may be marked as "deprecated" or "superseded" with a reference to its replacement. - -{Deprecated|Proposed|Accepted} - -## Consequences -> This section describes the consequences, after applying the decision. All consequences should be summarized here, not just the "positive" ones. - -### Positive - -### Negative - -### Neutral - -## References -> Are there any relevant PR comments, issues that led up to this, or articles referrenced for why we made the given design choice? If so link them here! - -* {reference link} diff --git a/docs/clients/README.md b/docs/clients/README.md new file mode 100644 index 000000000..835b6ee50 --- /dev/null +++ b/docs/clients/README.md @@ -0,0 +1,18 @@ +# Clients + +This section explains contains information on clients for SDK based blockchain. + +>*NOTE*: This section is a WIP. + +## Light-client + +Light-clients enable users to interact with your application without having to download the entire state history but with a good level of security. + +- [Overview of light clients](./lite/README.md) +- [Starting a light-client server](./lite/getting_started.md) +- [Light-client specification](./lite/specification.md) + +## Other clients + +- [Command-Line interface for SDK-based blockchain](./cli.md) +- [Service provider doc](./service-providers.md) diff --git a/docs/clients/cli.md b/docs/clients/cli.md index dbd234d37..fda1ef60b 100644 --- a/docs/clients/cli.md +++ b/docs/clients/cli.md @@ -1,8 +1,3 @@ # CLI -See `gaiacli --help` for more details. - -Also see the [testnet -tutorial](https://github.com/cosmos/cosmos-sdk/tree/develop/cmd/gaia/testnets). - -TODO: cleanup the UX and document this properly. +> TODO: Rewrite this section to explain how CLI works for a generic SDK app. diff --git a/docs/lite/getting_started.md b/docs/clients/lite/getting_started.md similarity index 94% rename from docs/lite/getting_started.md rename to docs/clients/lite/getting_started.md index b602bcc47..15bc8a279 100644 --- a/docs/lite/getting_started.md +++ b/docs/clients/lite/getting_started.md @@ -13,7 +13,7 @@ To start a REST server, we need to specify the following parameters: For example:: ```bash -gaiacli rest-server --chain-id=test \ +gaiacli advanced rest-server --chain-id=test \ --laddr=tcp://localhost:1317 \ --node tcp://localhost:26657 \ --trust-node=false @@ -22,7 +22,7 @@ gaiacli rest-server --chain-id=test \ The server listens on HTTPS by default. You can set the SSL certificate to be used by the server with these additional flags: ```bash -gaiacli rest-server --chain-id=test \ +gaiacli advanced rest-server --chain-id=test \ --laddr=tcp://localhost:1317 \ --node tcp://localhost:26657 \ --trust-node=false \ diff --git a/docs/lite/pics/C2H.png b/docs/clients/lite/pics/C2H.png similarity index 100% rename from docs/lite/pics/C2H.png rename to docs/clients/lite/pics/C2H.png diff --git a/docs/lite/pics/H2C.png b/docs/clients/lite/pics/H2C.png similarity index 100% rename from docs/lite/pics/H2C.png rename to docs/clients/lite/pics/H2C.png diff --git a/docs/lite/pics/MA.png b/docs/clients/lite/pics/MA.png similarity index 100% rename from docs/lite/pics/MA.png rename to docs/clients/lite/pics/MA.png diff --git a/docs/lite/pics/absence1.png b/docs/clients/lite/pics/absence1.png similarity index 100% rename from docs/lite/pics/absence1.png rename to docs/clients/lite/pics/absence1.png diff --git a/docs/lite/pics/absence2.png b/docs/clients/lite/pics/absence2.png similarity index 100% rename from docs/lite/pics/absence2.png rename to docs/clients/lite/pics/absence2.png diff --git a/docs/lite/pics/absence3.png b/docs/clients/lite/pics/absence3.png similarity index 100% rename from docs/lite/pics/absence3.png rename to docs/clients/lite/pics/absence3.png diff --git a/docs/lite/pics/architecture.png b/docs/clients/lite/pics/architecture.png similarity index 100% rename from docs/lite/pics/architecture.png rename to docs/clients/lite/pics/architecture.png diff --git a/docs/lite/pics/changeProcess.png b/docs/clients/lite/pics/changeProcess.png similarity index 100% rename from docs/lite/pics/changeProcess.png rename to docs/clients/lite/pics/changeProcess.png diff --git a/docs/lite/pics/commitValidation.png b/docs/clients/lite/pics/commitValidation.png similarity index 100% rename from docs/lite/pics/commitValidation.png rename to docs/clients/lite/pics/commitValidation.png diff --git a/docs/lite/pics/create-account.png b/docs/clients/lite/pics/create-account.png similarity index 100% rename from docs/lite/pics/create-account.png rename to docs/clients/lite/pics/create-account.png diff --git a/docs/lite/pics/deposit.png b/docs/clients/lite/pics/deposit.png similarity index 100% rename from docs/lite/pics/deposit.png rename to docs/clients/lite/pics/deposit.png diff --git a/docs/lite/pics/existProof.png b/docs/clients/lite/pics/existProof.png similarity index 100% rename from docs/lite/pics/existProof.png rename to docs/clients/lite/pics/existProof.png diff --git a/docs/lite/pics/high-level.png b/docs/clients/lite/pics/high-level.png similarity index 100% rename from docs/lite/pics/high-level.png rename to docs/clients/lite/pics/high-level.png diff --git a/docs/lite/pics/light-client-architecture.png b/docs/clients/lite/pics/light-client-architecture.png similarity index 100% rename from docs/lite/pics/light-client-architecture.png rename to docs/clients/lite/pics/light-client-architecture.png diff --git a/docs/lite/pics/loadbalanceDiagram.png b/docs/clients/lite/pics/loadbalanceDiagram.png similarity index 100% rename from docs/lite/pics/loadbalanceDiagram.png rename to docs/clients/lite/pics/loadbalanceDiagram.png diff --git a/docs/lite/pics/simpleMerkleTree.png b/docs/clients/lite/pics/simpleMerkleTree.png similarity index 100% rename from docs/lite/pics/simpleMerkleTree.png rename to docs/clients/lite/pics/simpleMerkleTree.png diff --git a/docs/lite/pics/substoreProof.png b/docs/clients/lite/pics/substoreProof.png similarity index 100% rename from docs/lite/pics/substoreProof.png rename to docs/clients/lite/pics/substoreProof.png diff --git a/docs/lite/pics/transfer-tokens.png b/docs/clients/lite/pics/transfer-tokens.png similarity index 100% rename from docs/lite/pics/transfer-tokens.png rename to docs/clients/lite/pics/transfer-tokens.png diff --git a/docs/lite/pics/transfer.png b/docs/clients/lite/pics/transfer.png similarity index 100% rename from docs/lite/pics/transfer.png rename to docs/clients/lite/pics/transfer.png diff --git a/docs/lite/pics/trustPropagate.png b/docs/clients/lite/pics/trustPropagate.png similarity index 100% rename from docs/lite/pics/trustPropagate.png rename to docs/clients/lite/pics/trustPropagate.png diff --git a/docs/lite/pics/updateValidatorToHeight.png b/docs/clients/lite/pics/updateValidatorToHeight.png similarity index 100% rename from docs/lite/pics/updateValidatorToHeight.png rename to docs/clients/lite/pics/updateValidatorToHeight.png diff --git a/docs/lite/pics/validatorSetChange.png b/docs/clients/lite/pics/validatorSetChange.png similarity index 100% rename from docs/lite/pics/validatorSetChange.png rename to docs/clients/lite/pics/validatorSetChange.png diff --git a/docs/lite/pics/withdraw.png b/docs/clients/lite/pics/withdraw.png similarity index 100% rename from docs/lite/pics/withdraw.png rename to docs/clients/lite/pics/withdraw.png diff --git a/docs/lite/readme.md b/docs/clients/lite/readme.md similarity index 58% rename from docs/lite/readme.md rename to docs/clients/lite/readme.md index 215bc25f7..3811f3c3e 100644 --- a/docs/lite/readme.md +++ b/docs/clients/lite/readme.md @@ -16,65 +16,61 @@ maintaining any full blockchain nodes. ### What is a lite Client -The LCD is split into two separate components. The first component is generic for any Tendermint +The Cosmos SDK Light Client (Gaia-lite) is split into two separate components. The first component is generic for any Tendermint based application. It handles the security and connectivity aspects of following the header chain and verify proofs from full nodes against locally trusted validator set. Furthermore it exposes exactly the same API as any Tendermint Core node. The second component is specific for the Cosmos -Hub (Gaiad). It works as a query endpoint and exposes the application specific functionality, which +Hub (`gaiad`). It works as a query endpoint and exposes the application specific functionality, which can be arbitrary. All queries against the application state have to go through the query endpoint. The advantage of the query endpoint is that it can verify the proofs that the application returns. ### High-Level Architecture -An application developer that would like to build a third party integration can ship his application -with the LCD for the Cosmos Hub (or any other zone) and only needs to initialise it. Afterwards his -application can interact with the zone as if it was running against a full node. - -![high-level](pics/high-level.png) - -An application developer that wants to build an third party application for the Cosmos Hub (or any -other zone) should build it against it's canonical API. That API is a combination of multiple parts. +An application developer that wants to build a third party client application for the Cosmos Hub (or any +other zone) should build it against its canonical API. That API is a combination of multiple parts. All zones have to expose ICS0 (TendermintAPI). Beyond that any zone is free to choose any combination of module APIs, depending on which modules the state machine uses. The Cosmos Hub will -initially support ICS0 (TendermintAPI), ICS1 (KeyAPI), ICS20 (TokenAPI), ICS21 (StakingAPI) and -ICS22 (GovernanceAPI). +initially support [ICS0](https://cosmos.network/rpc/#/ICS0) (TendermintAPI), [ICS1](https://cosmos.network/rpc/#/ICS1) (KeyAPI), [ICS20](https://cosmos.network/rpc/#/ICS20) (TokenAPI), [ICS21](https://cosmos.network/rpc/#/ICS21) (StakingAPI), +[ICS22](https://cosmos.network/rpc/#/ICS22) (GovernanceAPI) and [ICS23](https://cosmos.network/rpc/#/ICS23) (SlashingAPI). -All applications are expected to only run against the LCD. The LCD is the only piece of software +![high-level](./pics/high-level.png) + +All applications are expected to only run against Gaia-lite. Gaia-lite is the only piece of software that offers stability guarantees around the zone API. ### Comparision A full node of ABCI is different from its lite client in the following ways: -|| Full Node | LCD | Description| +|| Full Node | Gaia-lite | Description| |-| ------------- | ----- | -------------- | -| Execute and verify transactions|Yes|No|Full node will execute and verify all transactions while LCD won't| -| Verify and save blocks|Yes|No|Full node will verify and save all blocks while LCD won't| -| Participate consensus| Yes|No|Only when the full node is a validtor, it will participate consensus. LCD nodes never participate consensus| -| Bandwidth cost|Huge|Little|Full node will receive all blocks. if the bandwidth is limited, it will fall behind the main network. What's more, if it happens to be a validator,it will slow down the consensus process. LCD requires little bandwidth. Only when serving local request, it will cost bandwidth| +| Execute and verify transactions|Yes|No|Full node will execute and verify all transactions while Gaia-lite won't| +| Verify and save blocks|Yes|No|Full node will verify and save all blocks while Gaia-lite won't| +| Participate consensus| Yes|No|Only when the full node is a validtor, it will participate consensus. Lite nodes never participate consensus| +| Bandwidth cost|Huge|Little|Full node will receive all blocks. if the bandwidth is limited, it will fall behind the main network. What's more, if it happens to be a validator,it will slow down the consensus process. Light clients requires little bandwidth. Only when serving local request, it will cost bandwidth| | Computing resource|Huge|Little|Full node will execute all transactions and verify all blocks which require much computing resource| -| Storage resource|Huge|Little|Full node will save all blocks and ABCI states. LCD just saves validator sets and some checkpoints| -| Power consume|Huge|Little|Full nodes have to be deployed on machines which have high performance and will be running all the time. So power consume will be huge. LCD can be deployed on the same machines as users' applications, or on independent machines but with poor performance. Besides, LCD can be shutdown anytime when necessary. So LCD only consume very little power, even mobile devices can meet the power requirement| -| Provide APIs|All cosmos APIs|Modular APIs|Full node supports all cosmos APIs. LCD provides modular APIs according to users' configuration| -| Secuity level| High|High|Full node will verify all transactions and blocks by itself. LCD can't do this, but it can query any data from other full nodes and verify the data independently. So both full node and LCD don't need to trust any third nodes, they all can achieve high security| +| Storage resource|Huge|Little|Full node will save all blocks and ABCI states. Gaia-lite just saves validator sets and some checkpoints| +| Power consume|Huge|Little|Full nodes have to be deployed on machines which have high performance and will be running all the time. So power consume will be huge. Gaia-lite can be deployed on the same machines as users' applications, or on independent machines but with poor performance. Besides, lite clients can be shutdown anytime when necessary. So Gaia-lite only consume very little power, even mobile devices can meet the power requirement| +| Provide APIs|All cosmos APIs|Modular APIs|Full node supports all cosmos APIs. Gaia-lite provides modular APIs according to users' configuration| +| Secuity level| High|High|Full node will verify all transactions and blocks by itself. A light client can't do this, but it can query any data from other full nodes and verify the data independently. So both full nodes and light clients don't need to trust any third nodes, they all can achieve high security| -According to the above table, LCD can meet all users' functionality and security requirements, but +According to the above table, Gaia-lite can meet all users' functionality and security requirements, but only requires little resource on bandwidth, computing, storage and power. ## Achieving Security ### Trusted Validator Set -The base design philosophy of the LCD follows two rules: +The base design philosophy of Gaia-lite follows two rules: 1. **Doesn't trust any blockchain nodes, including validator nodes and other full nodes** 2. **Only trusts the whole validator set** The original trusted validator set should be prepositioned into its trust store, usually this -validator set comes from genesis file. During runtime, if LCD detects a different validator set, +validator set comes from genesis file. During runtime, if Gaia-lite detects a different validator set, it will verify it and save new validated validator set to the trust store. -![validator-set-change](pics/validatorSetChange.png) +![validator-set-change](./pics/validatorSetChange.png) ### Trust Propagation @@ -83,8 +79,8 @@ validator set evolution. Validator set is the foundation of trust, and the trust other blockchain data, such as block and transaction. The propagate architecture is shown as follows: -![change-process](pics/trustPropagate.png) +![change-process](./pics/trustPropagate.png) -In general, by trusted validator set, LCD can verify each block commit which contains all pre-commit +In general, by trusted validator set, a light client can verify each block commit which contains all pre-commit data and block header data. Then the block hash, data hash and appHash are trusted. Based on this and merkle proof, all transactions data and ABCI states can be verified too. diff --git a/docs/lite/specification.md b/docs/clients/lite/specification.md similarity index 95% rename from docs/lite/specification.md rename to docs/clients/lite/specification.md index 4feac508c..032f840a5 100644 --- a/docs/lite/specification.md +++ b/docs/clients/lite/specification.md @@ -12,7 +12,7 @@ we need to extract name, height and store root hash from these substores to buil Merkle leaf nodes, then calculate hash from leaf nodes to root. The root hash of the simple Merkle tree is the AppHash which will be included in block header. -![Simple Merkle Tree](pics/simpleMerkleTree.png) +![Simple Merkle Tree](./pics/simpleMerkleTree.png) As we have discussed in [LCD trust-propagation](https://github.com/irisnet/cosmos-sdk/tree/bianjie/lcd_spec/docs/spec/lcd#trust-propagation), the AppHash can be verified by checking voting power against a trusted validator set. Here we just @@ -65,7 +65,7 @@ type KeyExistsProof struct { The data structure of exist proof is shown as above. The process to build and verify existance proof is shown as follows: -![Exist Proof](pics/existProof.png) +![Exist Proof](./pics/existProof.png) Steps to build proof: @@ -92,12 +92,12 @@ the postition of the target key in the whole key set of this IAVL tree. As shown out the left key and the right key. If we can demonstrate that both left key and right key definitely exist, and they are adjacent nodes. Thus the target key definitely doesn't exist. -![Absence Proof1](pics/absence1.png) +![Absence Proof1](./pics/absence1.png) If the target key is larger than the right most leaf node or less than the left most key, then the target key definitely doesn't exist. -![Absence Proof2](pics/absence2.png)![Absence Proof3](pics/absence3.png) +![Absence Proof2](./pics/absence2.png)![Absence Proof3](./pics/absence3.png) ```go type proofLeafNode struct { @@ -147,7 +147,7 @@ in commitID equals to proof RootHash. If not, the proof is invalid. Then sort th commitInfo array by the hash of substore name. Finally, build the simple Merkle tree with all substore commitInfo array and verify if the Merkle root hash equal to appHash. -![substore proof](pics/substoreProof.png) +![substore proof](./pics/substoreProof.png) ```go func SimpleHashFromTwoHashes(left []byte, right []byte) []byte { @@ -187,7 +187,7 @@ Above sections refer appHash frequently. But where does the trusted appHash come appHash exist in block header, so next we need to verify blocks header at specific height against LCD trusted validator set. The validation flow is shown as follows: -![commit verification](pics/commitValidation.png) +![commit verification](./pics/commitValidation.png) When the trusted validator set doesn't match the block header, we need to try to update our validator set to the height of this block. LCD have a rule that each validator set change should not @@ -198,7 +198,7 @@ validator set update be accomplished. For instance: -![Update validator set to height](pics/updateValidatorToHeight.png) +![Update validator set to height](./pics/updateValidatorToHeight.png) * Update to 10000, tooMuchChangeErr * Update to 5050, tooMuchChangeErr diff --git a/docs/clients/rest.md b/docs/clients/rest.md deleted file mode 100644 index b48ec7d8f..000000000 --- a/docs/clients/rest.md +++ /dev/null @@ -1,6 +0,0 @@ -# REST - -See `gaiacli rest-server --help` for more. - -Also see the -[work in progress API specification](https://github.com/cosmos/cosmos-sdk/pull/1314) diff --git a/docs/clients/service-providers.md b/docs/clients/service-providers.md index 11fe2c779..b7e6d500c 100644 --- a/docs/clients/service-providers.md +++ b/docs/clients/service-providers.md @@ -83,7 +83,7 @@ It will display all the available commands. For each command, you can use the `- ## Setting up the Rest Server -The Rest Server acts as an intermediary between the front-end and the full-node. You don't need to run the Rest Server on the same machine as the full-node. If you intend to run the Rest Server on another machine, you need to go through the [Installation and configuration](#installation-and-configuration) again on this machine. +The Rest Server acts as an intermediary between the front-end and the full-node. You don't need to run the Rest Server on the same machine as the full-node. To start the Rest server: @@ -109,4 +109,6 @@ The Rest API documents all the available endpoints that you can use to interract The API is divided into ICS standards for each category of endpoints. For example, the [ICS20](https://cosmos.network/rpc/#/ICS20/) describes the API to interact with tokens. -To give more flexibility to implementers, we have separated the different steps that are involved in the process of sending transactions. You will be able to generate unsigned transactions (example with [coin transfer](https://cosmos.network/rpc/#/ICS20/post_bank_accounts__address__transfers)), [sign](https://cosmos.network/rpc/#/ICS20/post_tx_sign) and [broadcast](https://cosmos.network/rpc/#/ICS20/post_tx_broadcast) them with different API endpoints. This allows service providers to use their own signing mechanism for instance. +To give more flexibility to implementers, we have included the ability to generate unsigned transactions, [sign](https://cosmos.network/rpc/#/ICS20/post_tx_sign) and [broadcast](https://cosmos.network/rpc/#/ICS20/post_tx_broadcast) them with different API endpoints. This allows service providers to use their own signing mechanism for instance. + +In order to generate an unsigned transaction (example with [coin transfer](https://cosmos.network/rpc/#/ICS20/post_bank_accounts__address__transfers)), you need to use the flag `?generate_only`. diff --git a/docs/graphics/cosmos-docs.jpg b/docs/cosmos-docs.jpg similarity index 100% rename from docs/graphics/cosmos-docs.jpg rename to docs/cosmos-docs.jpg diff --git a/examples/README.md b/docs/examples/README.md similarity index 100% rename from examples/README.md rename to docs/examples/README.md diff --git a/examples/basecoin/app/app.go b/docs/examples/basecoin/app/app.go similarity index 98% rename from examples/basecoin/app/app.go rename to docs/examples/basecoin/app/app.go index 286afd003..9515feaad 100644 --- a/examples/basecoin/app/app.go +++ b/docs/examples/basecoin/app/app.go @@ -6,7 +6,7 @@ import ( bam "github.com/cosmos/cosmos-sdk/baseapp" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/basecoin/types" + "github.com/cosmos/cosmos-sdk/docs/examples/basecoin/types" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" "github.com/cosmos/cosmos-sdk/x/bank" diff --git a/examples/basecoin/app/app_test.go b/docs/examples/basecoin/app/app_test.go similarity index 97% rename from examples/basecoin/app/app_test.go rename to docs/examples/basecoin/app/app_test.go index c49d5d476..64bc6a86e 100644 --- a/examples/basecoin/app/app_test.go +++ b/docs/examples/basecoin/app/app_test.go @@ -5,7 +5,7 @@ import ( "testing" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/basecoin/types" + "github.com/cosmos/cosmos-sdk/docs/examples/basecoin/types" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" "github.com/stretchr/testify/require" diff --git a/examples/basecoin/cli_test/cli_test.go b/docs/examples/basecoin/cli_test/cli_test.go similarity index 100% rename from examples/basecoin/cli_test/cli_test.go rename to docs/examples/basecoin/cli_test/cli_test.go diff --git a/examples/basecoin/cmd/basecli/main.go b/docs/examples/basecoin/cmd/basecli/main.go similarity index 90% rename from examples/basecoin/cmd/basecli/main.go rename to docs/examples/basecoin/cmd/basecli/main.go index 1732e82eb..3a16c8a97 100644 --- a/examples/basecoin/cmd/basecli/main.go +++ b/docs/examples/basecoin/cmd/basecli/main.go @@ -7,8 +7,8 @@ import ( _ "github.com/cosmos/cosmos-sdk/client/lcd/statik" "github.com/cosmos/cosmos-sdk/client/rpc" "github.com/cosmos/cosmos-sdk/client/tx" - "github.com/cosmos/cosmos-sdk/examples/basecoin/app" - "github.com/cosmos/cosmos-sdk/examples/basecoin/types" + "github.com/cosmos/cosmos-sdk/docs/examples/basecoin/app" + "github.com/cosmos/cosmos-sdk/docs/examples/basecoin/types" "github.com/cosmos/cosmos-sdk/version" authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" bankcmd "github.com/cosmos/cosmos-sdk/x/bank/client/cli" @@ -39,10 +39,14 @@ func main() { // with the cdc. // add standard rpc, and tx commands - rpc.AddCommands(rootCmd) - rootCmd.AddCommand(client.LineBreak) - tx.AddCommands(rootCmd, cdc) - rootCmd.AddCommand(client.LineBreak) + rootCmd.AddCommand( + rpc.InitClientCommand(), + rpc.StatusCommand(), + client.LineBreak, + tx.SearchTxCmd(cdc), + tx.QueryTxCmd(cdc), + client.LineBreak, + ) // add query/post commands (custom to binary) rootCmd.AddCommand( diff --git a/examples/basecoin/cmd/basecoind/main.go b/docs/examples/basecoin/cmd/basecoind/main.go similarity index 98% rename from examples/basecoin/cmd/basecoind/main.go rename to docs/examples/basecoin/cmd/basecoind/main.go index e56654f51..7dd067358 100644 --- a/examples/basecoin/cmd/basecoind/main.go +++ b/docs/examples/basecoin/cmd/basecoind/main.go @@ -14,7 +14,7 @@ import ( "github.com/cosmos/cosmos-sdk/client" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/basecoin/app" + "github.com/cosmos/cosmos-sdk/docs/examples/basecoin/app" "github.com/cosmos/cosmos-sdk/server" "github.com/spf13/cobra" "github.com/spf13/viper" diff --git a/examples/basecoin/types/account.go b/docs/examples/basecoin/types/account.go similarity index 100% rename from examples/basecoin/types/account.go rename to docs/examples/basecoin/types/account.go diff --git a/examples/democoin/app/app.go b/docs/examples/democoin/app/app.go similarity index 95% rename from examples/democoin/app/app.go rename to docs/examples/democoin/app/app.go index e8ddd066d..bc618e360 100644 --- a/examples/democoin/app/app.go +++ b/docs/examples/democoin/app/app.go @@ -17,11 +17,11 @@ import ( "github.com/cosmos/cosmos-sdk/x/bank" "github.com/cosmos/cosmos-sdk/x/ibc" - "github.com/cosmos/cosmos-sdk/examples/democoin/types" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/cool" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/pow" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/simplestake" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/sketchy" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/types" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/cool" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/pow" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/simplestake" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/sketchy" ) const ( diff --git a/examples/democoin/app/app_test.go b/docs/examples/democoin/app/app_test.go similarity index 94% rename from examples/democoin/app/app_test.go rename to docs/examples/democoin/app/app_test.go index 5b3be00e1..93cef936c 100644 --- a/examples/democoin/app/app_test.go +++ b/docs/examples/democoin/app/app_test.go @@ -5,8 +5,8 @@ import ( "testing" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/types" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/cool" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/types" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/cool" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" "github.com/stretchr/testify/require" diff --git a/examples/democoin/cli_test/cli_test.go b/docs/examples/democoin/cli_test/cli_test.go similarity index 99% rename from examples/democoin/cli_test/cli_test.go rename to docs/examples/democoin/cli_test/cli_test.go index 8df97c987..9a3cba0df 100644 --- a/examples/democoin/cli_test/cli_test.go +++ b/docs/examples/democoin/cli_test/cli_test.go @@ -3,10 +3,11 @@ package clitest import ( "encoding/json" "fmt" - "github.com/cosmos/cosmos-sdk/cmd/gaia/app" "os" "testing" + "github.com/cosmos/cosmos-sdk/cmd/gaia/app" + "github.com/cosmos/cosmos-sdk/server" "github.com/cosmos/cosmos-sdk/tests" "github.com/stretchr/testify/require" diff --git a/examples/democoin/cmd/democli/main.go b/docs/examples/democoin/cmd/democli/main.go similarity index 81% rename from examples/democoin/cmd/democli/main.go rename to docs/examples/democoin/cmd/democli/main.go index 6bb2da868..adb6169c9 100644 --- a/examples/democoin/cmd/democli/main.go +++ b/docs/examples/democoin/cmd/democli/main.go @@ -16,11 +16,11 @@ import ( bankcmd "github.com/cosmos/cosmos-sdk/x/bank/client/cli" ibccmd "github.com/cosmos/cosmos-sdk/x/ibc/client/cli" - "github.com/cosmos/cosmos-sdk/examples/democoin/app" - "github.com/cosmos/cosmos-sdk/examples/democoin/types" - coolcmd "github.com/cosmos/cosmos-sdk/examples/democoin/x/cool/client/cli" - powcmd "github.com/cosmos/cosmos-sdk/examples/democoin/x/pow/client/cli" - simplestakingcmd "github.com/cosmos/cosmos-sdk/examples/democoin/x/simplestake/client/cli" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/app" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/types" + coolcmd "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/cool/client/cli" + powcmd "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/pow/client/cli" + simplestakingcmd "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/simplestake/client/cli" sdk "github.com/cosmos/cosmos-sdk/types" ) @@ -52,10 +52,15 @@ func main() { // with the cdc // add standard rpc, and tx commands - rpc.AddCommands(rootCmd) - rootCmd.AddCommand(client.LineBreak) - tx.AddCommands(rootCmd, cdc) - rootCmd.AddCommand(client.LineBreak) + + rootCmd.AddCommand( + rpc.InitClientCommand(), + rpc.StatusCommand(), + client.LineBreak, + tx.SearchTxCmd(cdc), + tx.QueryTxCmd(cdc), + client.LineBreak, + ) // add query/post commands (custom to binary) // start with commands common to basecoin diff --git a/examples/democoin/cmd/democoind/main.go b/docs/examples/democoin/cmd/democoind/main.go similarity index 98% rename from examples/democoin/cmd/democoind/main.go rename to docs/examples/democoin/cmd/democoind/main.go index 5acb459db..a6e9f8a22 100644 --- a/examples/democoin/cmd/democoind/main.go +++ b/docs/examples/democoin/cmd/democoind/main.go @@ -22,7 +22,7 @@ import ( gaiaInit "github.com/cosmos/cosmos-sdk/cmd/gaia/init" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/app" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/app" "github.com/cosmos/cosmos-sdk/server" sdk "github.com/cosmos/cosmos-sdk/types" ) diff --git a/examples/democoin/mock/validator.go b/docs/examples/democoin/mock/validator.go similarity index 100% rename from examples/democoin/mock/validator.go rename to docs/examples/democoin/mock/validator.go diff --git a/examples/democoin/types/account.go b/docs/examples/democoin/types/account.go similarity index 94% rename from examples/democoin/types/account.go rename to docs/examples/democoin/types/account.go index ad57c944d..341f6cfc8 100644 --- a/examples/democoin/types/account.go +++ b/docs/examples/democoin/types/account.go @@ -5,8 +5,8 @@ import ( sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/x/auth" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/cool" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/pow" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/cool" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/pow" ) var _ auth.Account = (*AppAccount)(nil) diff --git a/examples/democoin/x/assoc/validator_set.go b/docs/examples/democoin/x/assoc/validator_set.go similarity index 100% rename from examples/democoin/x/assoc/validator_set.go rename to docs/examples/democoin/x/assoc/validator_set.go diff --git a/examples/democoin/x/assoc/validator_set_test.go b/docs/examples/democoin/x/assoc/validator_set_test.go similarity index 97% rename from examples/democoin/x/assoc/validator_set_test.go rename to docs/examples/democoin/x/assoc/validator_set_test.go index 9fc6526f8..66f03c840 100644 --- a/examples/democoin/x/assoc/validator_set_test.go +++ b/docs/examples/democoin/x/assoc/validator_set_test.go @@ -10,7 +10,7 @@ import ( dbm "github.com/tendermint/tendermint/libs/db" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/mock" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/mock" "github.com/cosmos/cosmos-sdk/store" sdk "github.com/cosmos/cosmos-sdk/types" ) diff --git a/examples/democoin/x/cool/app_test.go b/docs/examples/democoin/x/cool/app_test.go similarity index 100% rename from examples/democoin/x/cool/app_test.go rename to docs/examples/democoin/x/cool/app_test.go diff --git a/examples/democoin/x/cool/client/cli/tx.go b/docs/examples/democoin/x/cool/client/cli/tx.go similarity index 96% rename from examples/democoin/x/cool/client/cli/tx.go rename to docs/examples/democoin/x/cool/client/cli/tx.go index c30743f03..a21685a24 100644 --- a/examples/democoin/x/cool/client/cli/tx.go +++ b/docs/examples/democoin/x/cool/client/cli/tx.go @@ -6,7 +6,7 @@ import ( "github.com/cosmos/cosmos-sdk/client/context" "github.com/cosmos/cosmos-sdk/client/utils" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/cool" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/cool" sdk "github.com/cosmos/cosmos-sdk/types" authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder" diff --git a/examples/democoin/x/cool/codec.go b/docs/examples/democoin/x/cool/codec.go similarity index 100% rename from examples/democoin/x/cool/codec.go rename to docs/examples/democoin/x/cool/codec.go diff --git a/examples/democoin/x/cool/errors.go b/docs/examples/democoin/x/cool/errors.go similarity index 100% rename from examples/democoin/x/cool/errors.go rename to docs/examples/democoin/x/cool/errors.go diff --git a/examples/democoin/x/cool/handler.go b/docs/examples/democoin/x/cool/handler.go similarity index 100% rename from examples/democoin/x/cool/handler.go rename to docs/examples/democoin/x/cool/handler.go diff --git a/examples/democoin/x/cool/keeper.go b/docs/examples/democoin/x/cool/keeper.go similarity index 100% rename from examples/democoin/x/cool/keeper.go rename to docs/examples/democoin/x/cool/keeper.go diff --git a/examples/democoin/x/cool/keeper_test.go b/docs/examples/democoin/x/cool/keeper_test.go similarity index 100% rename from examples/democoin/x/cool/keeper_test.go rename to docs/examples/democoin/x/cool/keeper_test.go diff --git a/examples/democoin/x/cool/types.go b/docs/examples/democoin/x/cool/types.go similarity index 100% rename from examples/democoin/x/cool/types.go rename to docs/examples/democoin/x/cool/types.go diff --git a/examples/democoin/x/oracle/README.md b/docs/examples/democoin/x/oracle/README.md similarity index 100% rename from examples/democoin/x/oracle/README.md rename to docs/examples/democoin/x/oracle/README.md diff --git a/examples/democoin/x/oracle/errors.go b/docs/examples/democoin/x/oracle/errors.go similarity index 100% rename from examples/democoin/x/oracle/errors.go rename to docs/examples/democoin/x/oracle/errors.go diff --git a/examples/democoin/x/oracle/handler.go b/docs/examples/democoin/x/oracle/handler.go similarity index 100% rename from examples/democoin/x/oracle/handler.go rename to docs/examples/democoin/x/oracle/handler.go diff --git a/examples/democoin/x/oracle/keeper.go b/docs/examples/democoin/x/oracle/keeper.go similarity index 100% rename from examples/democoin/x/oracle/keeper.go rename to docs/examples/democoin/x/oracle/keeper.go diff --git a/examples/democoin/x/oracle/keeper_keys.go b/docs/examples/democoin/x/oracle/keeper_keys.go similarity index 100% rename from examples/democoin/x/oracle/keeper_keys.go rename to docs/examples/democoin/x/oracle/keeper_keys.go diff --git a/examples/democoin/x/oracle/oracle_test.go b/docs/examples/democoin/x/oracle/oracle_test.go similarity index 98% rename from examples/democoin/x/oracle/oracle_test.go rename to docs/examples/democoin/x/oracle/oracle_test.go index 1284f75b6..b8b613e31 100644 --- a/examples/democoin/x/oracle/oracle_test.go +++ b/docs/examples/democoin/x/oracle/oracle_test.go @@ -10,7 +10,7 @@ import ( dbm "github.com/tendermint/tendermint/libs/db" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/mock" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/mock" "github.com/cosmos/cosmos-sdk/store" sdk "github.com/cosmos/cosmos-sdk/types" ) diff --git a/examples/democoin/x/oracle/types.go b/docs/examples/democoin/x/oracle/types.go similarity index 100% rename from examples/democoin/x/oracle/types.go rename to docs/examples/democoin/x/oracle/types.go diff --git a/examples/democoin/x/pow/app_test.go b/docs/examples/democoin/x/pow/app_test.go similarity index 100% rename from examples/democoin/x/pow/app_test.go rename to docs/examples/democoin/x/pow/app_test.go diff --git a/examples/democoin/x/pow/client/cli/tx.go b/docs/examples/democoin/x/pow/client/cli/tx.go similarity index 95% rename from examples/democoin/x/pow/client/cli/tx.go rename to docs/examples/democoin/x/pow/client/cli/tx.go index 22de7e098..548aa9910 100644 --- a/examples/democoin/x/pow/client/cli/tx.go +++ b/docs/examples/democoin/x/pow/client/cli/tx.go @@ -6,7 +6,7 @@ import ( "github.com/cosmos/cosmos-sdk/client/context" "github.com/cosmos/cosmos-sdk/client/utils" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/pow" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/pow" sdk "github.com/cosmos/cosmos-sdk/types" authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder" diff --git a/examples/democoin/x/pow/codec.go b/docs/examples/democoin/x/pow/codec.go similarity index 100% rename from examples/democoin/x/pow/codec.go rename to docs/examples/democoin/x/pow/codec.go diff --git a/examples/democoin/x/pow/errors.go b/docs/examples/democoin/x/pow/errors.go similarity index 100% rename from examples/democoin/x/pow/errors.go rename to docs/examples/democoin/x/pow/errors.go diff --git a/examples/democoin/x/pow/handler.go b/docs/examples/democoin/x/pow/handler.go similarity index 100% rename from examples/democoin/x/pow/handler.go rename to docs/examples/democoin/x/pow/handler.go diff --git a/examples/democoin/x/pow/handler_test.go b/docs/examples/democoin/x/pow/handler_test.go similarity index 100% rename from examples/democoin/x/pow/handler_test.go rename to docs/examples/democoin/x/pow/handler_test.go diff --git a/examples/democoin/x/pow/keeper.go b/docs/examples/democoin/x/pow/keeper.go similarity index 100% rename from examples/democoin/x/pow/keeper.go rename to docs/examples/democoin/x/pow/keeper.go diff --git a/examples/democoin/x/pow/keeper_test.go b/docs/examples/democoin/x/pow/keeper_test.go similarity index 100% rename from examples/democoin/x/pow/keeper_test.go rename to docs/examples/democoin/x/pow/keeper_test.go diff --git a/examples/democoin/x/pow/mine.go b/docs/examples/democoin/x/pow/mine.go similarity index 100% rename from examples/democoin/x/pow/mine.go rename to docs/examples/democoin/x/pow/mine.go diff --git a/examples/democoin/x/pow/types.go b/docs/examples/democoin/x/pow/types.go similarity index 100% rename from examples/democoin/x/pow/types.go rename to docs/examples/democoin/x/pow/types.go diff --git a/examples/democoin/x/pow/types_test.go b/docs/examples/democoin/x/pow/types_test.go similarity index 100% rename from examples/democoin/x/pow/types_test.go rename to docs/examples/democoin/x/pow/types_test.go diff --git a/examples/democoin/x/simplestake/client/cli/commands.go b/docs/examples/democoin/x/simplestake/client/cli/commands.go similarity index 97% rename from examples/democoin/x/simplestake/client/cli/commands.go rename to docs/examples/democoin/x/simplestake/client/cli/commands.go index b5c0730ef..3fe9c20c4 100644 --- a/examples/democoin/x/simplestake/client/cli/commands.go +++ b/docs/examples/democoin/x/simplestake/client/cli/commands.go @@ -7,7 +7,7 @@ import ( "github.com/cosmos/cosmos-sdk/client/context" "github.com/cosmos/cosmos-sdk/client/utils" "github.com/cosmos/cosmos-sdk/codec" - "github.com/cosmos/cosmos-sdk/examples/democoin/x/simplestake" + "github.com/cosmos/cosmos-sdk/docs/examples/democoin/x/simplestake" sdk "github.com/cosmos/cosmos-sdk/types" authcmd "github.com/cosmos/cosmos-sdk/x/auth/client/cli" authtxb "github.com/cosmos/cosmos-sdk/x/auth/client/txbuilder" diff --git a/examples/democoin/x/simplestake/codec.go b/docs/examples/democoin/x/simplestake/codec.go similarity index 100% rename from examples/democoin/x/simplestake/codec.go rename to docs/examples/democoin/x/simplestake/codec.go diff --git a/examples/democoin/x/simplestake/errors.go b/docs/examples/democoin/x/simplestake/errors.go similarity index 100% rename from examples/democoin/x/simplestake/errors.go rename to docs/examples/democoin/x/simplestake/errors.go diff --git a/examples/democoin/x/simplestake/handler.go b/docs/examples/democoin/x/simplestake/handler.go similarity index 100% rename from examples/democoin/x/simplestake/handler.go rename to docs/examples/democoin/x/simplestake/handler.go diff --git a/examples/democoin/x/simplestake/keeper.go b/docs/examples/democoin/x/simplestake/keeper.go similarity index 100% rename from examples/democoin/x/simplestake/keeper.go rename to docs/examples/democoin/x/simplestake/keeper.go diff --git a/examples/democoin/x/simplestake/keeper_test.go b/docs/examples/democoin/x/simplestake/keeper_test.go similarity index 100% rename from examples/democoin/x/simplestake/keeper_test.go rename to docs/examples/democoin/x/simplestake/keeper_test.go diff --git a/examples/democoin/x/simplestake/msgs.go b/docs/examples/democoin/x/simplestake/msgs.go similarity index 100% rename from examples/democoin/x/simplestake/msgs.go rename to docs/examples/democoin/x/simplestake/msgs.go diff --git a/examples/democoin/x/simplestake/msgs_test.go b/docs/examples/democoin/x/simplestake/msgs_test.go similarity index 100% rename from examples/democoin/x/simplestake/msgs_test.go rename to docs/examples/democoin/x/simplestake/msgs_test.go diff --git a/examples/democoin/x/simplestake/types.go b/docs/examples/democoin/x/simplestake/types.go similarity index 100% rename from examples/democoin/x/simplestake/types.go rename to docs/examples/democoin/x/simplestake/types.go diff --git a/examples/democoin/x/sketchy/handler.go b/docs/examples/democoin/x/sketchy/handler.go similarity index 100% rename from examples/democoin/x/sketchy/handler.go rename to docs/examples/democoin/x/sketchy/handler.go diff --git a/examples/kvstore/kvstore b/docs/examples/kvstore/kvstore similarity index 100% rename from examples/kvstore/kvstore rename to docs/examples/kvstore/kvstore diff --git a/examples/kvstore/main.go b/docs/examples/kvstore/main.go similarity index 100% rename from examples/kvstore/main.go rename to docs/examples/kvstore/main.go diff --git a/examples/kvstore/tx.go b/docs/examples/kvstore/tx.go similarity index 100% rename from examples/kvstore/tx.go rename to docs/examples/kvstore/tx.go diff --git a/docs/gaia/README.md b/docs/gaia/README.md new file mode 100644 index 000000000..9ecf3da23 --- /dev/null +++ b/docs/gaia/README.md @@ -0,0 +1,19 @@ +# Gaia Documentation + +Welcome to the `Gaia` docs. `Gaia` is the current name of the Cosmos SDK application for the Cosmos Hub. + +## Join the Cosmos Hub public testnet + +- [Install the `gaia` application](./installation.md) +- [Join the current testnet](./join-testnet.md) +- [Upgrade to a validator node](./validators/validator-setup.md) + +## Setup your own `gaia` testnet + +- [Setup your own `gaia` testnet](./networks.md) + +## Additional resources + +- [Intro to validators](./validators/overview.md) +- [Validator FAQ](./validators/validator-faq.md) +- [Validator security considerations](./validators/security.md) diff --git a/docs/sdk/clients.md b/docs/gaia/gaiacli.md similarity index 87% rename from docs/sdk/clients.md rename to docs/gaia/gaiacli.md index 4f0669183..5749e1ca7 100644 --- a/docs/sdk/clients.md +++ b/docs/gaia/gaiacli.md @@ -1,15 +1,7 @@ -# Clients - -::: tip Note -🚧 We are actively working on documentation for SDK clients. -::: +# Gaia client ## Gaia CLI -::: tip Note -🚧 We are actively working on improving documentation for Gaiacli and Gaiad. -::: - ::: tip Note If you receive this error message: @@ -208,13 +200,13 @@ On the upcoming mainnet, you can delegate `atom` to a validator. These [delegato You can query the list of all validators of a specific chain: ```bash -gaiacli query validators +gaiacli query stake validators ``` If you want to get the information of a single validator you can check it with: ```bash -gaiacli query validator +gaiacli query stake validator ``` #### Bond Tokens @@ -222,7 +214,7 @@ gaiacli query validator On the testnet, we delegate `steak` instead of `atom`. Here's how you can bond tokens to a testnet validator (*i.e.* delegate): ```bash -gaiacli tx delegate \ +gaiacli tx stake delegate \ --amount=10steak \ --validator= \ --from= \ @@ -248,7 +240,7 @@ Don't use more `steak` thank you have! You can always get more by using the [Fau Once submitted a delegation to a validator, you can see it's information by using the following command: ```bash -gaiacli query delegation \ +gaiacli query stake delegation \ --address-delegator= \ --validator= ``` @@ -256,7 +248,7 @@ gaiacli query delegation \ Or if you want to check all your current delegations with disctinct validators: ```bash -gaiacli query delegations +gaiacli query stake delegations ``` You can also get previous delegation(s) status by adding the `--height` flag. @@ -266,7 +258,7 @@ You can also get previous delegation(s) status by adding the `--height` flag. If for any reason the validator misbehaves, or you just want to unbond a certain amount of tokens, use this following command. You can unbond a specific `shares-amount` (eg:`12.1`\) or a `shares-fraction` (eg:`0.25`) with the corresponding flags. ```bash -gaiacli tx unbond begin \ +gaiacli tx stake unbond \ --validator= \ --shares-fraction=0.5 \ --from= \ @@ -280,7 +272,7 @@ The unbonding will be automatically completed when the unbonding period has pass Once you begin an unbonding-delegation, you can see it's information by using the following command: ```bash -gaiacli query unbonding-delegation \ +gaiacli query stake unbonding-delegation \ --address-delegator= \ --validator= \ ``` @@ -288,13 +280,13 @@ gaiacli query unbonding-delegation \ Or if you want to check all your current unbonding-delegations with disctinct validators: ```bash -gaiacli query unbonding-delegations +gaiacli query stake unbonding-delegations ``` Additionally, as you can get all the unbonding-delegations from a particular validator: ```bash - gaiacli query unbonding-delegations-from + gaiacli query stake unbonding-delegations-from ``` To get previous unbonding-delegation(s) status on past blocks, try adding the `--height` flag. @@ -304,7 +296,7 @@ To get previous unbonding-delegation(s) status on past blocks, try adding the `- A redelegation is a type delegation that allows you to bond illiquid tokens from one validator to another: ```bash -gaiacli tx redelegate begin \ +gaiacli tx stake redelegate \ --addr-validator-source= \ --addr-validator-dest= \ --shares-fraction=50 \ @@ -321,7 +313,7 @@ The redelegation will be automatically completed when the unbonding period has p Once you begin an redelegation, you can see it's information by using the following command: ```bash -gaiacli query redelegation \ +gaiacli query stake redelegation \ --address-delegator= \ --addr-validator-source= \ --addr-validator-dest= \ @@ -330,17 +322,55 @@ gaiacli query redelegation \ Or if you want to check all your current unbonding-delegations with disctinct validators: ```bash -gaiacli query redelegations +gaiacli query stake redelegations ``` Additionally, as you can get all the outgoing redelegations from a particular validator: ```bash - gaiacli query redelegations-from + gaiacli query stake redelegations-from ``` To get previous redelegation(s) status on past blocks, try adding the `--height` flag. +#### Query Parameters + +Parameters define high level settings for staking. You can get the current values by using: + +```bash +gaiacli query stake parameters +``` + +With the above command you will get the values for: + +- Unbonding time +- Maximum numbers of validators +- Coin denomination for staking + +All these values will be subject to updates though a `governance` process by `ParameterChange` proposals. + +#### Query Pool + +A staking `Pool` defines the dynamic parameters of the current state. You can query them with the following command: + +```bash +gaiacli query stake pool +``` + +With the `pool` command you will get the values for: + +- Loose and bonded tokens +- Token supply +- Current anual inflation and the block in which the last inflation was processed +- Last recorded bonded shares + +##### Query Delegations To Validator + +You can also query all of the delegations to a particular validator: +```bash + gaiacli query delegations-to +``` + ### Governance Governance is the process from which users in the Cosmos Hub can come to consensus on software upgrades, parameters of the mainnet or on custom text proposals. This is done through voting on proposals, which will be submitted by `Atom` holders on the mainnet. @@ -354,7 +384,7 @@ Some considerations about the voting process: - Voters can choose between options `Yes`, `No`, `NoWithVeto` and `Abstain` At the end of the voting period, a proposal is accepted if `(YesVotes/(YesVotes+NoVotes+NoWithVetoVotes))>1/2` and `(NoWithVetoVotes/(YesVotes+NoVotes+NoWithVetoVotes))<1/3`. It is rejected otherwise -For more information about the governance process and how it works, please check out the Governance module [specification](https://github.com/cosmos/cosmos-sdk/tree/develop/docs/spec/governance). +For more information about the governance process and how it works, please check out the Governance module [specification](./../spec/governance). #### Create a Governance proposal @@ -365,7 +395,7 @@ In order to create a governance proposal, you must submit an initial deposit alo - `type`: Type of proposal. Must be of value _Text_ (types _SoftwareUpgrade_ and _ParameterChange_ not supported yet). ```bash -gaiacli tx submit-proposal \ +gaiacli tx gov submit-proposal \ --title= \ --description=<description> \ --type=<Text/ParameterChange/SoftwareUpgrade> \ @@ -379,14 +409,13 @@ gaiacli tx submit-proposal \ Once created, you can now query information of the proposal: ```bash -gaiacli query proposal \ - --proposal-id=<proposal_id> +gaiacli query gov proposal --proposal-id=<proposal_id> ``` Or query all available proposals: ```bash -gaiacli query proposals +gaiacli query gov proposals ``` You can also query proposals filtered by `voter` or `depositer` by using the corresponding flags. @@ -396,7 +425,7 @@ You can also query proposals filtered by `voter` or `depositer` by using the cor In order for a proposal to be broadcasted to the network, the amount deposited must be above a `minDeposit` value (default: `10 steak`). If the proposal you previously created didn't meet this requirement, you can still increase the total amount deposited to activate it. Once the minimum deposit is reached, the proposal enters voting period: ```bash -gaiacli tx deposit \ +gaiacli tx gov deposit \ --proposal-id=<proposal_id> \ --deposit=<200steak> \ --from=<name> \ @@ -405,62 +434,54 @@ gaiacli tx deposit \ > _NOTE_: Proposals that don't meet this requirement will be deleted after `MaxDepositPeriod` is reached. +##### Query deposits + +Once a new proposal is created, you can query all the deposits submitted to it: + +```bash +gaiacli query gov deposits --proposal-id=<proposal_id> +``` + +You can also query a deposit submitted by a specific address: + +```bash +gaiacli query gov deposit \ + --proposal-id=<proposal_id> \ + --depositer=<account_cosmos> +``` + #### Vote on a proposal After a proposal's deposit reaches the `MinDeposit` value, the voting period opens. Bonded `Atom` holders can then cast vote on it: ```bash -gaiacli tx vote \ +gaiacli tx gov vote \ --proposal-id=<proposal_id> \ --option=<Yes/No/NoWithVeto/Abstain> \ --from=<name> \ --chain-id=<chain_id> ``` -##### Query vote +##### Query votes Check the vote with the option you just submitted: ```bash -gaiacli query vote \ +gaiacli query gov vote \ --proposal-id=<proposal_id> \ --voter=<account_cosmos> ``` -#### Query Parameters +You can also get all the previous votes submitted to the proposal with: -You can get the current parameters that define high level settings for staking: - -``` -gaiacli query parameters +```bash +gaiacli query gov votes --proposal-id=<proposal_id> ``` -With the above command you will get the values for: +#### Query proposal tally results -- Unbonding time -- Maximum numbers of validators -- Coin denomination for staking - -All this values can be updated though a `governance` process by submitting a parameter change `proposal`. - -#### Query Pool - -A staking `Pool` defines the dynamic parameters of the current state. You can query them with the following command: +To check the current tally of a given proposal you can use the `tally` command: +```bash +gaiacli query gov tally --proposal-id=<proposal_id> ``` -gaiacli query pool -``` - -With the `pool` command you will get the values for: - -- Loose and bonded tokens -- Token supply -- Current anual inflation and the block in which the last inflation was processed -- Last recorded bonded shares - - -## Gaia-Lite - -::: tip Note -🚧 We are actively working on documentation for Gaia-lite. -::: diff --git a/docs/getting-started/installation.md b/docs/gaia/installation.md similarity index 51% rename from docs/getting-started/installation.md rename to docs/gaia/installation.md index 66ccae4b8..c4b78b13c 100644 --- a/docs/getting-started/installation.md +++ b/docs/gaia/installation.md @@ -1,6 +1,6 @@ -# Install the SDK +# Install Gaia -This guide will explain how to install the [Cosmos SDK](/sdk/overview.md) onto your system. With the SDK installed on a server, you can participate in the latest testnet as either a [Full Node](./join-testnet.md#run-a-full-node) or a [Validator](/validators/validator-setup.md). +This guide will explain how to install the `gaiad` and `gaiacli` entrypoints onto your system. With these installed on a server, you can participate in the latest testnet as either a [Full Node](./join-testnet.md#run-a-full-node) or a [Validator](./validators/validator-setup.md). ## Install Go @@ -14,17 +14,13 @@ echo "export PATH=$PATH:$GOBIN" >> ~/.bash_profile ``` ::: tip -**Go 1.10+** is required for the Cosmos SDK. +**Go 1.11+** is required for the Cosmos SDK. ::: ## Install Cosmos SDK -Next, let's install the testnet's version of the Cosmos SDK. -You can find information about the latest testnet and the right -version of the Cosmos-SDK for it in the [testnets -repo](https://github.com/cosmos/testnets#testnet-status). -Here we'll use the `master` branch, which contains the latest stable release. -If necessary, make sure you `git checkout` the correct +Next, let's install the latest version of Gaia. Here we'll use the `master` branch, which contains the latest stable release. +If necessary, make sure you `git checkout` the correct [released version](https://github.com/cosmos/cosmos-sdk/releases). ```bash @@ -35,6 +31,8 @@ cd cosmos-sdk && git checkout master make get_tools && make get_vendor_deps && make install ``` +> *NOTE*: If you have issues at this step, please check that you have the latest stable version of GO installed. + That will install the `gaiad` and `gaiacli` binaries. Verify that everything is OK: ```bash @@ -44,4 +42,4 @@ $ gaiacli version ## Run a Full Node -With Cosmos SDK installed, you can run [a full node on the latest testnet](full-node.md). +With the binaries installed, you can run [a full node on the latest testnet](./join-testnet.md). diff --git a/docs/getting-started/join-testnet.md b/docs/gaia/join-testnet.md similarity index 71% rename from docs/getting-started/join-testnet.md rename to docs/gaia/join-testnet.md index ce070c4fe..560bbcf6e 100644 --- a/docs/getting-started/join-testnet.md +++ b/docs/gaia/join-testnet.md @@ -1,21 +1,23 @@ # Join the Testnet ::: tip Current Testnet -See the [testnet repo](https://github.com/cosmos/testnets) for +See the [testnet repo](https://github.com/cosmos/testnets) for information on the latest testnet, including the correct version of the Cosmos-SDK to use and details about the genesis file. ::: -Please ensure you have the [Cosmos SDK](/getting-started/installation.md) installed. If you ran a full node on a previous testnet, please skip to [Upgrading From Previous Testnet](#upgrading-from-previous-testnet). +**Please ensure you have the [gaia binaries](./installation.md) installed.** + +If you ran a full node on a previous testnet, please skip to [Upgrading From Previous Testnet](#upgrading-from-previous-testnet). ## Setting Up a New Node -These instructions are for setting up a brand new full node from scratch. +These instructions are for setting up a brand new full node from scratch. First, initialize the node and create the necessary config files: ```bash -gaiad init --skip-genesis --name <your_custom_name> +gaiad init ``` ::: warning Note @@ -75,10 +77,14 @@ git fetch --all && git checkout master make update_tools && make get_vendor_deps && make install ``` +::: tip +*NOTE*: If you have issues at this step, please check that you have the latest stable version of GO installed. +::: + Note we use `master` here since it contains the latest stable release. -See the [testnet repo](https://github.com/cosmos/testnets) -for details on which version is needed for which testnet, -and the [SDK release page](https://github.com/cosmos/cosmos-sdk/releases) +See the [testnet repo](https://github.com/cosmos/testnets) +for details on which version is needed for which testnet, +and the [SDK release page](https://github.com/cosmos/cosmos-sdk/releases) for details on each release. Your full node has been cleanly upgraded! @@ -94,20 +100,16 @@ mkdir -p $HOME/.gaiad/config curl https://raw.githubusercontent.com/cosmos/testnets/master/latest/genesis.json > $HOME/.gaiad/config/genesis.json ``` -Note we use the `latest` directory in the [testnets repo](https://github.com/cosmos/testnets) -which contains details for the latest testnet. If you are connecting to a different testnet, ensure you -get the right files. +Note we use the `latest` directory in the [testnets repo](https://github.com/cosmos/testnets) +which contains details for the latest testnet. If you are connecting to a different testnet, ensure you get the right files. ### Add Seed Nodes -Your node needs to know how to find peers. You'll need to add healthy seed nodes to `$HOME/.gaiad/config/config.toml`. Here are some seed nodes you can use: +Your node needs to know how to find peers. You'll need to add healthy seed nodes to `$HOME/.gaiad/config/config.toml`. The `testnets` repo contains links to the seed nodes for each testnet. If you are looking to join the running testnet please [check the repository for details](https://github.com/cosmos/testnets) on which nodes to use. -```toml -# Comma separated list of seed nodes to connect to -seeds = "718145d422a823fd2a4e1e36e91b92bb0c4ddf8e@gaia-testnet.coinculture.net:26656,5922bf29b48a18c2300b85cc53f424fce23927ab@67.207.73.206:26656,7c8b8fd03577cd4817f5be1f03d506f879df98d8@gaia-7000-seed1.interblock.io:26656,a28737ff02391a6e00a1d3b79befd57e68e8264c@gaia-7000-seed2.interblock.io:26656,987ffd26640cd03d08ed7e53b24dfaa7956e612d@gaia-7000-seed3.interblock.io:26656" -``` +If those seeds aren't working, you can find more seeds and persistent peers on the [Cosmos Explorer](https://explorer.cosmos.network/nodes). Open the the `Full Nodes` pane and select nodes that do not have private (`10.x.x.x`) or [local IP addresses](https://en.wikipedia.org/wiki/Private_network). The `Persistent Peer` field contains the connection string. For best results use 4-6. -If those seeds aren't working, you can find more seeds and persistent peers on the [Cosmos Explorer](https://explorecosmos.network/nodes). Open the the `Full Nodes` pane and select nodes that do not have private (`10.x.x.x`) or [local IP addresses](https://en.wikipedia.org/wiki/Private_network). The `Persistent Peer` field contains the connection string. For best results use 4-6. +You can also ask for peers on the [Validators Riot Room](https://riot.im/app/#/room/#cosmos-validators:matrix.org) For more information on seeds and peers, you can [read this](https://github.com/tendermint/tendermint/blob/develop/docs/using-tendermint.md#peers). @@ -130,4 +132,4 @@ View the status of the network with the [Cosmos Explorer](https://explorecosmos. ## Upgrade to Validator Node -You now have an active full node. What's the next step? You can upgrade your full node to become a Cosmos Validator. The top 100 validators have the ability to propose new blocks to the Cosmos Hub. Continue onto [the Validator Setup](../validators/validator-setup.md). +You now have an active full node. What's the next step? You can upgrade your full node to become a Cosmos Validator. The top 100 validators have the ability to propose new blocks to the Cosmos Hub. Continue onto [the Validator Setup](./validators/validator-setup.md). diff --git a/docs/clients/keys.md b/docs/gaia/keys.md similarity index 100% rename from docs/clients/keys.md rename to docs/gaia/keys.md diff --git a/docs/clients/ledger.md b/docs/gaia/ledger.md similarity index 100% rename from docs/clients/ledger.md rename to docs/gaia/ledger.md diff --git a/docs/getting-started/networks.md b/docs/gaia/networks.md similarity index 89% rename from docs/getting-started/networks.md rename to docs/gaia/networks.md index 67f0dab5a..3bc08c94b 100644 --- a/docs/getting-started/networks.md +++ b/docs/gaia/networks.md @@ -56,8 +56,9 @@ make build-linux localnet-stop localnet-start ### Configuration -The `make localnet-start` creates files for a 4-node testnet in `./build` by calling the `gaiad testnet` command. -This outputs a handful of files in the `./build` directory: +The `make localnet-start` creates files for a 4-node testnet in `./build` by +calling the `gaiad testnet` command. This outputs a handful of files in the +`./build` directory: ```tree -L 2 build/ build/ @@ -103,12 +104,29 @@ Each `./build/nodeN` directory is mounted to the `/gaiad` directory in each cont ### Logging -Logs are saved under each `./build/nodeN/gaiad/gaia.log`. Watch them stream in with, for example: +Logs are saved under each `./build/nodeN/gaiad/gaia.log`. You can also watch logs +directly via Docker, for example: ``` -tail -f build/node0/gaiad/gaia.log +docker logs -f gaiadnode0 ``` +### Keys & Accounts + +To interact with `gaiacli` and start querying state or creating txs, you use the +`gaiacli` directory of any given node as your `home`, for example: + +```shell +gaiacli keys list --home ./build/node0/gaiacli +``` + +Now that accounts exists, you may create new accounts and send those accounts +funds! + +::: tip +**Note**: Each node's seed is located at `./build/nodeN/gaiacli/key_seed.json`. +::: + ### Special binaries If you have multiple binaries with different names, you can specify which one to run with the BINARY environment variable. The path of the binary is relative to the attached volume. For example: diff --git a/docs/validators/overview.md b/docs/gaia/validators/overview.md similarity index 100% rename from docs/validators/overview.md rename to docs/gaia/validators/overview.md diff --git a/docs/validators/security.md b/docs/gaia/validators/security.md similarity index 100% rename from docs/validators/security.md rename to docs/gaia/validators/security.md diff --git a/docs/validators/validator-faq.md b/docs/gaia/validators/validator-faq.md similarity index 100% rename from docs/validators/validator-faq.md rename to docs/gaia/validators/validator-faq.md diff --git a/docs/validators/validator-setup.md b/docs/gaia/validators/validator-setup.md similarity index 85% rename from docs/validators/validator-setup.md rename to docs/gaia/validators/validator-setup.md index d07697136..61872bcf9 100644 --- a/docs/validators/validator-setup.md +++ b/docs/gaia/validators/validator-setup.md @@ -1,14 +1,14 @@ # Validator Setup -::: warning Current Testnet -The current testnet is `gaia-8000`. +::: tip +Information on how to join the current testnet (`genesis.json` file and seeds) is held [in our `testnet` repo](https://github.com/cosmos/testnets/tree/master/latest). Please check there if you are looking to join our latest testnet. ::: Before setting up your validator node, make sure you've already gone through the [Full Node Setup](/docs/getting-started/full-node.md) guide. ## Running a Validator Node -[Validators](/validators/overview.md) are responsible for committing new blocks to the blockchain through voting. A validator's stake is slashed if they become unavailable, double sign a transaction, or don't cast their votes. Please read about [Sentry Node Architecture](/validators/validator-faq.md#how-can-validators-protect-themselves-from-denial-of-service-attacks) to protect your node from DDOS attacks and to ensure high-availability. +[Validators](/validators/overview.md) are responsible for committing new blocks to the blockchain through voting. A validator's stake is slashed if they become unavailable or sign blocks at the same height. Please read about [Sentry Node Architecture](/validators/validator-faq.md#how-can-validators-protect-themselves-from-denial-of-service-attacks) to protect your node from DDOS attacks and to ensure high-availability. ::: danger Warning If you want to become a validator for the Hub's `mainnet`, you should [research security](/validators/security.md). @@ -22,14 +22,14 @@ Your `cosmosvalconspub` can be used to create a new validator by staking tokens. gaiad tendermint show-validator ``` -Next, craft your `gaiacli tx create-validator` command: +Next, craft your `gaiacli tx stake create-validator` command: ::: warning Note Don't use more `steak` thank you have! You can always get more by using the [Faucet](https://faucetcosmos.network/)! ::: ```bash -gaiacli tx create-validator \ +gaiacli tx stake create-validator \ --amount=5steak \ --pubkey=$(gaiad tendermint show-validator) \ --moniker="choose a moniker" \ @@ -44,6 +44,10 @@ __Note__: When specifying commission parameters, the `commission-max-change-rate is used to measure % _point_ change over the `commission-rate`. E.g. 1% to 2% is a 100% rate increase, but only 1 percentage point. +::: tip +Use `gaiacli tx stake create-validator -h` to get a list of all the available flags. +::: + ### Edit Validator Description You can edit your validator's public description. This info is to identify your validator, and will be relied on by delegators to decide which validators to stake to. Make sure to provide input for every flag below, otherwise the field will default to empty (`--moniker` defaults to the machine name). @@ -51,7 +55,7 @@ You can edit your validator's public description. This info is to identify your The `--identity` can be used as to verify identity with systems like Keybase or UPort. When using with Keybase `--identity` should be populated with a 16-digit string that is generated with a [keybase.io](https://keybase.io) account. It's a cryptographically secure method of verifying your identity across multiple online networks. The Keybase API allows us to retrieve your Keybase avatar. This is how you can add a logo to your validator profile. ```bash -gaiacli tx edit-validator +gaiacli tx stake edit-validator --moniker="choose a moniker" \ --website="https://cosmos.network" \ --identity=6A0D65E29A4CBC8E \ @@ -73,7 +77,7 @@ __Note__: The `commission-rate` value must adhere to the following invariants: View the validator's information with this command: ```bash -gaiacli query validator <account_cosmos> +gaiacli query stake validator <account_cosmos> ``` ### Track Validator Signing Information @@ -81,7 +85,7 @@ gaiacli query validator <account_cosmos> In order to keep track of a validator's signatures in the past you can do so by using the `signing-info` command: ```bash -gaiacli query signing-info <validator-pubkey>\ +gaiacli query slashing signing-info <validator-pubkey>\ --chain-id=<chain_id> ``` @@ -90,7 +94,7 @@ gaiacli query signing-info <validator-pubkey>\ When a validator is "jailed" for downtime, you must submit an `Unjail` transaction from the operator account in order to be able to get block proposer rewards again (depends on the zone fee distribution). ```bash -gaiacli tx unjail \ +gaiacli tx slashing unjail \ --from=<key_name> \ --chain-id=<chain_id> ``` @@ -124,7 +128,7 @@ gaiad start Wait for your full node to catch up to the latest block. Next, run the following command. Note that `<cosmos>` is the address of your validator account, and `<name>` is the name of the validator account. You can find this info by running `gaiacli keys list`. ```bash -gaiacli tx unjail <cosmos> --chain-id=<chain_id> --from=<from> +gaiacli tx slashing unjail <cosmos> --chain-id=<chain_id> --from=<from> ``` ::: danger Warning diff --git a/docs/getting-started/voyager.md b/docs/getting-started/voyager.md deleted file mode 100644 index 7fa17f24c..000000000 --- a/docs/getting-started/voyager.md +++ /dev/null @@ -1,9 +0,0 @@ -# Download Voyager - -Voyager is the official desktop app for the Cosmos Network. It provides an intuitive interface for managing accounts, creating transactions, delegation, and governance. - -Download the [latest Voyager release](https://github.com/cosmos/voyager/releases). - -### Delving Deeper - -If you're familar with the CLI, you should consider running a full node or validator for the latest Cosmos testnet. [Learn more](/getting-started/installation.md). diff --git a/docs/graphics/cosmos-sdk-image.png b/docs/graphics/cosmos-sdk-image.png deleted file mode 100644 index ab9a4abc6..000000000 Binary files a/docs/graphics/cosmos-sdk-image.png and /dev/null differ diff --git a/docs/intro/README.md b/docs/intro/README.md new file mode 100644 index 000000000..83dcc31c5 --- /dev/null +++ b/docs/intro/README.md @@ -0,0 +1,43 @@ +# SDK Intro + +The [Cosmos-SDK](https://github.com/cosmos/cosmos-sdk) is a framework for building multi-asset Proof-of-Stake (PoS) blockchains, like the Cosmos Hub, as well as Proof-Of-Authority (PoA) blockchains. + +The goal of the Cosmos SDK is to allow developers to easily create custom blockchains from scratch that can natively interoperate with other blockchains. We envision the SDK as the npm-like framework to build secure blockchain applications on top of [Tendermint](https://github.com/tendermint/tendermint). + +It is based on two major principles: + +- **Composability:** Anyone can create a module for the Cosmos-SDK, and integrating the already-built modules is as simple as importing them into your blockchain application. + +- **Capabilities:** The SDK is inspired by capabilities-based security, and informed by years of wrestling with blockchain state-machines. Most developers will need to access other 3rd party modules when building their own modules. Given that the Cosmos-SDK is an open framework, some of the modules may be malicious, which means there is a need for security principles to reason about inter-module interactions. These principles are based on object-cababilities. In practice, this means that instead of having each module keep an access control list for other modules, each module implements special objects called keepers that can be passed to other modules to grant a pre-defined set of capabilities. For example, if an instance of module A's keepers is passed to module B, the latter will be able to call a restricted set of module A's functions. The capabilities of each keeper are defined by the module's developer, and it's the developer's job to understand and audit the safety of foreign code from 3rd party modules based on the capabilities they are passing into each third party module. For a deeper look at capabilities, jump to [this section](./capabilities.md). + +## Learn more about the SDK + +- [SDK application architecture](./design.md) +- [SDK security paradigm: ocap](./ocap.md) + +## Creating a new SDK project + +To create a new project, you can either: + +- Fork [this repo](https://github.com/cosmos/sdk-application-tutorial/). Do not forget to remove the `nameservice` module from the various files if you don't need it. +- Copy the `docs/examples/basecoin` directory. +- Use community tools! More info to come. + +## SDK Directory Structure + +The SDK is laid out in the following directories: + +- `baseapp`: Defines the template for a basic [ABCI](https://github.com/tendermint/tendermint/tree/master/abci) application so that your Cosmos-SDK application can communicate with the underlying Tendermint node. +- `client`: CLI and REST server tooling for interacting with SDK application. +- `examples`: Examples of how to build working standalone applications. +- `server`: The full node server for running an SDK application on top of + Tendermint. +- `store`: The database of the SDK - a Merkle multistore supporting multiple types of underling Merkle key-value stores. +- `types`: Common types in SDK applications. +- `x`: Extensions to the core, where all messages and handlers are defined. + +## Languages + +The Cosmos-SDK is currently written in [Golang](https://golang.org/), though the +framework could be implemented similarly in other languages. +Contact us for information about funding an implementation in another language. diff --git a/docs/intro/ocap.md b/docs/intro/ocap.md new file mode 100644 index 000000000..7d57d0c34 --- /dev/null +++ b/docs/intro/ocap.md @@ -0,0 +1,102 @@ +# Object-Capability Model + +## Intro + +When thinking about security, it is good to start with a specific threat +model. Our threat model is the following: + +> We assume that a thriving ecosystem of Cosmos-SDK modules that are easy to compose into a blockchain application will contain faulty or malicious modules. + +The Cosmos SDK is designed to address this threat by being the +foundation of an object capability system. + +> The structural properties of object capability systems favor +> modularity in code design and ensure reliable encapsulation in +> code implementation. +> +> These structural properties facilitate the analysis of some +> security properties of an object-capability program or operating +> system. Some of these — in particular, information flow properties +> — can be analyzed at the level of object references and +> connectivity, independent of any knowledge or analysis of the code +> that determines the behavior of the objects. +> +> As a consequence, these security properties can be established +> and maintained in the presence of new objects that contain unknown +> and possibly malicious code. +> +> These structural properties stem from the two rules governing +> access to existing objects: +> +> 1. An object A can send a message to B only if object A holds a +> reference to B. +> 2. An object A can obtain a reference to C only +> if object A receives a message containing a reference to C. As a +> consequence of these two rules, an object can obtain a reference +> to another object only through a preexisting chain of references. +> In short, "Only connectivity begets connectivity." + +For an introduction to object-capabilities, see [this article](http://habitatchronicles.com/2017/05/what-are-capabilities/). + +Strictly speaking, Golang does not implement object capabilities +completely, because of several issues: + +- pervasive ability to import primitive modules (e.g. "unsafe", "os") +- pervasive ability to [override module vars](https://github.com/golang/go/issues/23161) +- data-race vulnerability where 2+ goroutines can create illegal interface values + +The first is easy to catch by auditing imports and using a proper +dependency version control system like Dep. The second and third are +unfortunate but it can be audited with some cost. + +Perhaps [Go2 will implement the object capability +model](https://github.com/golang/go/issues/23157). + +## Ocaps in practice + +The idea is to only reveal what is necessary to get the work done. + +For example, the following code snippet violates the object capabilities +principle: + +```go +type AppAccount struct {...} +var account := &AppAccount{ + Address: pub.Address(), + Coins: sdk.Coins{{"ATM", 100}}, +} +var sumValue := externalModule.ComputeSumValue(account) +``` + +The method `ComputeSumValue` implies a pure function, yet the implied +capability of accepting a pointer value is the capability to modify that +value. The preferred method signature should take a copy instead. + +```go +var sumValue := externalModule.ComputeSumValue(*account) +``` + +In the Cosmos SDK, you can see the application of this principle in the +[basecoin examples folder](../examples/basecoin). + +```go +// File: cosmos-sdk/docs/examples/basecoin/app/init_handlers.go +package app + +import ( + "github.com/cosmos/cosmos-sdk/x/bank" + "github.com/cosmos/cosmos-sdk/x/sketchy" +) + +func (app *BasecoinApp) initRouterHandlers() { + + // All handlers must be added here. + // The order matters. + app.router.AddRoute("bank", bank.NewHandler(app.accountKeeper)) + app.router.AddRoute("sketchy", sketchy.NewHandler()) +} +``` + +In the Basecoin example, the sketchy handler isn't provided an account +mapper, which does provide the bank handler with the capability (in +conjunction with the context of a transaction run). diff --git a/docs/intro/sdk-app-architecture.md b/docs/intro/sdk-app-architecture.md new file mode 100644 index 000000000..4b302bd33 --- /dev/null +++ b/docs/intro/sdk-app-architecture.md @@ -0,0 +1,62 @@ +# SDK Application Architecture + +## Parts of a SDK blockchain + +A blockchain application is just a [replicated deterministic state machine](https://en.wikipedia.org/wiki/State_machine_replication). As a developer, you just have to define the state machine (i.e. what the state, a starting state and messages that trigger state transitions), and [*Tendermint*](https://tendermint.com/docs/introduction/introduction.html) will handle replication over the network for you. + +>Tendermint is an application-agnostic engine that is responsible for handling the *networking* and *consensus* layers of your blockchain. In practice, this means that Tendermint is reponsible for propagating and ordering transaction bytes. Tendermint Core relies on an eponymous Byzantine-Fault-Tolerant (BFT) algorithm to reach consensus on the order of transactions. For more on Tendermint, click [here](https://tendermint.com/docs/introduction/introduction.html). + +Tendermint passes transactions from the network to the application through an interface called the [ABCI](https://github.com/tendermint/tendermint/tree/master/abci. If you look at the architecture of the blockchain node you are building, it looks like the following: + +``` ++---------------------+ +| | +| Application | +| | ++--------+---+--------+ + ^ | + | | ABCI + | v ++--------+---+--------+ +| | +| | +| Tendermint | +| | +| | ++---------------------+ +``` + +Fortunately, you do not have to implement the ABCI interface. The Cosmos SDK provides a boilerplate implementation of it in the form of [baseapp](#baseapp). + +## BaseApp + +Implements an ABCI App using a [MultiStore](../reference/store) for persistence and a Router to handle transactions. +The goal is to provide a secure interface between the store and the extensible state machine while defining as little about that state machine as possible (staying true to the ABCI). + +For more on `baseapp`, please go to the [baseapp reference](../reference/baseapp.md). + +## Modules + +The power of the SDK lies in its modularity. SDK blockchains are built out of customizable and interoperable modules. These modules are contained in the `x/` folder. + +In addition to the already existing modules in `x/`, that anyone can use in their app, the SDK lets you build your own custom modules. In other words, building a SDK blockchain consists in importing some modules and building others (the ones you need that do not exist yet!). + +Some core modules include: + +- `x/auth`: Used to manage accounts and signatures. +- `x/bank`: Used to enable tokens and token transfers. +- `x/staking` + `x/slashing`: Used to build Proof-Of-Stake blockchains. + +## Basecoin + +Basecoin is the first complete application in the stack. Complete applications require extensions to the core modules of the SDK to actually implement handler functionality. + +Basecoin implements a `BaseApp` state machine using the `x/auth` and `x/bank` modules, which define how transaction signers are authenticated and how coins are transferred. It should also use `x/ibc` and probably a simple staking extension. + +Basecoin and the native `x/` extensions use go-amino for all serialization needs, including for transactions and accounts. + +## SDK tutorial + +If you want to learn more about how to build an SDK application and get a deepeer understanding of the concepts presented above in the process, please check out the [SDK Application Tutorial](https://github.com/cosmos/sdk-application-tutorial). + + diff --git a/docs/introduction/cosmos-hub.md b/docs/introduction/cosmos-hub.md deleted file mode 100644 index 0e40b7b4c..000000000 --- a/docs/introduction/cosmos-hub.md +++ /dev/null @@ -1,32 +0,0 @@ -# The Cosmos Hub -The first blockchain in the Cosmos network is the Cosmos hub. The Cosmos hub connects to zones via the novel [IBC](/specs/ibc.md) (inter-blockchain communication) protocol and keeps a record of the total number of tokens in each zone. Because all inter-zone transfers go through the Cosmos Hub, you can send tokens from one zone to another, quickly and securely, without the need for a liquid exchange or trusted third party between zones. - -The Cosmos Hub can connect to many different kinds of zones - public or private - as long as each zone speaks IBC. Tendermint-based Zones are natively compatible with IBC, but any fast-finality consensus algorithm can be used as a replacement. Cosmos can thus support a wide variety of currencies and scripting languages like those found in Bitcoin, Ethereum, ZeroCash, CryptoNote, and more. Atom's are the native token of the Cosmos Hub. It is a license for holders to stake and participate in governance. - -## Proof-of-Stake -Blockchain networks are secured by a set of validators, who are responsible for committing new blocks in the blockchain. In Proof-Of-Work systems such as Bitcoin, validators are called miners, and the probability of a given miner to produce the next block is proportional to its computing power. In contrast, the Cosmos Hub is a public Proof-of-Stake blockchain. Proof-of-Stake is a category of consensus algorithm that relies on validators’ economic stake in the network. - -## Atoms -In the case of the Cosmos Hub, the frequency at which a validator is selected to produce the next block is proportional to the number of Atoms locked up (i.e. bonded, or staked). - -These Atoms can be locked up by validators themselves, or delegated to them by Atom holders that do not want or cannot run validator operations, called delegators. The sum of a validator’s self-bonded and delegated Atoms is called its stake. The Atom is the only staking token of the Cosmos Hub. In return for locking up their Atoms, delegators earn block provisions (in Atoms), block rewards (in Photons) and transaction fees (in whitelisted fee tokens). When a bonded Atom holder wants to retrieve its deposit, it must wait for a 3 week unbonding period. - -## Photons -Atoms are designed to be bonded on the Hub. This means that they are not ideal to pay fees or to move on other Zones of the Cosmos Ecosystem. This is why Photons will be introduced. Photon is a fee token with much greater liquidity and velocity than Atom. It is the second whitelisted fee token on the Hub after Atom and can move on all the Zones that are connected to the Hub. - -## Hard spoon -A hard spoon occurs when a new cryptocurrency is minted by replicating the account balances of an existing cryptocurrency. In our case, we are hard spooning Ethereum by taking the account balances of existing Ethereum holders and mirroring those values. This means that ETH holders will have their coins replicated in this EVM zone and will be redeemable as fee tokens–Photons–within Ethermint. - -After launch, Atom holders will be able to vote on the hard spoon, specifically: - -- Whether the hard spoon should happen or not -- When the snapshot will occur -- How Photons are distributed (what goes to Ethereum holders, what goes to Atom holders and Photon inflation) - -## Validators -Validators of the Cosmos Hub are responsible for creating new blocks of transactions that are added to the blockchain. Running a validator is non-trivial. It requires technical knowledge and hardware investment. Additionally, due to the way that Tendermint—the underlying consensus engine on which the Cosmos Hub is built—works, the number of validators must be limited. Initially, this limit is fixed to 100. This means that only the top 100 addresses with the most stake that declared their intention to become validator will be validators. As a result, most Atom holders will not be validators. Instead, they will become delegators, thereby participating in deciding who among the validator candidates actually become validators. - -If you are interested in becoming a validator: learn more about validators [here](/validators/overview.md). - -## Delegators -People that cannot, or do not want to run validator operations, can still participate in the staking process as delegators. Indeed, validators are not chosen based on their own stake but based on their total stake, which is the sum of their own stake and of the stake that is delegated to them. If you are interested in staking your atoms to a Validator to earn revenue, or just want to learn more about delegators, read the [Delegator FAQ](/resources/delegator-faq.md). diff --git a/docs/introduction/tendermint-cosmos.md b/docs/introduction/tendermint-cosmos.md deleted file mode 100644 index f7aeb7746..000000000 --- a/docs/introduction/tendermint-cosmos.md +++ /dev/null @@ -1,27 +0,0 @@ -# Tendermint and Cosmos - -Blockchains can be divided into three conceptual layers: - -- **Networking:** Responsible for propagating transactions. -- **Consensus:** Enables validator nodes to agree on the next set of transactions to process (i.e. add blocks of transactions to the blockchain). -- **Application:** Responsible for updating the state given a set of transactions, i.e. processing transactions. - -The *networking* layer makes sure that each node receives transactions. The *consensus* layer makes sure that each node agrees on the same transactions to modify their local state. As for the *application* layer, it processes transactions. Given a transaction and a state, the application will return a new state. In Bitcoin for example, the application state is a ledger or list of balances for each account (in reality, it's a list of UTXO, short for Unspent Transaction Output, but let's call them balances for the sake of simplicity), and the transactions modify the application's state by changing these list of balances. In the case of Ethereum, the application is a virtual machine. Each transaction goes through this virtual machine and modifies the application state according to the the smart contract that is called within it. - -Before Tendermint, building a blockchain required building all three layers from the ground up. It was such a tedious task that most developers preferred to fork or replicate the Bitcoin codebase, but were constrainted by the limitations of Bitcoin's protocol. The Ethereum Virtual Machine (EVM) was designed to solve this problem and simplify decentralized application development by allowing customizable logic to be executed through smart contracts. But it did not resolve the limitations (interoperability, scalability and customization) of blockchains themselves. Go-Ethereum remains a very monolithic tech stack that is difficult to hard-fork much like Bitcoin's codebase. - -Tendermint was designed to address these issues and provide developers with an laternative. The goal of Tendermint is to provide the *networking* and *consensus* layers of a blockchain as a generic engine to power any application developers want to build. With Tendermint, developers only have to focus on the *application* layer, thereby saving hundreds of hours of work and costly development set-ups. For reference, Tendermint also designates the name of the byzantine fault tolerant consensus algorithm used within the Tendermint Core engine. - -Tendermint connects the blockchain engine, Tendermint Core (*networking* and *consensus* layers), to the *application* layer via a socket protocol called the [ABCI](https://github.com/tendermint/abci), short for Application-BlockChain Interface. Developers only have to implement a few messages to build an ABCI-enabled application that runs on top of the Tendermint Core engine. ABCI is language agnostic, meaning that developers can build the *application* part of their blockchain in any programming language. Building on top of the Tendermint Core engine also provides the following benefits: - -- **Public or private blockchain capable.** Developers can deploy any blockchain application, permissioned (private) and permissionless (public), on top of Tendermint Core. -- **Performance.** Tendermint Core is a state-of-the-art blockchain consensus engine able to handle large number of transactions in short timespan. A block time on Tendermint Core can be as low as one second and can process thousands of transactions in that time period. -- **Instant finality.** A property of the Tendermint consensus algorithm is instant finality, meaning that forks are never created, as long as less than a third of the validators are malicious (byzantine). Users can be sure their transactions are finalized as soon as a block is created. -- **Security.** Tendermint Core's consensus is not only fault tolerant, it’s optimally Byzantine fault-tolerant (BFT), with accountability. If the blockchain forks, there is a way to determine liability. -- **Light-client support**. Tendermint provides built-in light-clients. - -But most importantly, Tendermint is natively compatible with the [Inter-Blockchain Communication Protocol](https://github.com/cosmos/cosmos-sdk/tree/develop/docs/spec/ibc) (IBC). This means that any Tendermint-based blockchain, whether public or private, can be natively connected to the Cosmos ecosystem and securely exchange tokens with other blockchains in the ecosystem. Note that benefiting from interoperability via IBC and Cosmos preserves the sovereignty of your Tendermint chain. Non-Tendermint chains can also be connected to Cosmos via IBC adapters or Peg-Zones, but this is out of scope for this document. - -For a more detailed overview of the Cosmos ecosystem, you can read [this article](https://blog.cosmos.network/understanding-the-value-proposition-of-cosmos-ecaef63350d). - -For more on Tendermint, go [here](tendermint.md) \ No newline at end of file diff --git a/docs/introduction/tendermint.md b/docs/introduction/tendermint.md deleted file mode 100644 index 4192a03ea..000000000 --- a/docs/introduction/tendermint.md +++ /dev/null @@ -1,47 +0,0 @@ -# Tendermint - -Tendermint is software for securely and consistently replicating an application on many machines. By securely, we mean that Tendermint works even if up to 1/3 of machines fail in arbitrary ways. By consistently, we mean that every non-faulty machine sees the same transaction log and computes the same state. Secure and consistent replication is a fundamental problem in distributed systems; it plays a critical role in the fault tolerance of a broad range of applications, from currencies, to elections, to infrastructure orchestration, and beyond. - -Tendermint is designed to be easy-to-use, simple-to-understand, highly performant, and useful for a wide variety of distributed applications. - -## Byzantine Fault Tolerance - -The ability to tolerate machines failing in arbitrary ways, including becoming malicious, is known as Byzantine fault tolerance (BFT). The theory of BFT is decades old, but software implementations have only became popular recently, due largely to the success of “blockchain technology” like Bitcoin and Ethereum. Blockchain technology is just a re-formalization of BFT in a more modern setting, with emphasis on peer-to-peer networking and cryptographic authentication. The name derives from the way transactions are batched in blocks, where each block contains a cryptographic hash of the previous one, forming a chain. In practice, the blockchain data structure actually optimizes BFT design. - -## Application Blockchain Interface - -Tendermint consists of two chief technical components: a blockchain consensus engine and a generic application interface. The consensus engine, called Tendermint Core, ensures that the same transactions are recorded on every machine in the same order. The application interface, called the Application Blockchain Interface (ABCI), enables the transactions to be processed in any programming language. Unlike other blockchain and consensus solutions developers can use Tendermint for BFT state machine replication in any programming language or development environment. Visit the [Tendermint docs](https://tendermint.com/docs/introduction/introduction.html#abci-overview) for a deep dive into the ABCI. - -## Understanding the roles of the different layers - -It is important to have a good understanding of the respective responsibilities of both the *Application* and the *Consensus Engine*. - -Responsibilities of the *Consensus Engine*: -- Propagate transactions -- Agree on the order of valid transactions - -Reponsibilities of the *Application*: -- Generate Transactions -- Check if transactions are valid -- Process Transactions (includes state changes) - -It is worth underlining that the *Consensus Engine* has knowledge of a given validator set for each block, but that it is the responsiblity of the *Application* to trigger validator set changes. This is the reason why it is possible to build both **public and private chains** with the Cosmos-SDK and Tendermint. A chain will be public or private depending on the rules, defined at application level, that governs a validator's set changes. - -The ABCI establishes the connection between the *Consensus Engine* and the *Application*. Essentially, it boils down to two messages: - -- `CheckTx`: Ask the application if the transaction is valid. When a validator's node receives a transaction, it will run `CheckTx` on it. If the transaction is valid, it is added to the mempool. -- `DeliverTx`: Ask the application to process the transaction and update the state. - -Let us give a high-level overview of how the *Consensus Engine* and the *Application* interract with each other. - -- At all times, when the consensus engine (Tendermint Core) of a validator node receives a transaction, it passes it to the application via `CheckTx` to check its validity. If it is valid, the transaction is added to the mempool. -- Let us say we are at block N. There is a validator set V. A proposer of the next block is selected from V by the *Consensus Engine*. The proposer gathers valid transaction from its mempool to form a new block. Then, the block is gossiped to other validators to be signed/commited. The block becomes block N+1 once 2/3+ of V have signed a *precommit* on it (For a more detailed explanation of the consensus algorithm, click [here](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm)). -- When block N+1 is signed by 2/3+ of V, it is gossipped to full-nodes. When full-nodes receive the block, they confirm its validity. A block is valid if it it holds valid signatures from more than 2/3 of V and if all the transactions in the block are valid. To check the validity of transactions, the *Consensus Engine* transfers them to the application via `DeliverTx`. After each transaction, `DeliverTx` returns a new state if the transaction was valid. At the end of the block, a final state is committed. Of course, this means that the order of transaction within a block matters. - -## Application frameworks - -Even if Tendermint makes it easy for developers to build their own blockchain by enabling them to focus on the *Application* layer of their blockchain, building an *Application* can be a challenging task in itself. This is why *Application Frameworks* exist. They provide developers with a secure and features-heavy environment to develop Tendermint-based applications. Here are some examples of *Application Frameworks* : - -- The [Cosmos SDK](/sdk/overview.md) is an ABCI framework written in Go. -- [Lotion JS](/lotion/overview.md) is an ABCI framework written in JavaScript. - diff --git a/docs/introduction/what-is-cosmos.md b/docs/introduction/what-is-cosmos.md deleted file mode 100644 index f9470d7c9..000000000 --- a/docs/introduction/what-is-cosmos.md +++ /dev/null @@ -1,7 +0,0 @@ -# What is Cosmos? - -Cosmos is a decentralized network of independent parallel blockchains, each powered by classical BFT consensus algorithms like [Tendermint](). - -The first blockchain in the Cosmos Network is the [Cosmos Hub](), whose native token is the Atom. Cosmos is a permission-less network, meaning that anybody can build a blockchain on it. - -Cosmos can interoperate with multiple other applications and cryptocurrencies. By creating a new zone, you can plug any blockchain system into the Cosmos hub and pass tokens back and forth between those zones, without the need for an intermediary. \ No newline at end of file diff --git a/docs/lotion/overview.md b/docs/lotion/overview.md deleted file mode 100644 index b28d2c75b..000000000 --- a/docs/lotion/overview.md +++ /dev/null @@ -1,54 +0,0 @@ -# Overview - -Lotion is an alternative to the Cosmos SDK and allows you to create blockchain apps in JavaScript. It aims to make writing new blockchain apps fast and easy by using the ABCI protocol to build on top of Tendermint. Lotion lets you write secure, scalable applications that can easily interoperate with other blockchains on the Cosmos Network using IBC. - -Lotion itself is a tiny framework; its true power comes from the network of small, focused modules built upon it. Adding a fully-featured cryptocurrency to your blockchain, for example, takes only a few lines of code. - -For more information see the [website](https://lotionjs.com) and [GitHub repo](https://github.com/keppel/lotion), for complete documentation which expands on the following example. - -## Building an App - -### Installation - -::: tip -Lotion requires __node v7.6.0__ or higher, and a mac or linux machine. -::: - -``` -$ npm install lotion -``` - -### Simple App - -`app.js`: - -```js -let lotion = require('lotion') - -let app = lotion({ - initialState: { - count: 0 - } -}) - -app.use(function (state, tx) { - if(state.count === tx.nonce) { - state.count++ - } -}) - -app.listen(3000) -``` - -run `node app.js`, then: - -```bash -$ curl http://localhost:3000/state -# { "count": 0 } - -$ curl http://localhost:3000/txs -d '{ "nonce": 0 }' -# { "ok": true } - -$ curl http://localhost:3000/state -# { "count": 1 } -``` diff --git a/docs/reference/baseapp.md b/docs/reference/baseapp.md new file mode 100644 index 000000000..3eaf1e2f7 --- /dev/null +++ b/docs/reference/baseapp.md @@ -0,0 +1,14 @@ +# baseApp + +`baseApp` requires stores to be mounted via capabilities keys - handlers can only access stores they're given the key to. The `baseApp` ensures all stores are properly loaded, cached, and committed. One mounted store is considered the "main" - it holds the latest block header, from which we can find and load the most recent state. + +`baseApp` distinguishes between two handler types - the `AnteHandler` and the `MsgHandler`. The former is a global validity check (checking nonces, sigs and sufficient balances to pay fees, +e.g. things that apply to all transaction from all modules), the later is the full state transition function. +During CheckTx the state transition function is only applied to the checkTxState and should return +before any expensive state transitions are run (this is up to each developer). It also needs to return the estimated +gas cost. + +During DeliverTx the state transition function is applied to the blockchain state and the transactions +need to be fully executed. + +BaseApp is responsible for managing the context passed into handlers - it makes the block header available and provides the right stores for CheckTx and DeliverTx. BaseApp is completely agnostic to serialization formats. \ No newline at end of file diff --git a/docs/reference/store/README.md b/docs/reference/store/README.md new file mode 100644 index 000000000..e69de29bb diff --git a/docs/resources/delegator-faq.md b/docs/resources/delegator-faq.md deleted file mode 100644 index 0de388a2c..000000000 --- a/docs/resources/delegator-faq.md +++ /dev/null @@ -1,69 +0,0 @@ -# Delegator FAQ - -## What is a delegator? - -People that cannot, or do not want to run [validator](/validators/overview.md) operations, can still participate in the staking process as delegators. Indeed, validators are not chosen based on their own stake but based on their total stake, which is the sum of their own stake and of the stake that is delegated to them. This is an important property, as it makes delegators a safeguard against validators that exhibit bad behavior. If a validator misbehaves, its delegators will move their Atoms away from it, thereby reducing its stake. Eventually, if a validator's stake falls under the top 100 addresses with highest stake, it will exit the validator set. - -Delegators share the revenue of their validators, but they also share the risks. In terms of revenue, validators and delegators differ in that validators can apply a commission on the revenue that goes to their delegator before it is distributed. This commission is known to delegators beforehand and can only change according to predefined constraints (see section below). In terms of risk, delegators' Atoms can be slashed if their validator misbehaves. For more, see Risks section. - -To become delegators, Atom holders need to send a "Bond transaction" from [Cosmos Voyager](/getting-started/voyager.md) where they specify how many Atoms they want to bond and to which validator. A list of validator candidates will be displayed in Cosmos Voyager. Later, if a delegator wants to unbond part or all of its stake, it needs to send an "Unbond transaction". From there, the delegator will have to wait 3 weeks to retrieve its Atoms. - -## Choosing a validator - -In order to choose their validators, delegators have access to a range of information directly in Cosmos Voyager. - -* Validator's name: Name that was chosen by the validator candidate when it declared candidacy. -* Validator's description: Description that was provided by the validator candidate when it declared candidacy. -* Validator's website: Link to the validator's website. -* Initial commission rate: The commission rate on revenue charged to any delegators (see below for more detail). -* Commission change rate: The maximum daily increase of the validator's commission -* Maximum commission: The maximum commission rate which this validator candidate can charge. -* Minimum self-bond amount: Minimum amount of Atoms the validator candidate need to have bonded at all time. If the validator's self-bonded stake falls below this limit, its entire staking pool (i.e. all its delegators) will unbond. This parameter exists as a safeguard for delegators. Indeed, when a validator misbehaves, part of its total stake gets slashed. This included the validator's own stake as well as its delegators' stake. Thus, a validator with a high amount of self-bonded Atoms has more skin-in-the-game than a validator with a low amount. The minimum self-bond amount parameter guarantees to delegators that a validator will never fall below a certain amount of self-bonded stake, thereby ensuring a minimum level of skin-in-the-game. - -## Directives of delegators - -Being a delegator is not a passive task. Here are the main directives of a delegator: - -* Perform careful due diligence on validators before delegating. If a validator misbehaves, part of its total stake, which includes the stake of its delegators, can be slashed. Delegators should therefore carefully select validators they think will behave correctly. -* Actively monitor their validator after having delegated. Delegators should ensure that the validators they're delegating to behaves correctly, meaning that they have good uptime, do not get hacked and participate in governance. They should also monitor the commission rate that is applied. If a delegator is not satisfied with its validator, it can unbond or switch to another validator. -* Participate in governance. Delegators can and are expected to actively participate in governance. A delegator's voting power is proportional to the size of its stake. If a delegator does not vote, it will inherit the vote of its validator. Delegators therefore act as a counterbalance to their validators. - -## Revenue - -Validators and delegators earn revenue in exchange for their services. This revenue is given in three forms: - -* Block provisions (Atoms): They are paid in newly created Atoms. Block provisions exist to incentivize Atom holders to stake. The yearly inflation rate fluctuates around a target of 2/3 bonded stake. If the total bonded stake is less than 2/3 of the total Atom supply, inflation increases until it reaches 20%. If the total bonded stake is more than 2/3 of the Atom supply, inflation decreases until it reaches 7%. This means that if total bonded stake stays less than 2/3 of the total Atom supply for a prolonged period of time, unbonded Atom holders can expect their Atom value to deflate by 20% per year. -* Block rewards ([Photons](https://blog.cosmos.network/cosmos-fee-token-introducing-the-photon-8a62b2f51aa): They are paid in Photons. Initial distribution of Photons will take the form of a hard spoon of the Ethereum chain. Atom holders will vote on the parameter of this hard spoon, like the date of the snapshot or the initial distribution. Additionally, bonded Atom holders will receive newly created Photons as block rewards. Photons will be distributed at a fixed rate in proportion to each bonded Atom holder's stake. This rate will be decided via governance. -* Transaction fees (various tokens): Each transfer on the Cosmos Hub comes with transactions fees. These fees can be paid in any currency that is whitelisted by the Hub's governance. Fees are distributed to bonded Atom holders in proportion to their stake. The first whitelisted tokens at launch are Atoms and Photons. - -## Validator's commission - -Each validator's staking pool receives revenue in proportion to its total stake. However, before this revenue is distributed to delegators inside the staking pool, the validator can apply a commission. In other words, delegators have to pay a commission to their validators on the revenue they earn. Let us look at a concrete example: - -We consider a validator whose stake (i.e. self-bonded stake + delegated stake) is 10% of the total stake of all validators. This validator has 20% self-bonded stake and applies a commission of 10%. Now let us consider a block with the following revenue: - -* 990 Atoms in block provisions -* 10 Photons in block reward -* 10 Atoms and 90 Photons in transaction fees. - -This amounts to a total of 1000 Atoms and 100 Photons to be distributed among all staking pools. - -Our validator's staking pool represents 10% of the total stake, which means the pool obtains 100 Atoms and 10 Photons. Now let us look at the internal distribution of revenue: - -* Commission = `10% * 80% * 100` Atoms + `10% * 80% * 10` Photons = 8 Atoms + 0.8 Photons -* Validator's revenue = `20% * 100` Atoms + `20% * 10` Photons + Commission = 28 Atoms + 2.8 Photons -* Delegators' total revenue = `80% * 100` Atoms + `20% * 10` Photons - Commission = 72 Atoms + 7.2 Photons - -Then, each delegator in the staking pool can claim its portion of the delegators' total revenue. - -## Risks - -Staking Atoms is not free of risk. First, staked Atoms are locked up, and retrieving them requires a 3 week waiting period called unbonding period. Additionally, if a validator misbehaves, a portion of its total stake can be slashed (i.e. destroyed). This includes the stake of their delegators. - -There are 3 main slashing conditions: - -* Double signing: If someone reports on chain A that a validator signed two blocks at the same height on chain A and chain B, this validator will get slashed on chain A -* Unavailability: If a validator's signature has not been included in the last X blocks, the validator will get slashed by a marginal amount proportional to X. If X is above a certain limit Y, then the validator will get unbonded -* Non-voting: If a validator did not vote on a proposal and once the fault is reported by a someone, its stake will receive a minor slash. - -This is why Atom holders should perform careful due diligence on validators before delegating. It is also important that delegators actively monitor the activity of their validators. If a validator behaves suspiciously or is too often offline, delegators can choose to unbond from it or switch to another validator. Delegators can also mitigate risk by distributing their stake across multiple validators. diff --git a/docs/resources/diagrams/todo.md b/docs/resources/diagrams/todo.md deleted file mode 100644 index b032b55cf..000000000 --- a/docs/resources/diagrams/todo.md +++ /dev/null @@ -1,17 +0,0 @@ -The following diagrams should be created to aid in comprehension of the SDK: - - Genesis circuit - - App structure (aka use of baseapp in something like gaia) - - Simulation framework - - Slashing Mechanism - - Staking Mechanism - - Staking/Slashing Mechanism specific to use of hooks - - Governance Mechanism - - Distribution Mechanism - - Inflation Mechanism (easier) - - IBC Mechanism - -These diagrams should reference specific structs/interfaces from the codebase, -logic flow and interconnectivity with other mechanisms etc. It's recommended that -https://www.draw.io/ be used, hence the raw diagram xml can be saved directly to -the Cosmos-SDK repo and adapted with the codebase. - diff --git a/docs/resources/faq.md b/docs/resources/faq.md deleted file mode 100644 index 91d902dbb..000000000 --- a/docs/resources/faq.md +++ /dev/null @@ -1,95 +0,0 @@ -# FAQ - -## Overview - -### How do I get Atoms? - -If you participated in the fundraiser, you can check your suggested atom balance at [fundraiser.cosmos.network](https://fundraiser.cosmos.network). -If not, you must wait until the [Cosmos Network launches](/roadmap) and Atoms are traded on exchanges. - -### Are Atoms listed on exchanges? - -No. The Cosmos Network mainnet has not yet launched, which means Atoms are _not_ on exchanges. $CMOS and $ATOM tokens are _not_ Cosmos Network native tokens. - -### How do I participate in the fundraiser? - -The [fundraiser](https://fundraiser.cosmos.network) is closed. The Interchain Foundation raised funds from private individuals and has hosted a public fundraising event on which ended on April 6, 2017. Both $ETH and $BTC were accepted in the fundraiser. The security of the fundraising process has been vetted extremely carefully. - -### What is the initial allocation of Atoms? - -As a public, decentralized network, the allocation of Atoms is decided by those who run the software for the Cosmos Hub. To faciliate a decision, we are creating a Swiss non-profit, the [Interchain Foundation](https://interchain.io), which is responsible for co-ordinating fundraising and allocating funds to get the network off the ground. The foundation will suggest a allocation of Atoms according to the results of the fundraiser. Users will ultimately decide the distribution for themselves when they run the software. - -The Interchain Foundation will suggest that 5% of the Atoms go to its initial donors, 10% go to the Interchain Foundation, 10% go to the company developing most of the software, and the remaining 75% to be distributed according to the results of the private and public fundraisers. - -### What is the team developing the Cosmos Network? - -The Cosmos Network is the first project being funded by the Interchain Foundation. Its development is led primarily by the [Tendermint team](/about/team). - -### What's the difference between Tendermint, the Cosmos Network, and the Cosmos Hub? - -- [Tendermint](https://tendermint.com) is a general purpose blockchain engine that uses a Byzantine-fault tolerant consensus protocol and allows applications to be written in any programming language. -- The Cosmos Network is a heterogenous network of Proof-of-Stake blockchains that can interoperate with one-another. -- The Cosmos Hub is the first Proof-of-Stake blockchain to be launched by the Cosmos Network; it uses Tendermint consensus, contains a built in governance protocol, and serves as co-ordinater for interoperability between other blockchains. -- Atoms: The native cryptocurrency on the Cosmos Hub. Atoms are necessary for participating in the consensus protocol and transacting on the network. - -### When will the Cosmos Network launch? - -Please check [our roadmap](https://cosmos.network/roadmap). - -### What is the utility of Atoms? - -Public, decentralized networks require high levels of security and spam-prevention that are best achieved by economic means: participants in the consensus must incur some economic cost, and all transactions processed by the network must pay a fee. Since we want to use Proof-of-Stake validators instead of Proof-of-Work miners, we require validators of the Cosmos Hub to make a large security deposit in Atoms - if they misbehave, their Atoms are revoked by the protocol! - -The more Atoms in security deposits, the more stake on the line; the more skin-in-the-game; the greater the economic security. In this sense, the Atoms act like virtual miners. - -To achieve spam-prevention, all transactions on the Cosmos Hub must pay a fee in Atoms. The fee may be proportional to the amount of computation required by the transaction, similar to Ethereum's concept of "gas". The fees are collected by the validators and distributed proportionately to the Atoms held in security deposits. - -## Interoperability - -### What's an IBC packet? - -[IBC packets](https://blog.cosmos.network/developer-deep-dive-cosmos-ibc-5855aaf183fe) are packets of data that one blockchain wishes to send to another blockchain. But instead of literally sending a packet of bytes via the TCP/IP or UDP/IP protocol (which is designed for singular, physical, machines), IBC packets require cryptographic proof-of-existence. Since no single node or validator has the authority to speak on behalf of the entire blockchain, and, since we don't want to rely on the integrity of the IP internet infrastructure, instead we rely on a cryptographic proof of a blockchain hash commit (+2/3 of signatures for that blockchain hash) along with a Merkle-proof from the aforementioned blockhash to a packet in the blockchain's "application state", which proves that the blockchain validators agreed to publish this packet of information. So, anyone who sees an IBC packet (regardless of the source of this data) can verify its integrity. - -### How does one exchange currencies in this system? - -For tokens outside the Cosmos system, they can only be introduced via pegged -derivatives. Read about interoperating with existing blockchains here: [Peggy](https://blog.cosmos.network/the-internet-of-blockchains-how-cosmos-does-interoperability-starting-with-the-ethereum-peg-zone-8744d4d2bc3f). - -``` - _ peg smart contract - / - [ Ethereum ] <--> [ EtherCosmos Peg Zone ] <-IBC-> [ Cosmos Hub ] <-IBC-> (Bitcoin) [ PoW/Casper ] - [ Tendermint ] [ Tendermint ] <-IBC-> (exchange) -``` - -### How does Cosmos manage governance? - -In Cosmos, the stakeholders are well defined, as is the prior social contract. Ethereum had a hard time with the fork because they had to ask the ether holders as well as the miners, but the ether holders had no prior social contract or obligation to partake in governance, so no quorum could be reached in time. Asking the miners is necessary to ensure that the hard-fork will have support, but after a while they tend to simply follow the money and incentives. - -Cosmos is different because instead of anonymous miners we have social contract bound validators and delegators who have stake, and, they have the obligation to partake in governance. - -## Validators - -### What is the maximum number of validators in Cosmos? What about nodes? - -We will start with 100 validators. Anyone else can be a node. To start, the validators will be the same across all shards - they will run the shards concurrently. Over time, these restrictions will be loosened. Misbehaviour in the consensus on any shard will result in security deposits being revoked. - -### What will be the process for abandoning validators that misbehave? - -If a validator misbehaves on its own by double-signing at the same height & round, then the evidence is very short and simple -- it's just the two conflicting votes. This evidence can be included in the the Cosmos Hub as a Slash transaction, and the validator will immediately become inactive and slashed after the Slash transaction gets committed. - -If there is a zone fork, either of the Cosmos Hub or any of the zones, the two conflicting commits also constitute evidence. This is a much more complicated data structure. It is guaranteed to slash at least 1/3 of the validators' atoms for that zone. - -### What's the difference between a Delegator and a Validator? - -A [validator](/staking/validators) has an active key involved in signing votes in the consensus protocol. A validator must also have some Atoms in a security deposit. Since there will only be a limitted number of validators, [other Atom holders can delegate](/staking/delegators) to the validators, thereby contributing to the economic security of the system by putting their funds on the line if the validator misbehaves. In return, they earn a share of the transaction fees and any inflationary rewards. - -### Can delegators also be validators? - -Delegators are never validators. If a validator wishes to delegate, they need to do so with their free and unbonded Atoms. - -### How are validator voting powers determined and changed? - -Validators are initially determined according to a public vote among Atom holders to be carried out before the launch of the Cosmos Hub. Atom holders delegate to the various candidates, and the top 100 candidates will be the initial validators. Once [the Hub launches](/roadmap), the vote will be a continuous process where users shuffle around their delegated Atoms, thereby changing the validator set. - -Part of the purpose of the fundraiser is to distribute Atoms across a wide variety of individuals and organizations so that the validator set will be sufficiently decentralized for a robust network. In the event of attacks or mishaps, the blockchain may need to purge bad actors through socially co-ordinated hard-forks. The ability to account for misbehaviour and co-ordinate hardforks helps make the system antifragile. diff --git a/docs/resources/whitepaper-ko.md b/docs/resources/whitepaper-ko.md deleted file mode 100644 index f87460c4c..000000000 --- a/docs/resources/whitepaper-ko.md +++ /dev/null @@ -1,756 +0,0 @@ -# 코스모스 - -분산원장 네트워크 - -**저자:** - -Jae Kwon <mailto:jae@tendermint.com><br> -Ethan Buchman <mailto:ethan@tendermint.com> - -For discussions, [join our community chat](https://riot.im/app/#/room/#cosmos:matrix.org)! - -**번역 및 제작:** - -한승환, 이승민, 김기현, 윤승완, HJ Kim(금마), 김석현 - -\[[toc]] - -## 서 론 (Introduction) - -오픈 소스 생태계, 탈중앙화 파일 공유, 퍼블릭 암호화폐 등이 연달아 성공하면서 탈중앙화 인터넷 프로토콜들이 사회경제적 인프라를 근본적으로 개선하는데 사용될 수 있다는 것을 알게 되었다. 비트코인[\[1\]](https://bitcoin.org/bitcoin.pdf)(암호화폐)과 제로캐시[\[2\]](http://zerocash-project.org/paper)(프라이버시용 암호화폐) 같은 특화된 블록체인 어플리케이션들이 있었고, Augur(예측 시장)와 TheDAO[\[4\]](https://download.slock.it/public/DAO/WhitePaper.pdf)(투자 클럽) 같은 무수한 분산 애플리케이션들을 가진 이더리움(Ethereum)[\[3\]](https://github.com/ethereum/wiki/wiki/White-Paper) 같은 범용 스마트 컨트랙트 플랫폼들이 있었다. - -그러나 지금까지 이런 블록체인들은 엄청난 에너지 비효율, 형편없거나 제한적인 성능, 미성숙한 거버넌스 메커니즘 등을 비롯해 많은 결함들을 경험해왔다. 세그위트(Segregated-Witness)[\[5\]](https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki)와 비트코인 NG(BitconNG)[\[6\]](https://arxiv.org/pdf/1510.02037v2.pdf) 같은 비트코인 이체 처리량을 확장하기 위한 여러 제안들이 있었지만, 완전 감사가능성(complete auditability)을 희생하지 않는 이상, 여전히 단일한 기계의 용량에 제한을 받는 수직적 스케일링 솔루션이다. 라이트닝 네트워크(Lightning Network)[\[7\]](https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf)는 원장에서 특정한 이체들을 완전히 제외하여 비트코인의 확장성에 도움을 주며, 소액결제와 프라이버시 보호형 지급 플랫폼(privacy preserving payment rails)에 적합하지만, 보다 범용적인 확장에는 적합하지 않을 수 있다. - -가장 이상적인 해결책은 다수의 병렬 블록체인들이 각자의 보안 특성을 유지하면서 호환되는 것이다. 하지만 작업증명의 경우, 이것이 불가능은 아니더라도 상당히 어렵다는 것이 증명되었다. 예를 들어, 병합채굴(merged-mining)은 부모체인의 보안을 위해 수행된 작업을 자녀체인에서 재사용하는 것을 허용하지만, 이체들은 여전히 차례대로 각 노드에 의해 검증돼야 하고, 부모체인의 해시파워 대부분이 자녀체인에서도 병합채굴을 하고 있지 않을 경우, 병합 채굴된(merged mined) 블록체인이 공격에 취약해진다. 우리는 [대안 블록체인 네트워크 아키텍처들(alternative blockchain network architectures)](http://vukolic.com/iNetSec_2015.pdf)에 대한 학술적 고찰을 하였고, 그 중 일부에 대해서는 요약과 결점들을 아래의 [관련 연구(Related Work)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#related-work)에서 다루었다. - -우리는 이 모든 문제들을 해결하는 새로운 블록체인 네트워크 아키텍처인 코스모스(Cosmos)를 제시하고자 한다. 코스모스는 존(zone)이라고 불리는 많은 독립된 블록체인들의 네트워크이다. 존은 텐더민트 코어(Tendermint Core)[\[8\]](https://github.com/tendermint/tendermint/wiki)를 통해 작동하는데, 텐더민트 코어는 일관적이고 안전한 고성능의 [유사](https://blog.cosmos.network/tendermint-vs-pbft-12e9f294c9ab?gi=7d54da26ffe)[PBFT](https://blog.cosmos.network/tendermint-vs-pbft-12e9f294c9ab?gi=c320a745ea23) 합의 엔진을 제공하며, 악의적인 공격자들에게 엄격한 [포크 책임(fork-accountability)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#fork-accountability)을 묻는다. 텐더민트 코어의 BFT 합의 알고리즘은 공개형 지분증명 블록체인들의 확장성을 확보하는데 적합하다. - -코스모스 상의 첫 번째 존을 코스모스 허브(Cosmos Hub)라고 부른다. 코스모스 허브는 단순한 거버넌스 메커니즘을 통해 네트워크가 적응하고 업그레이드 할 수 있게 하는 ‘다중 자산 지분증명 암호화폐(multi-asset proof-of-stake cryptocurrency)’이다. 그에 더해서, 코스모스 허브는 다른 존들을 연결함으로 확장될 수 있다. - -코스모스 네트워크의 허브와 존들은 블록체인 간 통신(IBC: inter-blockchain communication) 프로토콜을 통하여 상호 통신한다. 이 프로토콜은 블록체인들을 위한 일종의 가상 UDP 또는 TCP 역할을 한다. 토큰들은 존들 간 거래소의 유동성(exchange liquidity)없이도 안전하고 신속하게 하나의 존에서 다른 존으로 전송될 수 있다. 존들 간 토큰 전송은 코스모스 허브를 통과하며, 코스모스 허브는 각 존이 보유한 토큰 총액을 추적한다. 허브는 각 존을 다른 존들의 실패(failure)로부터 격리한다. 또한 누구든지 코스모스 허브에 새로운 존을 연결할 수 있기 때문에, 존들은 앞으로의 새로운 블록체인 혁신과의 장래 호환성도 확보한다. - -## 텐더민트 (Tendermint) - -본 절에서는 텐더민트 합의 프로토콜과 이를 통한 애플리케이션을 구축을 위해 사용되는 인터페이스를 다룬다. 상세한 내용은 [부록(appendix)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#appendix)참조 - -### 검증인 (Validators) - -전형적인 비잔틴 장애 허용 알고리즘들(BFT: Byzantine fault-tolerant algorithms)에서는 각 노드가 동일한 가중치를 갖는다. 텐더민트에서는 노드들이 ‘0 이상’의 *투표권(voting power)*을 가지며, 양(+)의 투표권을 갖는 노드들을 *검증인\*\*(validators)*라고 부른다. 검증인들은 다음 블록에 동의하는 암호서명(cryptographic signature), 즉 *투표(vote)*를 전파(broadcast)함으로써 합의 프로토콜에 참여한다. - -검증인(validator)들의 투표권은 제네시스(genesis) 당시에 결정되거나 블록체인에 의해 결정론적으로(deterministically) 변경되기도 한다. 예를 들어, 코스모스 허브와 같은 지분증명 애플리케이션에서는 투표권이 담보물로서 본딩된(bonded) 지분 토큰(staking token)의 양에 의해 결정될 수도 있다. - -_주: ⅔, ⅓ 과 같은 분수들은 모든 **검증인**들이 동등한 가중치를 갖지 않는 한, **검증인**들의 총수가 결코 아니라 총 투표권**에서 차지하는** **비중**을 가리킨다. 주: +⅔ 는 " ⅔ 초과"를 의미하**며**, ⅓+은 "⅓ 이상"을 의미한다._ - -### 합의 (Consensus) - -텐더민트는 DLS 합의 알고리즘에서 파생되는, 부분적 동기 BFT 합의 프로토콜(synchronous BFT consensus protocol)이다[\[20\]](http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf). 텐더민트는 단순함, 성능 그리고 [포크 책임(fork-accountability)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#fork-accountability)이 특징이다. 이 프로토콜은 공개되있고 고정된 세트의 검증인들(a fixed, known set of validators)을 요구하는데, 각 검증인 공개 키에 의해 식별된다. 검증인들은 한 번에 하나의 블록, 즉 이체 목록(a list of transactions)에 대해 합의를 시도한다. 블록에 대한 합의는 라운드(round)를 통해 진행된다. 각 라운드에는 블록을 제안하는 라운드 리더(round-leader), 즉 제안자(proposer)가 있다. 그 다음, 검증인들은 제안된 블록을 받아들일 것인지 또는 다음 라운드로 넘어갈 것인지에 대한 단계별 투표를 진행한다. 각 라운드의 제안자는 검증인 리스트(ordered list)에서 투표권에 비례해 결정론적으로 선택된다. - -프로토콜에 대한 세부정보는 [여기](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm)를 참조 - -텐더민트 보안은 압도적 다수결(+⅔)과 잠금 메커니즘(locking mechanism)을 통해 최적의 비잔틴 장애 허용을 사용하는 것으로부터 유래한다. 다음을 보장한다. - -- 안전성을 저해(violation of safety)하기 위해서는 ⅓+의 투표권이 비잔틴이어야 하고, ‘2 개를 초과하는 값들’이 커밋(commit)되어야 한다. - -- 프로토콜은 임의의 검증인들 세트가 안전성을 저해하는 데 성공하거나 또는 시도만 하더라도, 이들을 식별할 수 있다. 충돌하는 블록들에 찬성 투표 하거나 정당하지 않은 투표들을 전파하는 이들 모두가 포함된다. - -텐더민트는 강력한 보장들에 더해 성능면에서도 탁월하다. 5 개 대륙 7 개 데이터센터에 분산되어 있는 64 개 노드의 대중품 클라우드(commodity cloud)를 기준으로, 텐더민트 합의는 약 1~2 초의 커밋 지연속도(commit latencies)와 함께 초당 수천 개의 트랜잭션(transaction)를 처리한다. 특히, 검증인들이 실패하거나 악의적으로 조작된 투표를 전파하는 가혹한 공격 상황(adversarial conditions)에서도 초당 1,000 번 수준의 트랜잭션 성능은 가볍게 능가한다. _아래 그림 참조_ - -![Figure of Tendermint throughput performance](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/tendermint_throughput_blocksize.png) - -### 라이트 클라이언트 (Light Clients) - -텐더민트 합의 알고리즘의 특징은 단순화된 라이트 클라이언트 보안인데, 모바일 및 사물인터넷에 사용되기도 적합하다. 비트코인 라이트 클라이언트는 ‘블록헤더 체인’들을 동기화(sync)하여 가장 많은 작업증명을 가진 체인을 찾아야 하지만, 텐더민트 라이트 클라이언트는 ‘검증인 세트’의 변경을 추적해서, 최신 블록에서 +⅔ 프리커밋(PreCommit)을 검증하기만 하면 최신 상태를 결정할 수 있다. - -간결한 라이트 클라이언트 증명들은 [블록체인 간 통신(inter-blockchain communication)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#/h)도 가능하게 한다. - -### 공격 방지 (Preventing Attacks) - -텐더민트는 ‘[long-range-nothing-at-stake double spend](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#/h)’나 [검열(censorship)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#/h) 등을 방지하기 위한 다양한 보호장치를 가지고 있다. 세부사항은 [부록(appendix)](https://github.com/cosmos/cosmos/blob/master/WHITEPAPER.md#/h)참조 - -### TMSP - -텐더민트 합의 알고리즘은 텐더민트 코어(Tendermint Core)라는 프로그램에서 구현된다. 텐더민트 코어는 모든 결정적 블랙박스 애플리케이션(deterministic blackbox application)을 분산 복제 블록체인(distributedly replicated blockchain)으로 전환시킬 수 있는 어플리케이션 어그노스틱(application-agnostic) "합의 엔진"이다. 아파치 웹 서버나 Nginx 가 CGI 나 FastCGI 를 통해 워드프레스 애플리케이션에 접속하는 것처럼, 텐더민트 코어는 텐더민트 소켓 프로토콜(TMSP: Tendermint Socket Protocol)을 통해 블록체인 애플리케이션들과 연결된다. - -비트코인을 예로 들어보자면, 비트코인은 각 노드가 완전 감사되는 UTXO(fully audited Unspent Transaction Output) 데이터베이스를 유지하는 암호화폐 블록체인이다. 만일 누군가 비트코인과 같은 시스템(Bitcoin-like system)을 TMSP 상에 만들고자 하면, ‘**텐더민트 코어**’는 아래와 같은 부분들을 책임진다. - -- 노드 간 블록 및 이체 공유 - -- 정규(canomical)/변경배제(immutable) 이체순서의 확립(블록체인) - -반면, ‘**TMSP 의 어플리케이션**’은 다음을 책임진다. - -- UTXO 데이터베이스 유지 - -- 이체의 암호서명 검증 - -- 존재하지 않는 이체를 지불방지 - -- 클라이언트들의 UTXO 데이터베이스에 대한 query 허용 - -텐더민트는 애플리케이션 프로세스와 합의 프로세스 사이에 매우 단순한 API 를 제공함으로써 블록체인 설계를 분해할 수 있다. - -## 코스모스 개요 (Cosmos Overview) - -코스모스는 독립 병렬 블록체인들의 네트워크이며, 해당 블록체인들은 각기 텐더민트와 같은 전형적인 BFT 합의 알고리즘에 기반한다[1]. - -코스모스의 최초 블록체인이 곧 코스모스 허브가 된다. 코스모스 허브는 블록체인 간 통신(IBC) 프로토콜을 통해 다른 많은 블록체인들(즉, _존들_)과 연결된다. 코스모스 허브는 수많은 유형의 토큰을 추적하고, 연결된 각 존의 토큰 총 수를 기록한다. 모든 존들 간의 코인 전송은 코스모스 허브를 거치기 때문에 토큰은 존들 간의 유동 거래소(liquid exchange) 방식 없이 - -이 아키텍처는 어플리케이션 상호운용성, 확장성 그리고 무결절 업그레이드 가능성(seamless upgradability)을 포함한 블록체인이 직면한 많은 문제들을 해결한다. 예를 들면, Bitcoind, Go-Ethereum, 크립토노트(CryptoNote), ZCash 등을 포함한 임의의 블록체인 시스템으로부터 파생된 존들이 코스모스 허브에 연결될 수 있다. 이러한 존들을 통해 코스모스는 전역 이체(global transaction) 요구를 충족시킬 때까지 무한히 확장할 수 있다. 그리고 존들은 탈중앙화 거래소(distributed exchange)에도 매우 적합하며, 실제로 지원될 것이다. - -코스모스는 단일 분산 원장에 불과한 것이 아니며, 코스모스 허브도 폐쇄형 플랫폼(walled garden, 울타리 친 정원)이 아니다. 오히려 코스모스는 미래 금융 시스템의 새로운 기초가 될 수 있는 개방형 분산원장 네트워크를 위한 프로토콜이며, 암호학 원리, 잘 설계된 경제(sound economics), 합의 이론, 투명성, 책임(accountability) 등의 원칙에 기반하고 있다. - -### Tendermint-BFT DPoS - -코스모스 허브는 텐더민트 BFT 합의 알고리즘에 기반하는, 코스모스 네트워크 내 최초의 퍼블릭 블록체인이다. 텐더민트 오픈 소스 프로젝트는 비트코인의 작업증명(PoW) 합의 알고리즘이 지닌 속도, 확장성, 환경 등의 문제를 처리하기 위해서 2014 년에 탄생했다. 1988 년 MIT 에서 개발한 증명된 BFT 알고리즘들을 활용하고 개선해서, 텐더민트 팀은 제 1 세대 지분증명(PoS) 암호화폐들이 겪은 Nothing-at-stake 문제를 해결하는 지분증명 암호화폐를 최초로 개념증명하였다. - -오늘날, 대부분의 비트코인 모바일 지갑들은 이체 검증(transaction verification)을 제공하는 신뢰할 수 있는 서버(trusted server)를 이용한다. 이는 PoS 가 여러번의 이체확인(confirmation)을 기다려야만, 이체확정된 것으로 간주되기 때문이다. 이중지불(double-spend) 공격이 이미 코인베이스(CoinBase) 같은 서비스들에서 일어난 바 있다. - -다른 블록체인 합의 시스템들과는 달리 텐더민트는 즉각적이고 안전한(provably-secure) 모바일-클라이언트 지불 검증을 제공한다. 텐더민트는 분기(fork)할 수 없도록 설계되었기 때문에, 모바일 지갑들에서 즉시 이체확인이 가능하며 실제로 신뢰 없는(trustless) 지불을 실현한다. 이는 IoT 관련 어플리케이션들에도 큰 영향을 줄 것이다. - -(비트코인 채굴자들과 유사한 역할이지만 그 대신에 암호서명을 통해 투표하는) 코스모스의 검증인들은 블록을 커밋 할 책임이 있는 안전한 전용 머신들이어야 한다. 검증인이 아닌 이들(non-validators)은 ‘아톰’이라고 부르는 지분 토큰(staking tokens)을 임의의 검증인에게 위임하여 일정한 블록 수수료(block fees)와 아톰 보상(atom rewards)을 얻을 수는 있으나, 위임 검증인(delegate validator)이 해킹당하거나 프로토콜을 위반할 경우, 처벌을 받게 되는 리스크가 있다. 텐터민트 BFT 합의의 입증된 안정성 보장과 (검증인 및 위임자) 주주들의 담보 보증(collateral deposit)은 노드들에 심지어는 라이트 클라이언트들에게 증명 가능하고 정량화 가능한 보안성(provable, quantifiable security)을 제공한다. - -### 거버넌스 (Governance) - -‘공개형 분산원장’은 헌법(constitution)과 거버넌스 시스템(governance system)을 가져야 한다. 비트코인은 업그레이드 등의 조정을 위해 일정 부분 비트코인 재단(Bitcoin Foundation)이나 채굴에 의존하지만, 진행속도가 느리다. 이더리움의 경우에는 The DAO 해킹의 처리를 위한 하드포크 후 ETH 와 ETC 로 분할되었는데, 이는 사전에 그런 의사결정을 하기 위한 사회적 계약이나 메커니즘이 존재하지 않았기 때문이다. - -코스모스 허브의 검증인과 위임자는 코스모스 허브의 정책들을 위한 ‘코드가 아니라 평이한 언어로 된 헌법(규칙)’을 표결에 부쳐 수정할 뿐 아니라 ‘블록 가스 한계(block gas limit)’ 같은 시스템의 사전에 설정된 제한들을 업그레이드를 통해 자동으로 변경하는 프로포잘(proposals)도 표결에 부칠 수 있다. 헌법은 The DAO 사건 같은 절도 및 버그관련 문제들의 이해관계자들이 응집(cohesion)할 수 있도록 도우며, 보다 신속하고 깔끔한 해결책이 나올 수 있도록 한다. - -각 존 역시 자체적인 헌법과 거버넌스 메커니즘을 가질 수 있다. 예를 들어, 코스모스 허브가 변경 배제(immutability) -코스모스 허브의 노드 실행 버그를 제외하고는 롤백 금지(no roll-backs)와 같은- 헌법을 적용한 경우라면, 각 존은 절도와 버그를 위한 롤백 관련 자체 정책을 설정할 수 있다. - -코스모스 네트워크는 상이한 정책의 존들 간 상호운용성을 확보함으로 사용자들에게 궁극적 자유와 실험의 기회를 제공한다. - -## 허브와 존 (The Hub and Zones) - -여기서는 탈중앙화(decentralization)와 확장성(scalability)의 새로운 모델을 설명한다. 코스모스는 텐더민트에 기반한 복수의 블록체인들의 네트워크이다. 기존의 프로포잘들이 전역적 이체 순서(total global transaction ordering)를 가진 ‘단일 블록체인’을 목표로 하는 반면에, 코스모스는 수많은 블록체인들이 상호 병행 실행(run concurrently)하는 동시에 상호운용성을 확보하도록 한다. - -기본적으로 코스모스 허브는 ‘존(zone)’이라고 부르는 여러 독립된 블록체인들을 관리한다(‘존’은 ‘샤드(shard)’로도 불림, 데이터베이스의 스케일링 기법인 ‘샤딩(sharding)’참조). 존들은 최근의 블록 커밋들을 끊임없이 허브로 전송하며, 허브는 이를 통해 항상 최신 상태를 유지한다. 마찬가지로, 각 존도 허브의 상태를 제공받는다(다만 허브를 통한 간접적인 경우를 제외하고는 존들끼리는 서로 이러한 작업을 하지 않는다). 존들은 정보의 송신과 수신의 증거인 머클 증명(Merkle-proofs)을 포스팅함으로 정보의 패킷들을 교환한다. 이 메커니즘을 ‘블록체인 간 통신(inter-blockchain communication)’ 또는 IBC 라고 칭한다. - -![Figure of hub and zones acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/hub_and_zones.png) - -어느 존이든 스스로 허브가 되어 비순환 그래프(acyclic graph)를 형성할 수 있으나 명료성을 위해 우리는 단지 하나의 허브와 많은 비-허브 존들이 존재하는 단순한 구성만 설명할 것이다. - -### 허브 (The Hub) - -코스모스 허브는 멀티자산 분산 원장(multi-asset distributed ledger)을 관리하는 블록체인인데, 여기서 토큰은 개인 사용자들이나 존 자체가 보유할 수 있다. 이 토큰들은 ‘코인 패킷(coin packet)’이라 부르는 특수 IBC 패킷을 통해 하나의 존에서 다른 존으로 이동할 수 있다. 허브는 존들에 걸쳐 각 토큰 총액의 전역적 불변성(global invariance)을 보존한다. IBC 코인 패킷 이체는 반드시 송신자, 허브 및 수신자 블록체인에 의해 커밋 되어야 한다. - -코스모스 허브는 전체 시스템을 위한 토큰의 ‘중앙 원장’ 역할을 하기 때문에, 보안이 무엇보다도 중요하다. 각 존은 단 4 명(혹은 BFT 합의가 필요하지 않을 경우에는 훨씬 더 적은 수)의 검증인으로도 안전해지는 텐더민트 블록체인일 수 있는 반면, 허브는 대륙 네트워크 분할(continental network partition)이나 국가 주도적 공격(nation-state sponsored attack) 같은 가장 심각한 공격 시나리오들에도 견딜 수 있어야 한다. 따라서 전역적으로 탈중앙화된 검증인 세트로 보안되어야 한다. - -### 존 (The Zones) - -코스모스 존은 허브와 IBC 메시지를 교환하는 독립 블록체인이다. 허브의 관점에서, 존은 IBC 패킷을 사용하여 토큰을 송수신할 수 있는 ‘멀티자산 동적구성원 멀티시그너처 계정(multi-asset dynamic-membership multi-signature account)’이다. 암호화폐 계정과 마찬가지로 존은 보유 중인 토큰보다 더 많은 토큰을 전송할 수 없지만, 토큰을 보유한 다른 존으로부터 토큰을 수신할 수는 있다. 존은 토큰 유형들의 ‘소스(source)’로 지정되어, 해당 토큰을 추가 공급할 수도 있다. - -코스모스 허브의 아톰들은 허브의 검증인이 아니라, ‘허브에 연결된 존의 검증인’들에 의해 스테이킹될 수도 있다. 이 존들에 대한 이중지불 공격은 텐더민트의 포크 책임인 아톰의 대폭 감소(slashing)라는 결과를 가져오게 되는 반면, 투표권의 +⅔ 이 비잔틴인 존이라면 무효한 상태(invalid state)를 커밋할 수 있다. 코스모스 허브는 다른 존들에 커밋 된 이체들은 검증하거나 실행하지 않으므로, 자신이 신뢰하는 존으로 토큰을 전송하는 것은 사용자들의 책임이다. 장래에는 코스모스 허브의 거버넌스 시스템이 존 실패(failures)를 위한 허브 개선 프로포잘을 만들수도 있다. 예를 들면, 공격이 탐지될 때, 일부(또는 전체) 존들의 아웃바운드 토큰 전송을 억제함으로 ‘긴급 서킷 브레이킹(토큰 전송 일시중단)’을 할 수도 있다. - -## 블록체인 간 커뮤니케이션 (Inter-blockchain Communication (IBC)) - -여기서는 허브와 존 간의 상호 통신방법을 설명한다. 예를 들어, ‘존 1(Zone 1)’, ‘존 2(Zone 2)’와 ‘허브(Hub)’의 세 블록체인이 있고, ‘존 1’이 ‘허브’를 통과하여 ‘존 2’로 가는 패킷을 만들고자 한다. 패킷이 하나의 블록체인에서 다른 블록체인으로 이동하기 위해, 송신 체인(sending chain)이 수신지로 가는 패킷을 발행했다는 증거를 수신 체인(receiving chain)에 포스팅한다. 수신 체인이 이 증거를 확인하기 위해서는 송신자의 블록 헤더(block headers) 정보를 알아야 한다. 이 메커니즘은 사이드체인(sidechain)이 사용하는 메커니즘과 유사한데, 상호 작용하는 두 체인이 ‘존재 증명 데이터그램의 양방향 스트림(a bidirectional stream of proof-of-existence datagram)’을 통해 서로를 ‘인식’해야한다 (이체). - -IBC 프로토콜을 두 가지 이체 방식을 통해 정의할 수 있다: 블록체인이 최근의 블록 해시를 임의의 관측자에게 증명하는 IBCBlockCommitTx 이체, 그리고 블록체인이 송신자의 어플리케이션이 패킷을 머클증명을 통해 최근 블록해시로 전파했다는 것을 증명하는 IBCPacketTx 이체 - -IBC 구동방식을 IBCBlockCommitTx 와 IBCPacketTx 로 분할함으로, 수신체인의 내부 수수료 시장 메커니즘(native fee market-mechanism)이 커밋될(승인될) 패킷을 결정하도록 하며, 송신체인이 얼마나 많은 아웃바운드 패킷들을 허용할 지 자유롭게 정할 수 있도록 한다. - -![Figure of Zone1, Zone2, and Hub IBC without acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_without_ack.png) - -위의 예에서, ‘존 1’의 블록해시를 (또는 ‘존 2’ 상의 ‘허브’의 블록 해시를) 업데이트 하기 위해서는, IBCBlockCommitTx 이체가 ‘존 1’의 블록해시와 함께 ‘허브’에 (또는 ‘허브’의 블록해시와 함께 ‘존 2’에) 반드시 포스팅 되어야 한다. - -_두 IBC 이체 유형에 관한 보다 상세한 정보는 [IBCBlockCommitTx](#ibcblockcommittx) 와 [IBCPacketTx](#ibcpacketcommit) 를 참조._ - -## 사용사례 (Use Cases) - -### 분산거래소 (Distributed Exchange) - -비트코인이 원장을 복사하고 분산하여 더 안전해지는 것처럼, 거래소도 블록체인 상에 올림으로 내/외부의 해킹으로부터 더욱 안전해질 수 있다. - -현재 커뮤니티에서 칭하는 탈중앙화 거래소(decentralized exchange)는 ‘아토믹 크로스체인(Atomic Cross-Chain, AXC)’ 이체에 기반하고 있다. 이를 통해 두 명의 유저는 두 개의 서로 다른 원장(체인)에서 이체를 일으키고 해당 이체는 동시에 각 원장에 기록되거나 또는 그 어떤 원장에도 기록되지 않는다. 예를 들어, AXC 를 이용하면 두 명의 유저가 서로의 이더와 비트코인(또는 서로 다른 원장의 토큰)을 교환할 수 있다. AXC 기반 거래소의 장점은, 두 명의 유저 모두 서로 또는 거래소 서비스를 신뢰할 필요가 없다는 점이다. 그러나 거래를 위해서는 양 당사자가 반드시 온라인 상태여야만 한다는 점이 치명적인 단점이다. - -또 다른 방식의 탈중앙화 거래소는 자체적인 분산원장에서 작동하는 ‘대규모의 복제 분산 거래소(a mass-replicated distributed exchange)’이다. 유저는 ‘지정가 주문(limit order)’을 내고 컴퓨터를 종료한다. 이 경우, 유저가 온라인 상태가 아니어도 거래가 성사된다. 블록체인이 해당 거래를 매칭시키고 완료시키는 것이다. - -중앙화된 거래소의 경우, 이미 많은 지정가 주문이 걸려 있고 따라서 유저들을 더 쉽게 유치하게 된다. 거래소에서 유동성은 더 큰 유동성을 야기하고 강한 네트워크 효과(또는 승자독식)를 일으키게 된다. 현재 24 시간 거래량 기준, Poloniex 가 $20M 으로 가장 많은 거래량을 가지고 있고, Bitfinex 가 $5M 로 그 다음에 위치하고 있다. 이러한 사례를 생각해보면, AXC 기반의 새로운 탈중앙화 거래소가 중앙화된 거래소를 이길 가능성은 거의 없다. 탈중앙화된 거래소가 중앙화된 거래소와 경쟁하기 위해서는 많은 거래량을 확보해야 한다. 오직 ‘분산 거래소(distributed exchange)’만이 그것을 가능하게 할 수 있다. - -텐더민트는 또한 매우 빠른 이체 커밋을 제공한다. 일관성의 훼손없이 빠른 완결성을 우선순위에 둠으로써 코스모스 상의 존(zone)들의 이체를 빠르게 완결한다. 이는 거래소의 이체나 IBC 를 통한 존들 간의 이체에 동일하게 적용된다. - -현재 암호화폐 거래소들의 상황을 고려할 때, 코스모스를 이용해 분산거래소(일명 ‘코스모스 덱스’)를 만드는 것은 매우 적절하다. 코스모스 덱스(Cosmos DEX)의 이체 처리량과 커밋 속도는 중앙화된 거래소에 필적한다. 유저들은 지정가 주문을 올리고, 이 주문은 당사자의 온라인 상태여부에 관계없이 완결된다. 트레이더들은 텐더민트, 코스모스 허브, 그리고 IBC 를 통해, 빠른 속도로 자금을 존(zone)들이나 거래소로 이동시킬 수 있다. - -### 다른 화폐에 가치고정 (Pegging to Other Cryptocurrencies) - -특권 존(privileged zone)은 다른 암호화폐를 페깅한 토큰의 소스 역할을 할 수 있다. 페그(peg)는 코스모스 허브와 존의 관계와 본질적으로 유사한데, 코스모스 허브와 존 모두는 토큰이 한 쪽에서 다른 쪽으로 이동했다는 증명을 검증하기 위해 상대방의 최신 블록을 반드시 알고 있어야 한다. 코스모스 네트워크의 페그 존(peg-zone)은 다른 암호화폐뿐 아니라 허브도 추적한다. 페그 존을 통한 간접참조(indirection)는 허브의 로직이 단순하면서도 비트코인의 작업증명 채굴 같은 다른 블록체인 합의 전략들에 독립적(agnostic)인 상태를 유지할 수 있게 한다. - -예를 들어, 허브의 검증인 세트와 동일한 어떤 검증인 세트를 갖는 존이 이더-페그(ether-peg) 역할을 할 수 있을 것이다. 여기서 ‘페그-존(peg-zone)’의 TMSP 애플리케이션은 ‘외부 이더리움 블록체인의 페그 컨트랙트(peg contract)’와 IBC 메시지를 교환할 수 있다. 이더 보유자들이 이더리움 상의 페그 컨트랙트로 이더를 전송함으로 코스모스의 페그 존에 이더를 전송하게 된다. 이렇게 이더가 일단 한번 수신되면, 페그 컨트랙트가 페그 존으로부터 IBC 패킷을 수신하지 않는 한 이더는 인출되지 못 한다. 페그 컨트랙트에서 특정 이더리움 계정으로부터 이더를 수신했다는 IBC 패킷을 페그 존으로 보내면, 이에 대응되는 계정이 동일한 잔고와 함께 페그 존 위에 생성된다. 페그 존 위의 (페깅된) 이더를 허브로 또는 허브로부터 전송할 수 있으며, 페깅 이더를 폐기하면서 이더리움 상의 특정 인출 주소로 그 이더를 도로 전송할 수 있고, 이더의 인출을 허용하기 위해 페그 존에서 이체가 발생했음을 증명하는 IBC 패킷을 이더리움 페그 컨트랙트에. - -물론 그런 페깅 컨트랙트는 악의적 검증인 세트를 통한 리스크에 노출되어 있다. ⅓+ 비잔틴 투표권이 분기(fork)를 만들어 페깅된 이더를 페그 존에 유지하는 동시에 이더리움 위의 페그 컨트랙트를 통해 이더를 인출할 수도 있다. 더 좋지 않은 시나리오는 +⅔ 비잔틴 투표권이, 페그 존의 원래 페깅 로직으로부터 이탈하여, 페그 컨트랙트로 보내지는 모든 이더를 노골적으로 훔칠 수도 있다는 것이다. - -이런 문제들은 페그를 ‘완전 책임적(totally accountable)’으로 설계하여 해결할 수 있다. 예를 들면, 허브나 원점(페깅될 토큰의 원래 블록체인)으로부터의 모든 IBC 패킷이 페그-존의 응답(acknowledgment)을 요구하도록 할 수도 있을 것이다. 허브와 원점은 페그-존 검증인들이 담보물(collateral)을 포스팅(post) 할 수 있게 하고, 독립 감사들(independent auditors)의 도전을 허용하기 위해 페그 컨트랙트로부터 나가는 토큰 전송은 지연되도록 하는 식이다. 그리고 담보 언본딩(unbonding) 기간은 충분히 길어야 한다. 이 부분은 미확정 상태로 두고, 누구든지 향후에 관련 제안을 해서 코스모스 거버넌스 시스템에 승인을 받아 적용할 수 있도록 한다. - -정치사회적 분위기가 아직은 갖추어져 있지 않긴 하지만, 국가 통화의 책임기관들, 특히 은행들의 조합으로부터 검증인 세트를 형성함으로 국가의 법정통화에 페깅하는 존들을 허용할 수도 있다. 물론, 강력한 법체계에 기반하는 화폐만 허용하도록 충분한 고려가 되어야 할 것이며, 충분히 큰 규모의 공증인이나 기관들을 통해 감사시스템을 강제하여 은행들의 활동을 관리해야할 것이다. - -이렇게 통합이 된다면, 참여 은행에 은행계좌가 있는 누구든지 존에 있는 은행계좌로부터 존에 있는 다른 계좌나 허브나 혹은 또 다른 존으로 법정화폐를 자유롭게 이동시킬 수 있을 것이다. 이 점에서 코스모스 허브는 법정화폐들과 암호화폐들 간의 전달자 역할을 하고, 지금까지 상호운용성을 거래소(exchanges)의 영역으로 한정시킨 장벽들을 제거할 수 있다. - -### 이더리움 확장성 (Ethereum Scaling) - -이더리움의 해결되지 않은 쟁점 하나는 스케일링 문제를 해결하는 방법이다. 현재는 각 이더리움 노드들이 모든 이체를 처리하고 또 모든 상태를 저장한다. [링크](https://docs.google.com/presentation/d/1CjD0W4l4-CwHKUvfF5Vlps76fKLEC6pIwu1a_kC_YRQ/mobilepresent?slide=id.gd284b9333_0_28) - -텐더민트는 이더리움의 작업증명보다 훨씬 빠르게 블록들을 커밋 할 수 있기 때문에, 텐더민트 합의에 기반한 페깅된 이더(pegged-ether)를 운용하는 EVM 존들은 이더리움 블록체인에 보다 높은 성능을 제공할 수 있다. 뿐만 아니라, 비록 코스모스 허브 및 IBC 패킷 자체는 임의계약 로직 실행을 허용하지 않지만, 서로 다른 존들에서 실행되는 이더리움 컨트랙트 간 토큰이동의 조정을 위해 사용되며, 샤딩(sharding)을 통한 토큰 중심 이더리움 스케일링을 위한 기반을 제공할 수 있다. - -### 멀티 어플리케이션 통합 (Multi-Application Integration) - -코스모스 존들은 존의 생성 초기에 정해진 어플리케이션 로직에 따라 작동하며, 거버넌스를 통해 지속적으로 업데이트될 수 있다. 그런 유연성 덕분에 코스모스 존들이 이더리움이나 비트코인 같은 다른 암호화폐들의 페그 역할을 하고, 동일한 코드베이스를 활용하면서도 상이한 검증인 세트와 초기 분포(initial distribution)를 갖는 블록체인 파생상품(derivatives)을 만들 수 있다. 이를 통해 이더리움, 제로캐시, 비트코인, 크립토노트 등과 같은 기존의 암호화폐 프레임워크들이 ‘고성능 합의 엔진(텐더민트 코어)’을 통해 공통의 네트워크에서 사용될 수 있으며, 엄청난 상호운용성의 기회를 가질 수 있게 된다. 뿐만 아니라, 멀티 자산 블록체인으로서, 단일 이체는 다수의 인풋과 아웃풋을 포함할 수 있고, 어떤 토큰이든지 인풋이 될 수 있기 때문에, 주문들은 다른 플랫폼들을 통해 매칭되더라도 코스모스가 직접 탈중앙화 거래소(exchange)를 위한 플랫폼 역할을 할 수 있다. 또한 존은 오더북 기능이 있는 ‘분산형 장애-허용 거래소’ 역할을 할 수도 있는데, 해킹 공격에 당하곤 하는 기존의 중앙집중형 암호화폐 거래소들에 대한 강력한 개선책이 될 수 있다. - -존은 또한 기업 및 정부 시스템들의 블록체인 버전(blockchain-backed versions)을 지원할 수도 있다. 전통적으로 조직에 의해 운영된 서비스에 대해서 기반이 된느 부분은 통제권을 유지하고, 특정 부분만을 존 위의 TMSP 애플리케이션으로 운영한다면, 퍼블릭 코스모스 네트워크의 보안과 상호운용성을 활용할 수 있게 된다. 따라서 코스모스는 블록체인 기술을 활용하고자 하지만 (분산된) 제 3 자에게 통제를 완전히 내주는 것은 경계하는 조직들에 일거양득을 제공할 수 있다. - -### 네트워크 분할 완화 (Network Partition Mitigation) - -일각에서는 텐더민트 같은 ‘일관성선호 합의 알고리즘(consistency-favouring consensus algorithm)’에 심각한 문제가 있다고 주장한다. +⅔ 투표권(⅓+은 오프라인)을 갖는 분할이 생기지 않도록 방지하는 것이 합의과정 자체를 멈추도록 할 수 있다는 것이다. 코스모스 아키텍처는 지역 자치 존(regional autonomous zone)을 갖는 전역 허브(global hub)를 사용함으로 이 문제를 완화할 수 있는데, 이 지역적 자치 존은 각 존의 투표권이 지리적 위치에 기초하여 분배되는 존을 가리킨다. 예를 들어, 개별 도시들이나 지역들이 자체적인 존을 운영하면서 공통의 허브(예. 코스모스 허브)를 공유하여, 일시적인 네트워크 분할로 허브가 중단되더라도 자치활동은 지속될 수 있도록 할 수 있다. 강인한 연합형 장애허용 시스템(robust federated fault-tolerant system)이 설계되기 위해서는 실제 지리, 정치 및 네트워크 토폴로지 관련 특징들이 고려되어야 할 것이다. - -### 연합 명칭 결의 시스템 (Federated Name Resolution System) - -네임코인(NameCoin)은 비트코인 블록체인을 이용하여 ‘명칭 결의(name-resolution)’ 문제를 해결하고자 한 최초의 블록체인 중 하나였다. 안타깝지만 이 접근법에는 몇 가지 문제들이 있었다. - -네임코인으로 예를 들면, *@satoshi*라는 명칭이 과거 어느 시점에 특정 공개키로 등록되었다는 것은 검증할 수 있지만, 이후의 모든 블록들을 다운로드 하지 않는 한 그 공개키가 이후로 업데이트 된 적이 있는지는 알 수 없다. 이는 비트코인의 UTXO 머클화 모델의 한계 때문인데, 이 모델에서는 ‘상태’가 아닌 ‘이체의 기록’만 블록 해시로 머클화된다--이는 존재를 증명하지만, 이후로 업데이트가 없었다는 것을 증명하지는 못한다. 따라서 완전 노드를 신뢰하거나 전체 블록체인을 다운로드 하면서 상당한 비용을 지불해야만 가장 최근 값을 확실히 알 수 있다. - -머클 탐색 트리(Merkle-ized search tree)가 네임코인에 구현되더라도, 작업증명에 대한 의존성 때문에 라이트 클라이언트 검증에서 문제가 생긴다. 라이트 클라이언트는 전체 블록체인의 블록헤더들(또는 적어도 마지막 업데이트 이후의 모든 헤더들)을 모두 다운로드 해야 한다. 이는 대역폭 사용이 소모시간과 함께 선형으로 증가한다는 것을 의미한다[\[21\]]][21]. 또한 작업증명 블록체인 위의 명칭 변경을 위해서는 추가적인 블록확인을 기다려야 하는데 이것이 비트코인에서는 최대 1 시간까지도 걸릴 수 있다. - -텐더민트의 경우, 변경을 위해 필요한 것은 (투표권을 통해) 검증인 정족수가 서명한 최근의 블록 해시, 그리고 명칭에 대한 현재 값(current value)의 머클 증명이다. 이를 통해, ‘명칭 값’을 간결하고 신속하며 안전한 라이트 클라이언트 검증을 할 수 있다. - -코스모스에서 이 개념을 확장시킬 수 있다. 코스모스의 각 명칭등록 존(name-registration zone)은 ‘.com’이나 ‘.org’와 같은 최상위 도메인(TLD: top-level-domain)을 가질 수 있고, 자체적인 거버넌스와 등록규칙을 설정할 수 있다. - -## 발행과 인센티브 (Issuance and Incentives) - -### 아톰 토큰 (The Atom Token) - -코스모스 허브는 ‘다중자산 분산원장’이며 특별한 내부 토큰인 ‘_아톰(atom)_’을 가지고 있다. 아톰은 코스모스 허브의 유일한 지분 토큰(staking token)이다. 아톰은 보유자가 투표, 검증 또는 다른 검증인들에게 위임을 하기 위해 필요하다. 이더리움의 이더와 마찬가지로 아톰 역시 스팸공격 완화를 위한 이체수수료(transaction fees) 지불을 위해 사용될 수 있다. 추가되는 인플레이션 아톰(inflationary atoms)과 블록 이체수수료가 검증인들에게 그리고 검증인들에게 위임한 위임자들(delegators)에게 보상된다. - -BurnAtomTx 이체 기능을 통해 ‘지급 준비금 풀(reserve pool)’에서 일정 비율의 토큰 금액(proportionate amount of tokens)을 회수(recover)할 수 있다. - -#### 크라우드펀딩 (Fundraiser) - -코스모스가 최초로 생성될 때 아톰 토큰과 검증인의 배분은 코스모스 크라우드세일의 자금제공자들(75%), 사전 자금 제공자들(5%)과 코스모스 주식회사(Cosmos Corp)(20%)로 가게 된다. 그 이후, 아톰 총액 중 1/3 이 본딩된(bonded) 검증인들과 위임자들에게 매년 보상될 것이다. - -이 계획은 변경될 수 있으며 상세내용은 [Crowdfund Plan](https://github.com/cosmos/cosmos/blob/master/PLAN.md) 참조 - -### 검증인수의 제한 (Limitations on the Number of Validators) - -비트코인 등의 작업증명 블록체인들과는 달리, 텐더민트 블록체인은 검증인의 수가 많아질수록 통신의 복잡도가 증가하여 속도가 느려진다. 물론 충분한 검증인들을 통해 블록체인의 전세계 분산, 매우 빠른 이체확인 속도를 제공할 수 있다. 또한 대역폭, 저장공간 및 병렬 컴퓨팅 용량이 증가함에 따라 장래에 더욱 많은 검증인들을 지원할 수 있을 것이다. - -제네시스 시에는 최대 검증인 수가 100 명으로 설정될 것이고, 이 수치는 10 년 동안 13%의 비율로 증가하여 최종적으로는 총 300 명의 검증인을 가지게 될 것이다. - - Year 0: 100 - Year 1: 113 - Year 2: 127 - Year 3: 144 - Year 4: 163 - Year 5: 184 - Year 6: 208 - Year 7: 235 - Year 8: 265 - Year 9: 300 - Year 10: 300 - ... - -### 제네시스 일이후에 검증인되기 (Becoming a Validator After Genesis Day) - -아톰 보유자들은 ‘BondTx 이체’를 서명하여 제출함으로써 검증인이 될 수 있다. 담보로 제공된 아톰 액수는 ‘0’보다 많아야 한다. 현재 검증인 세트의 크기가 허용된 최대 검증인 수보다 많은 경우를 제외하고는 누구라도 언제든지 검증인이 될 수 있다. 만일 검증인의 수가 허용치 이상인 경우, 가장 적은 검증인이 보유한 유효 아톰의 액수보다 더 높은 아톰 액수(위임받은 아톰 포함)를 담보로 제공해야만 한다. 이러한 방식으로 새로운 검증인이 기존 검증인를 대체할 수 있으며, 기존 검증인은 비활성이 되고 모든 아톰과 위임 아톰(delegated atom)은 언본딩 상태(unbonding state)로 돌아간다. - -### 검증인에 대한 처벌 (Penalties for Validators) - -고의든 아니든 검증인이 정해진 프로토콜을 어길 때에는, 처벌이 주어져야 한다. 동일한 높이(height) 및 라운드에서의 이중 서명이나 "prevote-the-lock"(텐더민트 합의 프로토콜의 규칙)의 위반 같은 행위는 증거를 통해 즉시 인정된다. 이 경우, 검증인은 유효한 지위(good standing)를 상실하게 되며, 본딩된 아톰과 ‘지급준비금 풀(reserve pool)’내 토큰의 비례분 즉, 통칭하여 ‘지분(stake)’을 상당수 잃을 것이다. - -때로는 지역 네트워크 단절, 전원 장애나 그 밖의 이유들로 인해 검증인이 단절될 수 있을 것이다. 만일 ‘과거 어느 시점의 ValidatorTimeoutWindow 블록’에서 검증인의 커밋 투표가 이루어지지 않은 횟수가 ValidatorTimeoutMax Absent 횟수 이상인 경우, 해당 검증인은 비활성화 되고 지분의 ‘Validator TimeoutPenalty(디폴트 1%)’만큼 잃게 될 것이다. - -어떤 ‘악의적’ 행위는 블록체인에 명확한 증거를 남기지 않을 수 있다. 이런 경우, 압도적 다수의 합의가 존재한다면, 검증인들이 외부에서 합의한 뒤 악의적인 검증인을 강제로 타임아웃(timeout) 시킬 수 있다. - -‘투표권 ⅓+’이 악의적으로 연합하여 코스모스를 중단시키거나, 이들이 악의적 행동의 증거가 블록체인으로 들어오지 않게 검열하여 삭제하는 경우, 허브는 하드포크를 통한 블록재조정(reorg) 프로포잘로 복구(recover)되어야 한다. - -### 이체 수수료 (Transaction Fees) - -코스모스 허브 검증인들은 아톰 뿐 아니라, 어떠한 유형의 토큰이라도 이체수수료로 받을 수 있다. 각 검증인은, BlockGasLimit 을 초과하지 않는 한, 원하는 어떤 교환비율이든 주관적으로 정할 수 있고 원하는 어떤 이체든 선택할 수 있다. 징수된 수수료는 아래에 명시된 세금들을 제외하고, 매 ValidatorPayoutPeriod(검증인지불기간, 디폴트 1 시간)마다 본딩된 아톰(bonded atoms)에 비례하여 본딩된 주주들(bonded stakeholders)에게 재분배된다. - -징수된 이체수수료 중 ‘지급준비금 세금(Reserve Tax, 디폴트 2%)’은 ‘지금준비금 풀’에 충당되며 이를 통해 ‘지금준비금 풀’을 늘리고 코스모스 네트워크의 보안과 가치를 높이는데 사용될 것이다. 또한 공유세(Commons Tax, 디폴트 3%)는 공유재의 자금으로 충당될 것이다. 이 자금들은 CustodianAddress 로 가게 되고, 거버넌스 시스템의 결정에 따라 분배된다. - -투표권을 다른 검증인들에게 위임하는 아톰 보유자들은 위임 받은 검증인에게 커미션(commission)을 지불한다. 커미션은 각 검증인이 정할 수 있다. - -### 해커에 대한 인센티브 제공 (Incentivizing Hackers) - -코스모스 허브의 보안은 검증인들과 위임자들의 검증인 선택에 달려있다. 취약성 발견 및 조기 보고를 권장하기 위해 코스모스 허브는 해커들이 "이 검증인은 해킹 당했다. 본 주소로 포상금을 보내주기 바란다."라고 알릴 수 있는 ‘ReporthackTx 이체’를 통해 해킹성공 발표를 권장한다. 해킹 즉시, 해당 검증인과 위임자들은 비활성화되고, 이들의 아톰 일부가 ‘HackPunishmentRatio(해킹처벌비율, 디폴트 5%)’만큼 감소되고, 해커는 포상금 주소를 통해 ‘HackRewardRatio(해크보상비율, 디폴트 5%)’만큼 보상받는다. 검증인은 백업 키(backup key)를 사용하여 나머지 아톰을 회수(recover)해야 한다. - -이러한 방법을 이용해 ‘본딩이 완료되지 않은 비귀속(unvested) 아톰’을 전송하려는 악의적 시도를 막기위해, 감별자와 위임자들의 귀속과 비귀속 아톰 비율은 ReportHackTx 전후로 동일하게 유지될 것이다. 해커 포상금은, 존재할 경우, 얼마의 비귀속 아톰을 포함할 것이다. - -### 거버넌스 (Governance Specification) - -코스모스 허브는 소프트웨어 업그레이드와 헌법(규정) 수정뿐 아니라 시스템의 변수 파라미터들과 같은 블록체인의 다양한 변화를 조정하기 위해서 명확한 거버넌스 메커니즘을 가진 분산형 조직에 의해 운영된다. - -모든 검증인들은 모든 프로포잘에 대한 투표책임이 있다. 적시에 투표하지 않는 경우, 해당 검증인은 ‘결석처벌기간(AbsenteeismPenaltyPeriod-디폴트 1 주)’ 동안 자동으로 비활성화된다. - -위임자들(delegators)은 그들이 위임한 검증인의 투표를 자동으로 물려받는다. 이 투표는 수동으로 취소(overriden manually)될 수 있다. 언본딩된(unbonded) 아톰들은 어떤 투표권도 얻지 않는다. - -각 프로포잘은 ‘최소 프로포잘 보증금(MinimumProposalDeposit)’ 토큰을 요구하는데, 이는 아톰을 포함한 하나 이상의 토큰들일 수 있다. 각 프로포잘에 대해 투표자들은 보증금을 사용하기로 투표할 수 있다. 투표자의 절반 이상이 보증금을 사용하기로 선택할 경우, 그 보증금은 지급보증금 풀에 충당된다. 다만, 프로포잘이 스팸이었다거나 기타의 경우라서 소각되는(burned) 아톰은 제외한다. - -각 프로포잘에 대해, 투표자들은 다음의 옵션으로 투표할 수 있다: - -- Yay(찬성) -- YayWithForce(강력히 찬성) -- Nay(반대) -- NayWithForce(강력히 반대) -- Abstain(기권) - -프로포잘의 통과 여부를 결정시에는 과반수의 투표가 요구되지만, 1/3+이 "강력히(with force)" 반대 투표함으로써 과반수의 결정을 거부할 수 있다. 다만 이렇게 과반수가 거부될 경우, 모두가 ‘거부권패널티블록(VetoPenaltyFeeBlocks, 디폴트 1 일 가치의 블록)’을 통해 수수료를 상실함으로 처벌 받고, 과반수 결정을 거부한 당사자는 자신의 아톰 중 ‘거부권패널티아톰(VetoPenaltyAtoms, 디폴트 0.1%)’ 만큼을 추가로 상실한다. - -### 파라미터 변경 프로포잘 (Parameter Change Proposal) - -여기서 정의된 파라미터들 중 어느 것이든 ParameterChangeProposal 의 통과를 통해 변경될 수 있다. - -### 텍스트 프로포잘 (Text Proposal) - -다른 모든 프로포잘들(예. 업그레이드 프로포잘)은 일반적인 TextProposal 을 통해 조정된다. - -## 로드맵 (Roadmap) - -[Plan](https://github.com/cosmos/cosmos/blob/master/PLAN.md) 참조. - -## 관련 연구 (Related Work) - -지난 수십 년 동안 블록체인 합의구조와 확장성 부분에서 많은 혁신이 있었다. 아래에서 중요한 몇 가지를 간략히 개관한다. - -### 합의 시스템 (Consensus Systems) - -#### 고전적 비잔틴 장애 저항 (Classic Byzantine Fault Tolerance) - -악의적인 참여자가 있는 합의는 80 년대 초로 거슬러 올라가는 문제인데, 이때 레슬리 램포트(Leslie Lamport)는 프로세스가 단순한 ‘충돌 장애(crash fault)’와는 대조되는 것으로, 의도된 행위로부터 벗어나는 임의의 프로세스 행위(process behavior)를 가리키는 ‘비잔틴 장애(Byzantine fault)'라는 용어를 만들었다. 초기의 솔루션들은 메시지 지연시간에 상한(upper bound)이 존재하는 동기식 네트워크들을 위한 것이었다. 하지만 실제 사용은 항공기 제어기와 원자시계를 통해 동기화되는 데이터센터 같은 고도로 통제된 환경들로 제한되었다. 임의의 프로세스 행위를 최대 ⅓ 까지 허용할 수 있는 효율적인 ‘부분 동기 합의 알고리즘’으로서 ‘실용적 비잔틴 장애 허용(PBFT: Practical Byzantine Fault Tolerance)’[\[11\]](11)이 도입된 것은 90 년대 말이 되고 나서였다. PBFT 는 표준 알고리즘이 되었고, 많은 파생 변형들을 만들어내었다. 가장 최근에는 IBM 이 하이퍼레저(Hyperledger)에 기여를 위해 만든 변형이 있다. - -PBFT 와 비교할때 텐더민트 합의구조의 중요한 장점은 텐더민트가 단순하고 개선된 기반 구조를 가진다는 것이고, 이 구조 중의 일부는 블록체인 패러다임을 수용한 결과이다. 텐더민트 블록들은 순서대로 커밋되는데, 이것이 PBFT 의 관점 변화(view-changes)와 관련 있는 복잡도와 통신 오버헤드를 제거한다. 코스모스를 포함한 많은 암호화폐에서는, 블록 _N_ 자체가 아직 커밋되지 않았을 때에는 블록 _N+i_(_i >= 1_)의 커밋을 고려할 필요가 없다. 블록 *N*이 코스모스 존에 커밋되지 않은 이유가 대역폭인 경우, _N+i_ 블록들에 대해 대역폭 공유 투표(bandwidth sharing votes)를 사용해도 의미가 없다. 만일 네트워크 분할이나 오프라인 노드들 때문에 블록 *N*이 커밋되지 않은 경우, 블록 *N+i*은 어차피 커밋되지 않는다. - -뿐만 아니라, 블록들 속으로 이체를 배칭(batching)함으로 PBFT 의 체크포인팅(checkpointing) 기법 같은 주기적 다이제스트가 아닌, ‘어플리케이션 상태의 정규 머클 해싱’이 가능하다. 또한 이를 통해, 라이트 클라이언트들을 위한 보다 빠른 증명 가능한 이체 커밋과, 보다 빠른 블록체인 간 통신이 가능하다. - -텐더민트 코어는 PBFT 에 명시된 것 이상의 많은 최적화 사항들과 기능들을 가지고 있다. 예를 들면, 검증인이 제안하는 블록들은 전파 성능을 개선하는 방식으로 부분화되고 머클화(Merkle-ized) 되고 가십화(gossipped)된다(영감은 LibSwift[\[19\]](19) 참조). 또한 텐더민트 코어는 점 대 점(point-to-point) 접속에 관한 어떤 가정도 하지 않으며, P2P 네트워크가 연결되어 있기만 하다면 작동한다. - -#### 비트쉐어 위임 지분 (BitShares delegated stake) - -비트쉐어(BitShares)[\[12\]](12)는 최초의 지분증명(PoS: proof-of-stake) 적용 사례는 아니지만, PoS 블록체인들, 특히 ‘위임(delegated)’ PoS 블록체인의 연구와 채택에 상당한 기여를 했다. 비트쉐어에서 지분 보유자들은 이체명령 및 커밋 책임이 있는 '증인(witnesses)'과 소프트웨어 업데이트와 패러미터 변경 책임이 있는 '델리게이트(delegates)'를 선출한다. 비트쉐어는 이상적인 상태에서 고성능(100k tx/s, 1 초 지연시간)을 달성하기는 하지만, 악의적인 증인들이 아무런 경제적 처벌 없이 블록체인을 분기해 이중지불 공격을 가할 수 있다. 즉 '무보증(Nothing-at-Stake)' 문제를 겪는다. 비트쉐어는 이체들이 최근의 블록-해시들을 참조하게 함으로써 이 문제를 완화하고자 한다. 물론 주주들은 부정행위 증인들을 매일 제거하거나 대체할 수 있다. 하지만 이것이 성공한 이중지불공격에 대한 분명한 처벌은 결코 아니다. - -#### 스텔라 (Stellar) - -리플(Ripple)이 개척한 방식을 토대로, 스텔라(Stellar)[\[13\]](13)는 ‘연합형 비잔틴 합의(Federated Byzantine Agreement)’ 모델을 개선했는데, 여기에서는 합의참여 프로세스에서 ‘전역적으로 알려진 고정된 집합’을 요구하지는 않는다. 오히려 프로세스 노드는 각기 신뢰할 수 있는 프로세스 집합을 구성하는 하나 이상의 '정족수 슬라이스(quorum slices)'를 배포한다. 스텔라에서 '정족수'는 노드들의 집합(set)이며, 적어도 각 노드당 하나 이상의 정족수 슬라이스를 포함하여 합의에 도달하도록 한다. - -스텔라 메커니즘의 보안은 _임의의_ 두 정족수의 교집합이 비공(non-empty)이라는 가정에 의존한다. 또한 노드의 가용성을 위해서는, 정족수 슬라이스들 중 적어도 하나가 완전히 올바른(entirely correct) 노드들로 구성되어야 하며, 신뢰에 관한 중요한 가정 없이는 큰 정족수 슬라이스와 작은 정족수 슬라이스 사용 간에 발생하는 상호관계의 균형유지가 어려울 수 있다. 궁극적으로는 노드들은 충분한 장애허용이 가능한 ‘적절한 정족수 슬라이스’ 또는 ‘온전한 노드들(intact nodes)’을 어떻게든 선택해야 한다. 또한 그런 구성을 보장하는 유일한 전략은 계층적이고, 경계 경로 프로토콜(BGP: Border Gateway Protocol)과도 유사하다. 이는 전역 라우팅 테이블 확립을 위해 인터넷의 최상위 계층 ISP 들에 의해 사용되며, TLS 인증서를 관리하기 위해 브라우저들이 사용하기도 한다. 그러나 두 가지 모두 약한 보안성으로 악명높다. - -스텔라 논문에서 비판한 ‘텐더민트 기반 지분 증명 시스템’들은 이곳에 기술된 토큰 전략을 통해 해명할 수 있다. 이 전략에서는 미래의 수수료 및 보상에 대한 권리인 *아톰*이라는 이름의 새로운 유형의 토큰이 발행된다. ‘텐더민트 기반 지분 증명’은 상대적으로 단순성을 유지하면서도, 충분한 그리고 입증 가능한 보안을 제공한다는 점에서 이점이 있다. - -#### 비트코인 NG (BitcoinNG) - -BitcoinNG 는 블록 크기 확장과 같은 수직 확장성을 제공하는 방법이다. 또한 이러한 확장이 초래하는 부정적 경제 요인들을 최소화하였다. 이러한 개선은 리더 선정(leader election)과 이체 전파를 분리시킴으로 가능하다: 리더들은 우선 ‘마이크로 블록(micro-block)’들에 있는 작업증명(PoW)을 통해 선정되며, 그 이후 새로운 마이크로블록이 발견될 때까지 커밋할 이체내역들을 전파한다. 이러한 방식은 PoW 경쟁을 이기기 위해 소요되는 대역폭 요구량을 줄여준다. 또한 소규모 채굴자들이 더욱 공정하게 경쟁하고 이체가 더 정기적으로 커밋될 수 있도록 한다. - -#### 캐스퍼 (Casper) - -캐스퍼[\[16\]](16)는 이더리움 용으로 제안된 지분증명(PoS) 합의 알고리즘이다. 주요 운용 방식은 '베팅에 의한 합의(consensus-by-bet)'이며, 검증인들은 지금까지 본 다른 베팅들(bets)에 기초해 블록체인에 커밋 될 것으로 생각되는 블록에 반복적으로 베팅하게 되며, 이러한 방식으로 결국 완결성(finality)이 달성된다는 논리를 전제한다. [링크](https://blog.ethereum.org/2015/12/28/understanding-serenity-part-2-casper/). 이러한 ‘합의 베팅’이 캐스퍼 팀의 주요 연구영역인데, 도전이 되는 부분은 베팅 메커니즘을 ‘진화적으로 안정된 전략(evolutionarily stable strategy)’이 되도록 설계해야 한다는 점이다. 텐더민트와 비교할 때 캐스퍼의 강점은 '일관성(consistency)보다 가용성(availability)'을 제공하는 것으로 볼 수 있다. 합의는 투표권의 +⅔ 정족수를 요구하지 않으며 커밋 속도나 구현 복잡도를 희생했다고 볼 수 있다. - -### 수평 스케일링 (Horizontal Scaling) - -#### 인터레저 프로토콜 (Interledger Protocol) - -인터레저 프로토콜(Interledger protocol)[\[14\]](14)은 엄밀히는 확장성 솔루션이 아니다. 느슨하게 연결된 ‘쌍방 관계 네트워크’를 통해 상이한 원장 시스템들 간 ‘애드혹 상호운용성(ad hoc interoperation)’을 제공하는 방식이다. 라이트닝 네트워크(Lightning Network)의 경우처럼, ILP 는 지불을 용이하게 하는 프로토콜인데, 특히 서로 다른 유형의 원장들 간 지불에 초점을 맞추며 ‘아톰 이체 메커니즘(atomic transaction mechanism)’을 확장하여 ‘해시 잠금(hash-locks)’과 ‘공증인 정족수(quorum of notaries)’를 포함하로독 했으며, 이를 ‘아톰 전송 프로트콜(Atomic Transport Protocol)’로 칭한다. ‘원장 간 이체의 원자성(atomicity)’을 확보하기 위한 이러한 메커니즘은 텐더민트의 ‘라이트 클라이언트 SPV’ 작동방식과도 유사하다. 하단에서 ILP 와 코스모스/IBC 를 비교해본다. - -1. ILP 의 커넥터(connector) 공증인들은 구성원(membership) 변경을 지원하지 않으며 공증인들 간 유연한 가중치 부여를 지원하지 않는다. 반면에 IBC 는 블록체인 전용으로 설계되었고, 검증인들이 상이한 가중치를 가질 수 있으며 블록체인을 통해 구성원들의 변경이 가능하다. - -2. 라이트닝 네트워크나 ILP 에서는 지불 수신자가 송신자에게 ‘확인’을 되돌려 보내기 위해 반드시 온라인 상태여야 한다. IBC 를 통한 토큰 전송에서는, 수신자가 아니라, 수신자 블록체인의 검증인 세트가 ‘확인’을 제공할 책임이 있다. - -3. 가장 근본적인 차이는 ILP 커넥터들은 지불에 대한 책임을 지거나 권한 상태를 유지하고 있지 않은 반면에, 코스모스에서는 허브의 검증인들이 IBC 토큰 전송과 각 존이 가진 토큰 총액에 대한 권한을 가진다는 점이다(그러나 존 내 각 계정이 보유한 토큰 금액에 대한 권한은 없다). 이는 존에서 존으로 비대칭 토큰 전송을 안전하게 실행하기 위한 근본적 혁신이다. - -4. ILP 에서 원장 간 지불을 하기 위해서는 거래소 오더북(exchange orderbook)의 지원이 필요한데, 이는 하나의 원장에서 다른 원장으로의 ‘코인 비대칭 전송’이 없고 ‘가치나 시장 등가물(market equivalents)’의 전송만 있기 때문이다. - -#### 사이드체인 (Sidechains) - -사이드체인(sidechains)[\[15\]](15)은 비트코인 블록체인에 '페깅(pegged)'된 ‘대안 블록체인’을 통해 비트코인 네트워크를 확장하고자 하는 시도이다. 사이드체인은 비트코인이 비트코인 블록체인으로부터 사이드체인으로 효과적으로 이동할 수 있게 하고 사이드체인에서 각 체인의 특징을 이용한 실험들을 가능하게 한다. 코스모스 허브와 마찬가지로, 사이드체인과 비트코인은 상호 간 라이트 클라이언트 역할을 하며 코인 전송 시점을 결정하기 위해 SPV 증명을 사용한다. 물론 비트코인은 작업증명을 사용하기 때문에 비트코인 중심의 사이드체인들은 작업증명 합의 메커니즘으로부터 발생하는 여러 문제들과 리스크를 갖는다. 이는 ‘비트코인 극단주의자(Bitcoin-maximalist)’적 솔루션이며, 코스모스처럼 자체적으로 다양한 토큰들과 존 간 네트워크 위상(inter-zone network topology)을 지원하지는 않는다. 하지만, ‘양방향 페그(two way peg)’의 핵심 메커니즘 자체는 코스모스 네트워크의 방식과 원칙적으로 동일하다. - -#### 이더리움 확장성 노력 (Ethereum Scalability Efforts) - -이더리움은 확장성 확보를 위해 ‘이더리움 블록체인 상태’를 샤딩(sharding)하기 위한 여러 전략들을 연구 중이다. 현재의 이더리움 가상 머신(EVM)를 통해 ‘공유 상태 공간(shared state space)’에 ‘대한 추상적 계층(abstraction layer)’을 유지하는 방식이다. 또한 다수의 연구들이 진행되고 있다.[\[18\]](18)[\[22\]](22) - -#### 코스모스 vs 이더리움 2.0 Mauve (Cosmos vs Ethereum 2.0 Mauve) - -코스모스와 이더리움 2.0 Mauve[\[22\]](22)의 설계 목표에는 차이가 있다. - -- 코스모스가 토큰들에 대한 것이라면 Mauve 는 일반 계산(general computation)의 스케일링에 관한 것이다. - -- 코스모스는 EVM 에 구속되지 않으며, 심지어는 서로 다른 VM 들이 상호 운용될 수 있다. - -- 코스모스는 존의 검증 책임자를 존의 생성자가 결정하도록 한다. - -- (거버넌스 시스템의 결정과 상충되지만 않는다면) 누구든지 코스모스에서 새 존을 시작할 수 있다. - -- 허브는 존 실패(zone failures)를 격리시키며, 이를 통해 ‘전역 토큰 불변성(global token invariants)’을 보증한다. - -### 일반 스케일링 (General Scaling) - -#### 라이트닝 네트워크 (Lightning Network) - -라이트닝 네트워크는 비트코인 블록체인(그리고 그 밖의 퍼블릭 블록체인들)의 상위 계층에서 운용되도록 제안된 토큰 전송 네트워크로, 합의원장(consensus ledger) 외부의 대다수 이체들을 소위 ‘지불채널(payment channels)’로 이동시켜서 이체처리량(throughput)을 획기적으로 개선시킨다. 온 체인(on-chain) 암호화폐 스크립트를 통해 당사자들이 쌍무 상태기반 계약(bilateral stateful contracts)을 체결할 수 있도록 한다. 계약들의 상태는 디지털 서명에 의해 업데이트 되며, 또한 ‘크로스 체인 아토믹 스왑(cross-chain atomic swap)’를 통해 최초에 설정된 방식에 따라 합당한 증명을 블록체인에 전파함으로 계약이 마감된다. 다수의 당사자와 함께 지불 채널을 개방함으로, 라이트닝 네트워크의 참여자들은 당사자들이 지불을 라우팅하는 ‘포컬포인트(focal point)’의 역할을 할 수 있다. 또한 이러한 지불 채널에 실제 자본을 고정시킴으로 완전히 연결된 지불 채널을 개설하는 것도 가능하다. - -라이트닝 네트워크는 다수의 블록체인들에 걸쳐 확장되면서 교환시장을 통해 *가치(value)*를 전송하도록 하지만, 하나의 블록체인에서 다른 블록체인으로 *토큰(token)*을 비대칭적으로 전송하는 것은 불가능하다. 이와 관련된 코스모스 네트워크의 주된 이점은 그러한 토큰의 직접 전송을 지원한다는 것이다. 물론, 비용 절감과 프라이버시 차원에서 토큰 전송 메커니즘이 지불 채널들과 라이트닝 네트워크와 함께 적용되기를 기대한다. - -#### 세그위트 (Segregated Witness) - -세그위트(Segregated Witness)는 블록 당 이체처리량을 2-3 배 가량 증가시키는 것을 목적으로 하며, 동시에 신규 노드들이 더욱 빠르게 블록 동기화(block syncing)를 하도록 돕는 비트코인 개선 제안이다([참조](https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki)). 이 솔루션은 비트코인의 현재 프로토콜 내에서 작동하면서 소프트 포크 업그레이드(soft-fork upgrade)를 허용하는 방식이다. 따라서, 이전의 소프트웨어 버전을 가진 클라이언트들도 업그레이드를 받아 정상적으로 기능할 수 있다. 텐더민트의 경우, 새로운 프로토콜을 만드는데 아무런 제약이 없기 때문에 확장성에 대한 새로운 우선순위를 가진다. 주로 텐더민트는 채굴 대신 암호 서명에 기반한 ‘BFT 라운드 로빈 알고리즘(BFT round-robin algorithm)’을 사용한다. 이를 통해, 다중 병렬 블록체인을 이용한 수평 스케일링을 가능하게 하면서도, 보다 정기적이고 자주 발생하는 블록 커밋에는 수직 스케일링을 적용할 수도 있다. - -<hr/> - -## 부 록 (Appendix) - -### 포크 책임 (Fork Accountability) - -잘 설계된 합의 프로토콜이라면 허용 용량(tolerance capacity)이 초과되거나 합의가 실패할 경우에도 문제가 없어야 한다. 이는 ‘비잔틴 행동’에 상당한 금전적 보상이 있을 수 있는 경제시스템에서 특히 중요하다. 그러한 보장 중 가장 중요한 것이 ‘_포크 책임(fork-accountability)_’인데, 포킹과 같은 합의실패를 초래한 프로세스들이 프로토콜 규칙이나 법적 체계(가능하다면)에 의해 식별되고 처벌받을 수 있다. 법 체계를 신뢰할 수 없거나 적용하기에 과도한 비용이 들 경우, 검증인들이 담보 보증(security deposits)을 내고 참여하도록 하고 악영향을 주는 행위 시 해당 담보를 차감하거나 강제로 환수함으로 처벌할 수 있다[\[10\]](10). - -‘포크 책임’에서 다루는 것은 비트코인에서처럼 ‘네트워크 비동시성’과 ‘부분적 해시충돌’ 특성으로 인해 일어나는 정상적 분기와는 큰 차이가 있다. 많은 경우, 비동시성 때문에 악의적 포킹을 가려내는 것이 거의 불가능하기 때문에 비트코인의 경우 채굴자들이 포킹을 시도하다가 고아블록이 되어버리는 기회비용을 제외하고는 특별히 ‘포크 책임’을 물게하는 것이 어렵다. - -### 텐더민트 합의 (Tendermint Consensus) - -투표 단계들은 *프리보트(PreVote)*와 *프리커밋(PreCommit)*으로 나뉜다. 투표는 특정 블록이나 ‘_무효(Nil)_’를 위해 행사될 수 있다. 동일 라운드에서 단일 블록에 대한 +⅔ PreVote 집합을 *폴카(Polka)*라고 부르며, 동일 라운드에서의 단일 블록에 대한 +⅔ PreCommit 집합을 *커밋(Commit)*이라고 부른다. 만일 동일 라운드에서 ‘무효(Nil)’에 +⅔ PreCommit 상태라면, 다음 라운드로 이동한다. - -결함 있는 리더들을 검출하고 무시하기 위해 엄격한 결정성(determinism)을 사용하다가 오히려 약한 동시성(synchrony)을 초래할 수 있다. 따라서 검증인들은 무효(Nil)를 Prevote 하기 전에, 일정 시간(*TimeoutPropose)*을 기다리게 되며 이 TimeoutPropose 값은 매 라운드마다 점점 증가한다. 검증인이 네트워크의 +⅔ 로부터 전파받을 때에만 단 한번 진행이 되므로, 라운드의 나머지 부분 진행은 완전히 비동시적이다. 실제로, 약한 동시성 가정( weak synchrony assumption)을 계속해서 좌절시키고 결국 블록 커밋을 위한 합의를 실패시키기 위해서는 극단적으로 강한 공격자가 필요하다. 또한 각 검증인에게 서로 다른 TimeoutPropose 랜덤값을 적용한다면, 공격은 더욱 어려워진다. - -‘추가 제약조건 집합’, 즉 ‘잠금 규칙(Locking Rules)’은 네트워크가 각 높이에서 단 하나의 블록만을 커밋하도록 보장한다. 특정한 높이에서 두 개 이상의 블록을 커밋되게 하려는 시도는 모두 발각될 수 있다. 첫째로, 블록에 대한 프리커밋(PreCommit)은 해당 블록에 대한 폴카의 형태로 정당화(justification)되어야 한다. 만일 검증인이 이미* R_1* 라운드에서 블록을 프리커밋했다면, 이는 해당 블록에 로킹*(locked)*되며, _R_2_ 라운드의 새로운 프리커밋은 반드시 *R_1 < R_polka <= R_2*인 ‘R_polka 라운드’에서 발생해야한다. 둘째로, 검증인들은 자신들이 로킹되어 있는 블록을 제안(Propose)하거나 아니면 사전투표(PreVote)하거나 또는 둘을 함께 해야한다. 이를 통해 검증인은 충분한 증거 없이 프리커밋하지 못하며, 이미 프리커밋한 검증인의 경우 다른 블록을 동시에 프리커밋할 수 없게 된다. 이렇게 합의 알고리즘의 안전성(safety)과 라이브성(liveness)을 보장하게 된다. - -프로토콜의 전체 세부내용은 [여기](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm)를 참조 - -### 텐더민트 라이트 클라이언트 (Tendermint Light Clients) - -대안 체인(포크)의 존재 덕분에 본딩된(bonded) 지분을 ⅓+이 대폭 삭감할(slashed) 수 있기 때문에, 텐더민트 PoS 에서는 모든 블록 헤더들을 동기화(sync)할 필요가 없다. 물론 이러한 삭감을 위해서는 누군가가 포킹의 증거를 공유해야 하기 때문에, 라이트 클라이언트는 전파받는 모든 블록해시 커밋을 저장해야 한다. 뿐만 아니라, 라이트 클라이언트들은 ‘검증인 세트의 변경’과 주기적으로 동기화하여 [장기 공격(long range attacks)](#preventing-long-range-attacks)을 피할 수 있을 것이다(물론 다른 방법도 적용가능). - -이더리움과 유사한 의도로, 텐더민트는 애플리케이션들이 각 블록에 ‘전역 머클 루트 해시(global Merkle root hash)’를 임베딩(embed)하도록 하고, 계정 잔고, 컨트랙트내 저장 값, UTXO 등 각 어플리케이션에 적합한 쉽게 검증이 가능한 ‘상태 쿼리(state queries)’를 이용하도록 한다. - -### 장기 공격 방지 (Preventing Long Range Attacks) - -‘충분한 회복력(resilient)의 전파 네트워크 집합’과 ‘정적(static) 검증인 세트’가 있다면, 블록체인의 모든 포킹이 검출될 수 있으며 문제가 되는 검증인들의 보증금을 대폭 삭감시킬 수 있다. 2014 년 초에 비탈릭 뷰터린(Vitalik Buterin)은 처음 지분증명 암호화폐들의 ‘무처벌(nothing-at-stake)’ 문제에 대한 해결책을 제시한다([관련 연구](#related-work) 참조). 그러나 검증인 세트들이 항상 고정되어 있는 것이 아니기 때문에 오랜 기간에 걸쳐 원래의 검증인들은 모두 언본딩 될 수 있고(may all become unbonded) 묶어둔 보증금이 사라지면서, 어떤 비용도 치르지 않고 제네시스 블록으로부터 새로운 체인을 자유로이 생성할 수 있게 된다. 이 공격은 단기 공격(Short Range Attack)과 대조되는 개념으로 ‘장기 공격(Long Range Attack)’으로 알려지게 되었다. 텐더민트 합의와 같은 ‘포크 책임 BFT 알고리즘(folk-accountable BFT algorithm)’이 존재한다면 현재 본딩된 검증인들이 포크를 초래하는 단기공격은 처벌이 가능하다. 반면 장기 공격은 지분증명 방식에 치명적인 공격으로 간주된다. - -다행히 ‘장기공격(LRA)’을 부분적으로 방지하는 것이 가능하다. 첫째로, 검증인이 보증금을 언본딩하여 보증금을 회수하고 동시에 더이상의 합의참여 수수료를 포기하기 위해서는 정해진 일정 기간, 즉 ‘언본딩 기간(unbonding period)’을 채워야 한다. 그 기간은 수주 또는 수개월이 된다. 둘째로, 라이트 클라이언트는 최초로 네트워크에 연결될 때 신뢰할 수 있는 한개 이상의 출처를 통해 네트워크의 최근 블록해시를 검증해야 한다. 이러한 조건을 ‘약한 주체성(weak subjectivity)’으로 부르기도 한다. 마지막으로, 보안성을 유지하기 위해서 적어도 매 ‘언본딩 기간’을 주기로 ‘최종 검증인 세트’와 동기화되어야 한다. 이를 통해, 라이트 클라이언트는 검증인이 보증금을 빼내고 ‘무보증 상태(Nothing at Stake)’가 되었는지 여부를 파악할 수 있게 된다. 이를 파악하지 못한다면, 검증인은 최초 본딩이 시작된 블록으로 돌아가 새로우 블록들을 다시 생성함으로 장기공격을 감행할 수 있다(물론 이를 위해서는 검증인이 충분히 많은 초기 개인키를 가지고 있어야 함). - -이러한 방식의 LRA 대응 방안은 기존 ‘작업증명(Proof-of-Work)’ 방식의 보안 기법에 대한 완전한 분해정비를 필요로 한다. ‘작업증명’에서는, 라이트 클라이언트가 언제든 전체 블록헤더의 작업증명을 처리하기만 한다면 신뢰할 수 있는 제네시스 블록을 통해 최신 블록체인을 동기화할 수 있다. 그러나 LRA 에 대응하기 위해서는, 라이트 클라이언트가 정기적으로 온라인 상태가 되어 검증인 세트를 추적해야하며, 최초 온라인이 되었을 때 반드시 신뢰할 만한 출처를 통해 네트워크에서 제공하는 정보의 진실성을 확인해야만 한다. 물론 신뢰할만한 출처를 이용하는 방법은 비트코인에서도 동일한데, 프로토콜이나 소프트웨어를 반드시 신뢰할 수 있는 출처에서 얻어야 하기 때문이다. - -LRA 를 방지하는 상기의 방법은, 검증인들과 텐더민트 기반 블록체인의 완전 노드들에 적합하다. 이 노드들은 네트워크에 계속 연결되어 있기 때문이다. 또한 이 방법은 네트워크에 자주 연결될 수 있는 라이트 클라이언트들에도 적합하다. 하지만 라이트 클라이언트들이 자주 연결될 수 없는 경우, 다른 방법이 사용될 수 있다. 검증인이 아닌 토큰 보유자들이 상당히 긴 기간동안 자신의 토큰을 담보로 제공하는 것이다. 그리고 라이트 클라이언트들에게 이차적인 방법으로 최근과 이전의 블록해시 값을 전달하는 것이다. 이러한 토큰들이 블록체인 합의와 관련된 보안에는 영향을 미치지 않으나 라이트 클라이언트에게는 강력한 보장을 제공할 수 있다. 역사적 블록-해시 질의가 이더리움에서 지원될 경우, 누구든지 특별 설계된 스마트 컨트랙트에서 자신의 토큰을 본딩하고 지불을 위한 인증 서비스(attestation services for pay)를 제공하여 라이트 클라이언트 LRA 보안 시장을 효과적으로 창출할 수 있을 것이다. - -### 포크 및 검열 공격 극복 (Overcoming Forks and Censorship Attacks) - -블록 커밋의 정의로 인해, ⅓+ 투표권 연합은 오프라인이 되거나 투표전파를 거부함으로 블록체인을 중단시킬 수 있다. 그러한 연합을 통해 특정한 이체를 포함한 블록을 검열하여 탈락시킬 수도 있으며, 이 경우 상당한 양의 프로포잘이 거부되고 블록커밋 속도를 늦추어 블록체인의 효용과 가치를 낮출수도 있다. 또한 연합은 소량의 투표만을 전파하여 블록체인의 블록커밋 속도를 서서히 늦출 수도 있다. 결정적으로 이들은 이중서명이나 잠금규정을 지키지 않음으로 블록체인을 분기(포킹)시킬 수도 있다. - -전역적 능동 공격자(global active adversary)가 관여할 경우, 네트워크를 분할시키고 다른 이가 책임이 있는 것처럼 보이게 할 수도 있다. 이는 텐더민트만의 문제는 아니며 네트워크 상에 능동 공격자가 존재하는 모든 합의 프로토콜이 가진 한계이다. - -이러한 유형의 공격이 가능해지려면, 발리데이터 부분집합(a subset of the validators)이 합동하여 외부수단을 이용해 포킹 목적의 블록재조정 프로포잘에 서명해야하고, 초기 발리데이터 부분집합도 이에 서명해야한다. 클라이언트들은 블록재조정 프로포잘의 서명을 유심히 검증하고, 이를 근거로 판단하거나 그 결정을 최종사용자에게 넘길 수 있다. - -어떤 ‘비동시성 비잔틴 장애 저항 알고리즘(non-synchronous Byzantine fault-tolerant algorithm)’도 투표권의 ⅓+이 부정직할 경우, 합의에 도달할 수 없다. 하지만, 포크의 경우 투표권의 ⅓+이 이미 이중 서명이나 잠금 변경(lock-changing)에 의해 부정직한 것으로 가정한다. 따라서 블록재조정 프로포잘(reorg-proposal)에 서명하는 것은 어떤 비동시성 프로토콜에서도(즉, 자동적으로 그리고 기반 네트워크의 신뢰성에 관한 가정없이는) 해결할 수 없는 조정 문제이다. 현재로서는 ‘블록재조정 프로포잘 조정(reorg-proposal coordination)’ 문제를 인터넷 매체 상에서 일어나는 인간들의 사회적 합의에 맡기고 있다. 상충되는 두 블록재조정 프로포잘에 서명하는 것을 방지하기 위해, 검증인들은 서명에 앞서 남은 네트워크 분할이 없도록 주의해야 한다. - -외부의 조정 수단 및 프로토콜이 강인하다면, 네트워크분기(포크)가 검열 공격보다는 덜 치명적이다. - -⅓ 비잔틴 투표권을 요구하는 포크 및 검열 외에, +⅔ 투표권의 연합이 임의의 무효 상태를 커밋할 수도 있다. 이는 모든 BFT 합의 시스템의 특징이다. 쉽게 검증 가능한 포크가 생성되는 이중서명과는 달리, 무효 상태 커밋(commitment of an invalid state)을 검증하기 위해서는 발리데이터가 아닌 피어들(non-validating peers)도 전체 블록들을 검증해야한다. 이는 피어들이 상태의 지역 사본(local copy)을 가지고 각 이체를 실행하며, 상태 루트(state root)를 독립적으로 계산한다는 것을 의미한다. 또한 무효상태커밋이 검출되면, 사회적 합의만이 이를 처리하는 유일한 방법이다. 예를 들어, (2013 년 3 월처럼) 소프트웨어 버그로 인한 분기이든, 또는 (2015 년 7 월처럼) 채굴자들의 비잔틴 행동으로 인한 무효 상태 커밋이든 비트코인이 실패한 상황들에서, 비즈니스, 개발자, 채굴자 및 그 밖의 조직들의 잘 연결된 커뮤니티는 네트워크 회복을 위해서 참여자들의 무엇을 해야하는지에 대한 사회적 합의를 확립했다. 텐더민트의 경우, 블록체인의 검증인들이 식별될 수 있을 것으로 예상되기 때문에, 필요한 경우, 무효 상태의 커밋을 법률이나 어떤 외부 관할권을 통해 처벌하는 것까지도 가능할 수 있다. - -### TMSP 명세 (TMSP specification) - -TMSP 는 코어에서 애플리케이션으로 전달되는 3 가지 주요 메시지 유형으로 구성된다. 애플리케이션은 대응하는 응답 메시지로 응답한다. - -AppendTx 메시지는 애플리케이션의 핵심 작업수단이다. 블록체인에서의 이체는 이 메시지와 함께 전달된다. 애플리케이션은 AppendTx 메시지와 함께 수신된 각 이체를 현재의 상태, 애플리케이션 프로토콜, 이체의 암호학적 요건 등과 비교해서 검증한다. 이후, 검증된 이체는 값을 키-값 저장장치(key values store)으로 결합하거나 UTXO 데이터베이스를 업데이트하여 애플리케이션 상태를 업데이트해야 한다. - -CheckTx 메시지는 AppendTx 와 유사하지만 이체 검증에만 사용된다. 텐더민트 코어의 mempool(메모리 풀)은 CheckTx 로 이체의 유효성을 검사하고 유효한 이체만 피어들에 전파한다. 애플리케이션들은 이체의 증가 논스 (incrementing nonce)를 검사하고 논스가 오래된 경우, CheckTx 에 오류를 반환할 수도 있다. - -Commit 메시지는 다음 블록 헤더에 포함될 현재 애플리케이션 상태에 대한 암호학적 커밋(cryptographic commitment)을 계산하기 위해 사용된다. 이를 통해 몇가지를 얻을 수 있다. 해당 상태 업데이트에서 불일치들이 포크의 형태로 나타나게 된다. 또한 머클-해시 증명들이 블록-해시와의 비교를 통해 검증될 수 있고 블록-해시는 (투표권에 의한) 검증인 정족수에 의해 서명되기 때문에, 단순한 방식으로 안전한 라이트 클라이언트 개발하는데 도움이 된다. - -추가적 TMSP 메시지를 통해 애플리케이션이 검증인 세트를 추적(keep track of) 및 변경하고, 높이와 커밋 투표 같은 블록 정보를 수신할 수 있다. - -TMSP 요구/응답(TMSP requests/responses)은 간단한 Protobuf 메시지들이다. [스키마 파일(schema file)](https://github.com/tendermint/abci/blob/master/types/types.proto) 확인 - -##### AppendTx - -- **Arguments**: - - `Data ([]byte)`: The request transaction bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: Result bytes, if any - - `Log (string)`: Debug or error message -- **사용법**:<br/> - 이체를 덧붙이고 실행한다. 이체가 유효할 경우, CodeType.OK 를 반환한다. - -##### CheckTx - -- **Arguments**: - - `Data ([]byte)`: The request transaction bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: Result bytes, if any - - `Log (string)`: Debug or error message -- **사용법**:<br/> - 이체를 검증한다. 이 메시지가 상태를 변경(mutate)해서는 안 된다. 이체들은 mempool 계층의 피어들에게 전파되기 전, CheckTx 를 통해 먼저 실행된다. 동일 블록 내 이체의 발생 순서를 고려하기 위해 CheckTx 를 semi-stateful(준-상태기반)로 만들고 Commit 또는 BeginBlock 시에 상태를 클리어(clear) 할 수 있다. - -##### 커밋 (Commit) - -- **Returns**: - - `Data ([]byte)`: The Merkle root hash - - `Log (string)`: Debug or error message -- **사용법**:<br/> - 애플리케이션 상태의 머클 루트 해시를 반환한다. - -##### Query - -- **Arguments**: - - `Data ([]byte)`: The query request bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: The query response bytes - - `Log (string)`: Debug or error message - -##### Flush - -- **사용법**:<br/> - 응답 대기행렬(response queue)을 플러시(flush)한다. types.Application 을 구현하는 애플리케이션들은 이 메시지를 구현할 필요가 없다--프로젝트에 의해 처리되기 때문이다. - -##### Info - -- **Returns**: - - `Data ([]byte)`: The info bytes -- **사용법**:<br/> - 애플리케이션 상태에 관한 정보를 반환한다. 애플리케이션 특정적이다. - -##### SetOption - -- **Arguments**: - - `Key (string)`: Key to set - - `Value (string)`: Value to set for key -- **Returns**: - - `Log (string)`: Debug or error message -- **사용법**:<br/> - 애플리케이션 옵션을 설정한다. 예를 들어, mempool 연결을 위해서는 Key='mode', Value='mempool', 또는 합의 연결을 위해서는 Key='mode', Value='consensus'. 다른 옵션들은 각 애플리케이션마다 다르다. - -##### InitChain - -- **Arguments**: - - `Validators ([]Validator)`: Initial genesis-validators -- **사용법**:<br/> - 제네시스 시, 1 회 호출됨. - -##### BeginBlock - -- **Arguments**: - - `Height (uint64)`: The block height that is starting -- **사용법**:<br/> - 새 블록의 시작을 알린다. 모든 AppendTx 에 앞서 호출됨. - -##### EndBlock - -- **Arguments**: - - `Height (uint64)`: The block height that ended -- **Returns**: - - `Validators ([]Validator)`: Changed validators with new voting powers (0 - to remove) -- **사용법**:<br/> - 블록의 끝을 알린다. 이체 후, 각 커밋에 앞서 호출됨. - -상세한 내용은 [TMSP 리포지토리(TMSP repository)](https://github.com/tendermint/abci) 참조. - -### IBC 패킷 전송 확인응답 (IBC Packet Delivery Acknowledgement) - -발신자가 수신 체인의 패킷전송 확인응답을 원할 여러 가지 경우가 존재한다. 예를 들어 장애가 있을 경우, 송신자는 수신 체인의 상태를 모를 수 있다. 또는 송신자는 (MaxHeight 패킷 필드로) 패킷에 타임아웃을 부여하기를 원하는 반면, 수신 체인이 갑작스런 수신 패킷 수 급증에 의한 서비스 거부(denial-of-service) 공격을 받을 수도 있다. - -이런 경우, 송신자는 초기 패킷 상태를 AckPending 으로 설정하여 전송 확인응답을 요구할 수 있다. 그러면 앱 머클 해시에 abbreviatedIBCPacket 을 포함하여 전송을 확인해 주는 것이 수신 체인의 책임이다. - -![Figure of Zone1, Zone2, and Hub IBC with acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack.png) - -먼저, 'Zone 1'에 IBCPacket 이 존재함을 증명하는 '허브'가 있다. 그리고 이 허브 위에 IBCBlockCommit 과 IBCPacketTx 가 포스팅된다. 예를 들어 IBCPacketTx 는 다음과 같은 값을 갖는다: - -- `FromChainID`: "Zone1" -- `FromBlockHeight`: 100 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 (say) - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 (say "Hub" is currently at height 300) - - `Payload`: <The bytes of a "coin" payload> - -다음으로, '허브'에 IBCPacket 이 존재함을 증명하는 'Zone 2' 위에 IBCBlockCommit 과 IBCPacketTx 가 포스팅된다. 예를 들면, IBCPacketTx 는 다음과 같다: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 300 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 - - `Payload`: <The same bytes of a "coin" payload> - -그 다음, 'Zone 2'는 앱-해시(app-hash)에 새로운 상태의 AckSent 를 보이는 생략형 패킷을 반드시 포함해야 한다. 생략형 IBCPacket 이 ‘Zone 2' 위에 존재함을 증명하는 '허브' 위에 IBCBlockCommit 와 IBCPacketTx 가 다시 포스팅된다. 예를 들면 IBCPacketTx 는 다음과 같다: - -- `FromChainID`: "Zone2" -- `FromBlockHeight`: 400 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckSent` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <The hash bytes of the same "coin" payload> - -끝으로, '허브'는 패킷의 상태를 AckPending 에서 AckReceived 로 업데이트 해야 한다. 이 새로이 완결된 상태의 증거가 'Zone 2'로 되돌아간다. 예를 들면, IBCPacketTx 는 다음과 같은 값을 갖는다: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 301 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckReceived` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <The hash bytes of the same "coin" payload> - -한편, 'Zone 1'은 ‘코인’패킷 전송의 실패가 '허브' 상에서 증명되지 않는 한, 성공적으로 전송될 것이라고 가정할 것이다. 위의 예에서, '허브'는 ‘블록 350’를 통해 'Zone 2'의 AckSent 상태를 수신하지 않는다면, 상태를 타임아웃(Timeout)으로 자동 설정했을 것이다. 이 타임아웃 증거가 'Zone 1'에 다시 포스팅 될 수 있고, 모든 토큰은 반환될 수 있다. - -![Figure of Zone1, Zone2, and Hub IBC with acknowledgement and timeout](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack_timeout.png) - -### 머클 트리와 증명 명세 (Merkle Tree & Proof Specification) - -두 가지의 머클 트리 유형이 텐더민트/코스모스 생태계에서 지원된다: 단순 트리(Simple Tree)와 IAVL+ 트리. - -#### 단순 트리 (Simple Tree) - -단순트리는 요소들(elements)의 정적 리스트(static list)를 위한 머클 트리이다. 항목의 수가 2 의 거듭제곱이 아닐 경우, 잎들이 서로 다른 수준에 위치할 수 있다. 단순 트리는 트리의 양측면을 동일한 높이로 유지하고자 하지만, 좌측이 더 클 수도 있다. 이 머클 트리는 블록의 이체들을 그리고 애플리케이션 상태 루트의 최상위 요소들을 머클화 하기 위해 사용된다. - - * - / \ - / \ - / \ - / \ - * * - / \ / \ - / \ / \ - / \ / \ - * * * h6 - / \ / \ / \ - h0 h1 h2 h3 h4 h5 - - 7요소의 단순트리 - -#### IAVL+ Tree - -IAVL+ 데이터 구조의 목적은 애플리케이션 상태의 키-값 쌍들에 영구 저장(persistent storage)을 제공하는 것이다. 그렇게 하여 결정론적 머클 루트 해시를 효율적으로 계산한다. 트리는 [AVL 알고리즘](https://en.wikipedia.org/wiki/AVL_tree)의 변형(variant)을 사용하여 밸런싱되며(balancing), 모든 연산(operations)은 O(log(n))이다. - -AVL 트리에서는 임의 노드의 2 개 ‘자식 서브트리(child subtree)’의 높이가 최대 1 차이가 난다. 업데이트 시, 이 조건이 위반될 때마다 트리는 이전 트리의 비수정 노드들을 가리키는 O(log(n))의 새 노드들을 만들어서 다시 균형을 잡는다. 기존 AVL 알고리즘에서는 내부 노드들이 키-값 쌍들도 보유할 수 있다. AVL+ 알고리즘(플러스에 주의한다)은 잎 노드 상의 모든 값을 유지하기 위해 AVL 알고리즘을 수정하며, 가지 노드들만 사용하여 키를 저장한다. 이를 통해 알고리즘을 단순화하고, 머클 해시 트레일(merkle hash trail)을 짧게 유지한다. - -AVL+ 트리는 이더리움의 [패트리샤 트라이(Patricia tries)](https://en.wikipedia.org/wiki/Radix_tree)와 유사하다. 장단점이 존재한다. 키들은 IAVL+ 트리에 삽입되기 전에 해싱될 필요가 없고, 따라서 키 공간(key space)에서의 빠른 순서반복(ordered iteration)을 제공한다. 로직의 구현이 간편하고 내부 노드(inner nodes)와 잎 노드(leaf nodes)만 필요로 한다. 머클 증명은 일반적으로 짧고 균형 잡힌 이진 트리이다. 반면에 IAVL+ 트리의 머클 루트는 업데이트의 순서(order of updates)에 의존한다. - -이진 변형(binary variant) 사용이 가능해지면, 이더리움의 패트리샤 트라이 같은 효율적 머클 트리들을 추가로 지원할 것이다. - -### 이체 유형 (Transaction Types) - -일반적으로, 이체들은 TMSP 인터페이스를 통해 코스모스 허브 애플리케이션으로 흘러간다. - -코스모스 허브는 앞으로 SendTx, BondTx, UnbondTx, ReportHackTx, SlashTx, BurnAtomTx, ProposalCreateTx, ProposalVoteTx 등을 포함한 많은 주요 이체 유형들(primary transaction types)을 추가할 것이고 문서화할 것이다. 여기서는 IBC 의 두 주요 이체 유형인 IBCBlockCommitTx 와 IBCPacketTx 에 대해 설명한다. - -#### IBCBlockCommitTx - -IBCBlockCommitTx 이체는 다음으로 구성된다: - -- `ChainID (string)`: The ID of the blockchain -- `BlockHash ([]byte)`: The block-hash bytes, the Merkle root which includes the - app-hash -- `BlockPartsHeader (PartSetHeader)`: The block part-set header bytes, only - needed to verify vote signatures -- `BlockHeight (int)`: The height of the commit -- `BlockRound (int)`: The round of the commit -- `Commit ([]Vote)`: The +⅔ Tendermint `Precommit` votes that comprise a block - commit -- `ValidatorsHash ([]byte)`: A Merkle-tree root hash of the new validator set -- `ValidatorsHashProof (SimpleProof)`: A SimpleTree Merkle-proof for proving the - `ValidatorsHash` against the `BlockHash` -- `AppHash ([]byte)`: A IAVLTree Merkle-tree root hash of the application state -- `AppHashProof (SimpleProof)`: A SimpleTree Merkle-proof for proving the - `AppHash` against the `BlockHash` - -#### IBCPacketTx - -IBCPacket 은 다음으로 구성된다: - -- `Header (IBCPacketHeader)`: The packet header -- `Payload ([]byte)`: The bytes of the packet payload. _Optional_ -- `PayloadHash ([]byte)`: The hash for the bytes of the packet. _Optional_ - -Payload 또는 PayloadHash 가 반드시 존재해야 한다. IBCPacket 의 해시는 Header 와 Payload 의 단순 머클 루트이다. 전체 페이로드(full payload)가 없는 IBCPacket 은 _생략형 패킷_(_abbreviated packet_)이라고 부른다. - -IBCPacketHeader 는 다음으로 구성된다. - -- `SrcChainID (string)`: The source blockchain ID -- `DstChainID (string)`: The destination blockchain ID -- `Number (int)`: A unique number for all packets -- `Status (enum)`: Can be one of `AckPending`, `AckSent`, `AckReceived`, - `NoAck`, or `Timeout` -- `Type (string)`: The types are application-dependent. Cosmos reserves the - "coin" packet type -- `MaxHeight (int)`: If status is not `NoAckWanted` or `AckReceived` by this - height, status becomes `Timeout`. _Optional_ - -IBCPacketTx 이체는 다음으로 구성된다: - -- `FromChainID (string)`: The ID of the blockchain which is providing this - packet; not necessarily the source -- `FromBlockHeight (int)`: The blockchain height in which the following packet - is included (Merkle-ized) in the block-hash of the source chain -- `Packet (IBCPacket)`: A packet of data, whose status may be one of - `AckPending`, `AckSent`, `AckReceived`, `NoAck`, or `Timeout` -- `PacketProof (IAVLProof)`: A IAVLTree Merkle-proof for proving the packet's - hash against the `AppHash` of the source chain at given height - -'허브'를 통해 '존 1'에서 '존 2'로 패킷을 송신하기 위한 순서가 {그림 X}에 도시되어 있다. 먼저, IBCPacketTx 가 패킷이 '존 1'의 앱-상태(app-state)에 포함되어 있음을 '허브'에 증명한다. 그 다음에 또 다른 IBCPacketTx 가 패킷이 '허브'의 앱-상태에 포함되어 있음을 '존 2'에 증명한다. 이 절차 동안, IBCPacket 필드들은 동일하다: SrcChainID 는 언제나 'Zone1(존 1)'이고 DstChainID 는 언제나 'Zone2(존 2)'이다. - -PacketProof 은 반드시 다음과 같은 올바른 머클 증명 경로를 포함해야 한다: - - IBC/<SrcChainID>/<DstChainID>/<Number> - -'존 1'이 '허브'를 통해 '존 2'로 패킷을 전송할 경우, 패킷이 '존 1', '허브' 또는 '존 2'의 어디에서 머클화 되든지 상관 없이 IBCPacket 데이터는 동일하다. 변경 가능한 유일한 필드는 전송 추적을 위한 Status(상태)이다. - -## 감사의 글 (Acknowledgements) - -We thank our friends and peers for assistance in conceptualizing, reviewing, and -providing support for our work with Tendermint and Cosmos. - -- [Zaki Manian](https://github.com/zmanian) of - [SkuChain](http://www.skuchain.com/) provided much help in formatting and - wording, especially under the TMSP section -- [Jehan Tremback](https://github.com/jtremback) of Althea and Dustin Byington - for helping with initial iterations -- [Andrew Miller](https://soc1024.com/) of [Honey - Badger](https://eprint.iacr.org/2016/199) for feedback on consensus -- [Greg Slepak](https://fixingtao.com/) for feedback on consensus and wording -- Also thanks to [Bill Gleim](https://github.com/gleim) and [Seunghwan - Han](http://www.seunghwanhan.com) for various contributions. -- **Your name and organization here for your contribution** - -## 인용 (Citations) - -- [1] Bitcoin: <https://bitcoin.org/bitcoin.pdf> -- [2] ZeroCash: <http://zerocash-project.org/paper> -- [3] Ethereum: <https://github.com/ethereum/wiki/wiki/White-Paper> -- [4] TheDAO: <https://download.slock.it/public/DAO/WhitePaper.pdf> -- [5] Segregated Witness: <https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki> -- [6] BitcoinNG: <https://arxiv.org/pdf/1510.02037v2.pdf> -- [7] Lightning Network: <https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf> -- [8] Tendermint: <https://github.com/tendermint/tendermint/wiki> -- [9] FLP Impossibility: <https://groups.csail.mit.edu/tds/papers/Lynch/jacm85.pdf> -- [10] Slasher: <https://blog.ethereum.org/2014/01/15/slasher-a-punitive-proof-of-stake-algorithm/> -- [11] PBFT: <http://pmg.csail.mit.edu/papers/osdi99.pdf> -- [12] BitShares: <https://bitshares.org/technology/delegated-proof-of-stake-consensus/> -- [13] Stellar: <https://www.stellar.org/papers/stellar-consensus-protocol.pdf> -- [14] Interledger: <https://interledger.org/rfcs/0001-interledger-architecture/> -- [15] Sidechains: <https://blockstream.com/sidechains.pdf> -- [16] Casper: <https://blog.ethereum.org/2015/08/01/introducing-casper-friendly-ghost/> -- [17] TMSP: <https://github.com/tendermint/abci> -- [18] Ethereum Sharding: <https://github.com/ethereum/EIPs/issues/53> -- [19] LibSwift: <http://www.ds.ewi.tudelft.nl/fileadmin/pds/papers/PerformanceAnalysisOfLibswift.pdf> -- [20] DLS: <http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf> -- [21] Thin Client Security: <https://en.bitcoin.it/wiki/Thin_Client_Security> -- [22] Ethereum 2.0 Mauve Paper: <https://cdn.hackaday.io/files/10879465447136/Mauve%20Paper%20Vitalik.pdf> - -#### 기타 링크 (Unsorted links) - -- [https://www.docdroid.net/ec7xGzs/314477721-ethereum-platform-review-opportunities-and-challenges-for-private-and-consortium-blockchains.pdf] diff --git a/docs/resources/whitepaper-pt.md b/docs/resources/whitepaper-pt.md deleted file mode 100644 index aff8c86ff..000000000 --- a/docs/resources/whitepaper-pt.md +++ /dev/null @@ -1,1482 +0,0 @@ -# Cosmos - -Uma Rede de Distribuição de Ledgers - -Jae Kwon jae@tendermint.com<br/> -Ethan Buchman ethan@tendermint.com - -Para discussões, [entre no nosso Matrix](https://riot.im/app/#/room/#cosmos:matrix.org)! - -_NOTA: Se você pode ler isso no GitHub, então ainda estamos desenvolvendo este documento ativamente. Por favor, cheque regularmente as atualizações!_ - -\[[toc]] - -O sucesso combinado do ecossistema de código aberto, compartilhamento -de arquivos descentralizado e criptomoedas públicas tem inspirado um conhecimento sobre -protocolos descentralizados na Internet que podem ser utilizados para melhorar radicalmente -a infraestrutura. Vimos aplicações de blockchain especializadas como Bitcoin -[\[1\]][1] (uma criptomoeda), Zerocash [\[2\]][2] (uma criptomoeda para privacidade -), and generalized smart contract platforms such as Ethereum [\[3\]][3], -com inúmeras aplicações distribuídas para a Etherium Virtual Machine (EVM), como Augur (uma previsão -de mercado) e TheDAO [\[4\]][4] (um clube de investimento). - -Contudo, até à data, estas blockchains sofreram uma série de inconvenientes, -incluindo sua ineficiência energética, desempenho fraco ou limitado e -mecanismos de governança imaturos. Propostas de escala -de processamento de transações da Bitcoin, como Testemunhas Separadas [\[5\]][5] e -BitcoinNG [\[6\]][6], soluções de escalonamento vertical que permanecem -limitadas pela capacidade de uma única máquina física, a fim de -proporcionar uma auditabilidade completa. A Rede Lightning [\[7\]][7] pode ajudar -o Bitcoin no quesito volume de transações, deixando algumas transações completamente -fora da carteira, e é bem adequado para micropagamentos e preservando a privacisadade por pagamentos -Rails, mas pode não ser adequado para necessidades de escala mais abrangente. - -Uma solução ideal é a de permitir blockchains paralelos múltiplos para -interoperação, mantendo suas propriedades de segurança. Isto provou -ser difícil, se não impossível, com prova de trabalho. A mineração combinada, por exemplo, -permite que o trabalho feito para proteger uma blockchain mãe seja reutilizado em uma blockchain nova, -mas as transações ainda devem ser validadas, em ordem, por cada nó, e uma -blockchain Merge-mined é vulnerável a ataques se a maioria do poder de -hashing sobre a mãe não é ativamente merge-mined da nova. Uma revisão acadêmica -do [arquiteturas de redes alternativas blockchain -](http://vukolic.com/iNetSec_2015.pdf) é fornecida para -contextualizar, e fornecemos resumos de outras propostas e suas desvantagens em -[Trabalho relatado](#trabalho-relatado). - -Nesse relato nós apresentamos a Cosmos, uma novela da arquitetura de rede blockchain que aborda todos -esses problemas. Cosmos é uma rede de muitos blockchains independentes, chamados -Zonas. As zonas são alimentadas pelo Tendermint Coreork [\[8\]][8], que fornece uma -alta performace, consistência, segurança -[PBFT-como](https://blog.cosmos.network/tendermint-vs-pbft-12e9f294c9ab?gi=20a63f2a00ee) um mecanismo de consenso -rigoroso, onde [fork-responsável](#fork-responsável) tem-se garantias de deter -comportamentos maliciosos. O algoritmo de consenso BFT do Tendermint Core é -bem adaptado para integrar blockchains públicas de prova de estaca. - -A primeira zona na Cosmos é chamada de Cosmos Hub. A Cosmos Hub é uma criptomoeda -multi-asset de prova de estaca com um simples mecanismo de governança -o qual permite a rede se adaptar e atualizar. Além disso, a Cosmos Hub pode ser -extendida por conexão com outras zonas. - -O hub e as zonas da rede Cosmos comunicam-se uma com a outra através de um -protocolo de comunicação Inter-blockchain (IBC), um tipo de UDP ou TCP virtual para -blockchains. Os tokens podem ser transferidos de uma zona para outra com segurança e -rapidez sem necessidade de liquidez cambial entre as zonas. Em vez disso, todas -as transferências de tokens inter-zonas passam pelo Hub Cosmos, que mantêm -a quantidade total de tokens detidas por cada zona. O hub isola cada zona da -falha das outras zonas. Porque qualquer um pode conectar uma nova zona no Hub Cosmos, -o que permite futuras compatibilidades com novas blockchains inovadoras. - -## Tendermint - -Nesta seção, descrevemos o protocolo de consenso da Tendermint e a interface -usada para construir aplicações através dele. Para mais detalhes, consulte o [apêndice](#apêndice). - -### Validadores - -No algorítimo de tolerância e falhas clássicas Bizantinas (BFT), cada node tem o mesmo -peso. Na Tendermint, nodes tem uma quantidade positiva de _poder de voto_, e -esses nodes que tem poder de voto positivo são chamados de _validadores_. Validadores -participam de um protocolo de consenso por transmissão de assinaturas criptográficas, -ou _votos_, para concordar com o próximo bloco. - -Os poderes de voto dos validadores são determinados na gênese, ou são alterados -de acordo com a blockchain, dependendo da aplicação. Por exemplo, -em uma aplicação de prova de participação, como o Hub da Cosmos, o poder de voto pode ser -determinado pela quantidade de tokens usados como garantia. - -_NOTA: Frações como ⅔ e ⅓ referem-se a frações do total de votos, -nunca o número total de validadores, a menos que todos os validadores tenham -peso._ -_NOTE: +⅔ significa "mais do que ⅔", enquanto ⅓+ significa "⅓ ou mais"._ - -### Consenso - -Tendermint é um protocolo de consenso BFT parcialmente sincronizado e derivado do -algoritmo de consenso DLS [\[20\]][20]. Tendermint é notável por sua simplicidade, -desempenho, e [fork-responsável](#fork-responsável). O protocolo -requer um grupo determinado de validadores, onde cada validador é identificado por -sua chave pública. Validadores chegarão a um consenso em um bloco por vez, -onde um bloco é uma lista de transações. A votação para o consenso sobre um bloco -acontece por rodada. Cada rodada tem uma líder-de-rodada, ou proponente, que propõe um bloco. Os -validadores, em seguida, votam, por etapas, sobre a aceitação do bloco proposto -ou passam para a próxima rodada. O proponente de uma rodada é escolhido -de acordo com uma lista ordenada de validadores, proporcionalmente à seu -poder de voto. - -Os detalhes completos do protocolo estão descritos -[aqui](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - -A segurança da Tendermint é baseada na tolerância e falhas clássicas Bizantinas ótimas -através de super-maioria (+⅔) e um mecanismo de bloqueio. Juntas, elas garantem -isso: - -- ⅓+ o poder de voto deve ser bizantino devido a violações de segurança, onde mais -   que dois valores são comprometidos. -- se algum conjunto de validadores tiver sucesso em violar a segurança, ou mesmo tentarem - para isso, eles podem ser identificados pelo protocolo. Isso inclui tanto o voto - para blocos conflitantes quanto a transmissão de votos injustificados. - -Apesar de suas fortes garantias, a Tendermint oferece um desempenho excepcional. Dentro -de Benchmarks de 64 nós distribuídos em 7 datacenters em 5 continentes, em -nuvens de commodities, o consenso da Tendermint pode processar milhares de -transações por segundo, com tempo de resposta entre um a dois -segundos. Notavelmente, o desempenho muito além de mil transações por segundo -é mantido mesmo em condições adversas, com validadores falhando ou -combinando votos maliciosamente. Veja a figura abaixo para mais detalhes. - -![Figura do desempenho da Tendermint](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/tendermint_throughput_blocksize.png) - -### Clientes Light - -O principal benefício do algoritmo de consenso da Tendermint é um cliente leve e simplificado -de segurança, tornando-o um candidato ideal para o uso de dispositivos móveis e casos de uso na -internet. Enquanto um cliente leve do Bitcoin deve sincronizar blockchains e encontrar -o que tem mais prova de trabalho, os clientes light da Tendermint precisa apenas -das alterações feitas pelo conjunto dos validadores, em seguida, verifica-se o +⅔ PreCommits -no último bloco para determinar o estado atual. - -Provas claras e sucintas do cliente também permite [comunicação-inter- -blockchain](#comunicação-inter-blockchain-ibc). - -### Previnindo ataques - -A Tendermint dispõe de medidas de proteção para evitar -ataques, como [gastos duplos em longa-distância-sem-estaca double -spends](#previnindo-ataques-de-longa-distância) e -[censura](#superando-forks-e-censurando-ataques). Esses são discutidos -completamente no [apêndice](#apêndice). - -### TMSP - -O algoritmo de consenso Tendermint é implementado através de um programa chamado Tendermint -Core. O Tendermint Core é um "mecanismo de consenso" independente de aplicações que -transformam qualquer aplicação blackbox em uma réplica distribuída na -Blockchain. Tendermint Core conecta-se ao blockchain -através de aplicações do Tendermint Socket Protocol (TMSP) [\[17\]][17]. Assim, o TMSP -permite que as aplicações da blockchain sejam programadas em qualquer idioma, não apenas -a linguagem de programação que o mecanismo de consenso é escrito, além disso, -o TMSP torna possível a troca fácil da camada de consenso de qualquer -tipo de blockchain. - -Nós fizemos uma analogia com a bem conhecida criptogradia do Bitcoin. Bitcoin é uma -blockchain de criptomoedas onde cada nó mantém uma Unspent totalmente auditada -e banco de dados de saída de transação (UTXO). Se alguém quisesse criar um Bitcoin-like -TMS, a Tendermint Core seria responsável por - -- Compartilhar blocos e transações entre os nós -- Estabelecer uma ordem de transações canônica/imutável (a blockchain) - -Entretanto, o aplicativo TMSP seria responsável por - -- Manter o banco de dados UTXO -- Validar a criptografia das assinaturas das transações -- Previnir transações vindas de gastos de fundos não exisentes -- Permitir aos clientes a consulta do banco de dados UTXO - -Tendermint é capaz de decompor o design da blockchain, oferecendo um simples -API entre o processo da aplicação e o processo do consenso. - -## Visão Geral da Cosmos - -Cosmos é uma rede de blockchains paralelos e independentes que são alimentadas pelo -clássico algorítimo de consenso BFT como a Tendermint -[1](https://github.com/tendermint/tendermint). - -A primeira blockchain dessa rede será a Cosmos Hub. A Cosmos Hub -conecta as outras blockchains (ou _zonas_) através do protocolo de comunicação-inter- -blockchain. A Cosmos Hub rastreia vários tipos de tokens e mantém -registo do número total de tokens em cada zona ligada. Os tokens podem ser -transferidos de uma zona para outra de forma segura e rápida, sem necessidade de -uma troca líquida entre zonas, porque todas as transferências de moedas ocorre -através da Cosmos Hub. - -Essa arquitetura resolve muitos dos problemas enfrentados atualmente pelas blockchains, -tais como interoperabilidade de aplicativos, escalabilidade e capacidade de atualização contínua. -Por exemplo, as zonas baseadas do Bitcoin, Go-Ethereum, CryptoNote, ZCash, ou qualquer -sistema blockchain pode ser ligado ao Cosmos Hub. Essas zonas permite a Cosmos -o escalonamento infinito para atender a demanda global de transações. As Zonas também são um grande -apoio para a exchange distribuída, que também serão apoiadas. - -Cosmos não é apenas uma única ledger distribuídos, o Cosmos Hub não é um -jardim cercado ou o centro do universo. Estamos elaborando um protocolo para -uma rede aberta de legers distribuídos que pode servir como um novo -futuros para sistemas financeiros, baseados em princípios de criptografia, economia -teoria de consenso, transparência e responsabilidade. - -### Tendermint-BFT - -O Cosmos Hub é a primeira blockchain pública na rede Cosmos, alimentada pelo -algoritimo de consenso BFT Tendermint. A Tendermint é um projeto de fonte aberta que -nasceu em 2014 para abordar a velocidade, a escalabilidade e as questões -do algoritimo de consenso da prova-de-trabalho do Bitcoin. Usando e melhorando -algoritmos BFT comprovados e desenvolvidos no MIT em 1988 [\[20\]][20], o time Tendermint foi o primeiro a -que demonstrou conceitualmente uma prova de estaca das criptomoedas que aborda o -problema de "sem-estaca" sofrido pelas criptomoedas da primeira geração -tais como NXT e BitShares. - -Hoje, praticamente todas carteiras móveis de Bitcoin usam servidores confiáveis, que fornece -a elas transações com verificação. Isso porque a prova-de-trabalho exige -muitas confirmações antes que uma transação possa ser considerada -irreversivel e completa. Os ataques de gasto-duplo já foram demonstrados em -serviços como a CoinBase. - -Ao contrário de outros sistemas de consenso blockchain, a Tendermint oferece -comprovação segura de pagamento para o cliente móvel. Uma vez que a Mint é -projetada para nunca passar por um fork, carteiras móveis podem receber confirmações de transações -instantâneas, o que torna os pagamentos confiáveis e práticos através de -smartphones. Isto tem implicações significativas para as aplicações da Internet. - -Validadores na Cosmos tem uma função similar aos mineiros do Bitcoin, mas usam -assinaturas criptografadas para votar. Validadores são máquinas seguras e dedicadas -que são responsáveis por validar os blocos. Os não validadores podem delegar através de seus tokens estacados -(chamados "atoms") a qualquer validador para ganhar uma parcela das taxas da blockchain -e recompensas de atoms, mas eles correm o risco de serem punidos (cortados) se o -o validador de delegados for invadido ou violar o protocolo. A segurança comprovada -garantida pelo consenso BFT da Tendermint, e o depósito de garantia das -partes interessadas - validadores e delegados - fornecem dados prováveis, -segurança para os nós e clientes light. - -### Governança - -Ledgers de distribuição pública devem ser constituídos de um sistema de governança. -O Bitcoin confia na Fundação Bitcoin e na mineração para -coordenar upgrades, mas este é um processo lento. Ethereum foi dividido em ETH e -ETC depois de hard-fork para se recuperar do hack TheDAO, em grande parte porque não havia -contrato sócial prévio, nem um mecanismo para tomar tais decisões. - -Os validadores e os delegados do Cosmos Hub podem votar propostas que -alteraram automaticamente os parâmetros predefinidos do sistema (tal como o gás limite do -bloco), coordenar upgrades, bem como votar em emendas para a -constituição que governa as políticas do Cosmos Hub. A Constituição -permite a coesão entre as partes interessadas em questões como o roubo -e bugs (como o incidente TheDAO), permitindo uma resolução mais rápida e mais limpa. - -Cada zona pode ter sua própria constituição e mecanismo de governança. -Por exemplo, o Cosmos Hub pode ter uma constituição que reforça a imutabilidade -no Hub (sem roll-backs, a não ser por bugs em implementações dos nós do Cosmos Hub), -enquanto cada zona pode ter sua própria política sobre os roll-backs. - -Ao disponibilizar a interoperabilidade em diferentes políticas das zonas, a rede Cosmos -dá aos usuários total liberdade e potenciais permissões para -experimentos. - -## O Hub e as Zonas - -Aqui nós descrevemos o modelo do roteiro de descentralização e ecalabilidade. Cosmos é uma -rede de muitas blockchains alimentadas pela Tendermint. Enquanto existirem propostas visando -criar um"blockchain solitário" com ordens de transações cheias, a Cosmos -permite que muitas blockchains rodem junto de outra enquanto mantêm a -interoperabilidade. - -Basicamente, o Cosmos Hub gerencia várias blockchains independentes chamadas "zonas" -(as vezes chamadas de "shards", em referência a técnica de escalonamento de -bando de dados conhecida como "sharding"). Uma constante transmissão de blocos recentes das -zonas atuando no Hub permite ao Hub manter o estado de cada zona atualizado. -Sendo assim, cada zona mantêm ativa o estado do Hub (mas as zonas não se mantêm ativas -com qualquer outro exceto o Hub). Pacotes de informação são -então comunicados de uma zona para outra atráves de Merkle-proofs como evidências, -essas informações são enviadas e recebidas. Esse mecanismo é chamado de -comunicação inter-blockchain, ou IBC para encurtar. - -![Figura de reconhecimento -do hub e das zonas](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/hub_and_zones.png) - -Qualquer uma das zonas podem ser hubs para formar gráficos acíclicos, mas -mas para deixar claro, nós vamos apenas descrever uma simples configuração para -um único hub, e várias zonas que não são hubs. - -### O Hub - -O Cosmos Hub é uma blockchain que hospeda um ledger de distribuíção de multi-asset, -onde os tokens podem ser mantidos por usuários individuais ou pelas próprias zonas. Esses -tokens podem ser movidos de uma zona para outra em um pacote IBC especial chamado -"coin packet". O hub é responsavel por preservar a manutenção global de -toda a quantia de cada token nas zonas. As transações de moedas no pacote IBC -precisam ser feitas pelo remetente, hub, e blockchain recebedor. - -Desde a atuação do Cosmos Hub como ledger principal para todos o -sistema, a segurança do Hub é de suma importância. Enquanto cada -zona pode ser uma blockchain Tendermint que é segurada por 4((ou talvez -menos caso o consenso BFT não seja necessário), o Hub precisa ser segurado por uma descentralização -globalizada feita pelos validadores que podem evitar os mais severos tipos de -ataques, como uma partição de rede continental ou um estado-nação fazendo -ataques. - -### As Zonas - -Uma zona Cosmos é uma blockchain independente das trocas de mensagens IBC com o -Hub. Na perspectiva do Hub, uma zona é uma conta multi-asset dynamic-membership -multi-signature que pode enviar e receber tokens usando pacotes IBC. Como -uma conta de criptomoeda, uma zona não pode transferir mais tokens do que ela possui, mas -pode receber tokens de outras que os tem. Uma zona pode ser usada como uma -"fonte" de um ou mais tipos de tokens, garantindo o poder de aumentar o estoque desse -token. - -Os atoms do Cosmos Hub podem ser estacados por validadores de uma zona conectada ao -Hub. Enquanto os ataques de gasto-duplo nesses zonas podem resultar em um core dos -atoms com o fork-responsável da Tendermint, uma zona onde +⅔ do poder de voto -são Bizantinos podem deixar o estado inválido. O Cosmos Hub não verifica ou -executa transações ocorridas em outras zonas, então essa é uma responsabilidade dos -usuários para enviar os tokes ara zonas que eles confiem. Futuramente, o sistema de -governança do Cosmos Hub irá implementar propostas para o Hub e para as falhas -das zonas. Por exemplo, um token de saída transferido para algumas (ou todas) zonas podem ser -segurados em caso de uma emergência de quebra de circuito das zonas(uma parada temporária -nas transferências dos tokens) quando um ataque é detectado. - -## Comunicação Inter-blockchain (IBC) - -Agora nós olhamos para como o Hub e as zonas vão se comunicar. Por exemplo, se -aqui são três blockchains, "Zona1", "Zona2", and "Hub", e nós queremos que a -"Zona1" produza um pacote destinado para a "Zona2" indo através do "Hub". Para mover um -pacote de uma blockchain para outra, uma prova é feita na -cadeia recebedora. A prova atesta esse envio publicado na cadeia de destino por uma alegação -de pacote. Para a cadeia recebedora checar essa prova, isso é possível -por um block principal de envio. Esse mecanismo é similar ao usado por -cadeias paralelas, que requerem duas cadeias interagindo uma com a outra via -transmissões bidirecionais por dados de prova-de-existência (transações). - -O protocolo IBC pode naturalmente ser definido usando dois tipos de transações: uma -transação `IBCBlockCommitTx`, a qual permite uma blockchain provar para qualquer -espectador o mais recente hash-de-bloco, e uma transação `IBCPacketTx`, a qual -permite uma blockchain provar para qualquer espectador que o pacote recebido foi realmente -publicado pelo remetente, via Merkle-proof para um hash-de-bloco -recente. - -Ao misturar o mecanismo ICB em duas transações separadas, nós permitimos -que o mecanismo de mercado de taxa nativa da blockchain recebedora determine quais pacotes -irão se comprometer (isto é, ser reconhecido), permitindo simultaneamente que uma -blockchain envie de quantos pacotes de saída forem permitidos. - -![Figura da Zona1, Zona2, e Hub IBC sem -reconhecimento](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_without_ack.png) - -No exemplo acima, para atualizar o hash de blocos da -"Zona1" no "Hub" (ou do "Hub" para a "Zona2"), uma transação `IBCBlockCommitTx` -precisa ser feita no "Hub" com o hash de bloco da "Zona1" (ou na -"Zona2" com o hash de bloco do "Hub"). - -_Veja [IBCBlockCommitTx](#ibcblockcommittx) e [IBCPacketTx](#ibcpacketcommit) -para mais informações sobre os 2 tipos de transação IBC._ - -## Casos de Uso - -### Exchange Distribuídas - -Da mesma forma que Bitcoin é mais seguro por ter uma distribuíção, -e replicação em massa, podemos tornar as exchanges menos vulneráveis a -Hacks internos executando-a no blockchain. Chamamos isso de exchange -distribuída. - -O que a comunidade de criptomoedas chama hoje de intercâmbio descentralizado -baseado em algo chamado transações "atomic cross-chain" (AXC). Com uma transação -AXC, dois usuários em duas diferentes cadeias podem fazer duas transações -de transferências que serão feitas juntas nas duas ledgers, ou nenhuma (isto é, -Atomicamente). Por exemplo, dois usuários podem trocar bitcoins por ether (ou qualquer dois -Tokens em dois ledgers diferentes) usando transações AXC, mesmo que o Bitcoin -e o Ethereum não estão conectados entre si. O benefício de executar um -troca em transações AXC é que nenhum dos usuários precisam confiar um no outro ou -no serviço de correspondência comercial. A desvantagem é que ambas as partes precisam estar -on-line para o negócio ocorrer. - -Outro tipo de intercâmbio descentralizado é um sistema de -exchange que funciona em seu próprio blockchain. Os usuários deste tipo de exchange podem -enviar uma ordem de limite e desligar o computador, e o negócio pode ser executado -sem que o usuário esteja online. O blockchain combina e completa o negócio -em nome do negociante. - -Uma exchange centralizada pode criar um vasto livro de ordens de ordens e -atrair mais comerciantes. A liquidez gera mais liquidez no mundo cambial, -e assim há um forte efeito na rede (ou pelo menos efeito de vencedor-leva-mais) -no negócio de câmbio. A atual líder para troca de criptomoedas -hoje é a Poloniex com um volume de 24 milhões de dólares por dia, e em segundo lugar a -Bitfinex com um volume de US$5 milhões por dia. Dados esses fortes efeitos na rede, -é improvável que as exchanges descentralizadas baseadas no AXC ganhem volume -centrais. Para uma exchange descentralizada competir com um -exchange centralizada, seria necessário dar suporte aos livros de -ordens. Somente uma exchange distribuída em uma blockchain pode fornecer isso. - -Tendermint fornece benefícios adicionais para realizar uma transação mais rápida. Com a -finalidade de dar prioridade a rapidez sem sacrificar a consistência, as zonas no Cosmos podem -finalizar transações rápidas - tanto para transações de ordem de -transferências de tokens quanto para outras zonas IBC. - -Dado o estado das exchanges de criptomoedas hoje em dia, uma grande -exchange distribuída da Cosmos (aka o Cosmos DEX). A transação e -a capacidade de processamento, bem como a latência de processos, podem ser -centrais. Os comerciantes podem enviar ordens de limite que podem ser executadas -sem que ambas as partes tenham que estar online. E com Tendermint, o Cosmos Hub, -e o IBC, os comerciantes podem mover fundos dentro e fora da exchange e para outras -zonas com rapidez. - -### Pegging para Outras Criptomoedas - -Uma zona privilegiada pode agir como token simbolico de uma -criptomoeda. A peg é semelhante à relação entre uma -zona e o Cosmos Hub; Ambos devem manter-se atualizados com os -outros, afim de verificar provas de que os tokens passaram de um para o outro. A -Peg-zone na rede Cosmos mantém-se com o Hub, bem como o -outra cryptomoeda. A indireção através da peg-zone permite a lógica de -que o Hub permaceça simples e imutável para outras estratégias de consenso blockchain -como a mineração de prova-de-trabalho do Bitcoin. - -Por exemplo, uma zona Cosmos com um conjunto validador específico, possivelmente o mesmo que -o Hub, poderia atuar como um ether-peg, onde a aplicação TMSP sobre -a zona ("peg-zone") tem mecanismos para trocar mensagens IBC com um -Peg-contract na blockchain (a "origem"). Este contrato -permite que os titulares de ether enviem ether para a zona de peg, enviando-o para -Peg-contract na Ethereum. Uma vez que o ether é recebido pelo peg-contract, o ether -não pode ser retirado a menos que um pacote IBC apropriado seja recebido pelo -Peg-contract da peg-zone. Quando uma zona recebe um pacote IBC provando -que o ether foi recebido no peg-contract para uma determinada conta Ethereum, -a conta correspondente é criada na peg-zone com esse saldo. O ether na -peg-zone ("pegged-ether") pode então ser transferido para o Hub, -e mais tarde ser destruído com uma transação que envia para um determinado -endereço de retirada no Ethereum. Um pacote IBC provando que a transação -na Peg-Zone podem ser lançados no peg-contract Ethereum para permitir que o -Ether seja retirado. - -Naturalmente, o risco do contrato do pegging e um conjunto de validadores desonestos. -bizantino. -O poder de voto bizantino poderia causar um fork, retirando o ether do -peg-contract mantendo o pegged-ether na peg-zone. Na pior das hipóteses, -\+⅔ do poder de voto bizantino pode roubar o ether daqueles que o enviaram para o -peg-contract, desviando-se da pegging e da peg-zone de origem. - -É possível abordar essas questões projetando o peg para ser totalmente -responsável. Por exemplo, todos os pacotes IBC, a partir do hub de -origem, poderão exigir um reconhecimento pela zona de fixação de tal forma que -as transições de estados da peg-zone podem ser desafiadas de forma eficiente e verificadas pelo -hub ou pelo peg-contract de origem. O Hub e a origem devem -permitir que os validadores da zona de fixação apresentem garantias e as transferências -contratuais devem ser atrasadas (e um prazo de vinculação de colateral suficientemente -longo) para permitir que quaisquer desafios sejam feitos por auditores independentes. Nós saímos -da concepção, da especificação e implementação deste sistema aberto como uma -futura proposta de melhoria da Cosmos, a ser aprovada pela governança do sistema do Cosmos -Hub. - -Embora a atmosfera sociopolítica ainda não esteja bastante desenvolvida, podemos -aumentar o mecanismo para permitir que as zonas se liguem as moedas FIAT de um -estado-nação, formando um validador responsável estabelecido a partir de uma combinação da -moeda da nação, mais particularmente, pelos seus bancos. Claro, -precauções adicionais devem ser tomadas para aceitar apenas moedas apoiadas por -sistemas que possam reforçar a capacidade de auditoria das atividades dos bancos -e de notário grupos de grandes instituições de confiança. - -Um resultado dessa integração poderia ser, por exemplo, permitir que -uma conta em um banco na zona possa mover dólares de sua conta bancária -para outras contas na zona, ou para o hub, ou inteiramente para outra zona. -Nesse sentido, o Cosmos Hub pode atuar como um canal sem -moedas e criptomoedas, removendo as barreiras que limitariam -sua interoperabilidade com o mundo dos intercâmbios. - -### Ethereum Scaling - -Resolver o problema de escalonamento é um problema aberto para a Ethereum. Atualmente, -os nós Ethereum processam cada transação única e também armazenam todos os estados. -[link](https://docs.google.com/presentation/d/1CjD0W4l4-CwHKUvfF5Vlps76fKLEC6pIwu1a_kC_YRQ/mobilepresent?slide=id.gd284b9333_0_28). - -Desde que a Tendermint pode realizar os blocos muito mais rápido do que a prova-de-trabalho da Ethereum, -as zonas EVM alimentadas e operando pelo consenso da Tendermint -fornecem maior desempenho para blocos da blockchain Ethereum. Além disso, embora o -Cosmos Hub e o mecanismo de pacotes IBC não permitam a execução da lógica de contratos -arbitrários, podem ser usados para coordenar os movimentos Ethereum e a execução de -contratos simbólicos em diferentes zonas, fornecendo uma base para -o token ethereum através de sharding. - -### Integração de Multi-Aplicação - -As zonas Cosmos executam lógica de aplicação arbitrária, que é definida no início da -vida da zona e podem potencialmente ser atualizados ao longo do tempo pela governança. Essa flexibilidade -permite que as zonas Cosmos ajam como pegs para outras criptomoedas como Ethereum ou -Bitcoin, e também permite derivados desses blockchains, que utilizam a -mesma base de código, mas com um conjunto de validador diferente e distribuição inicial. Isto -permite que muitos tipos de criptomoedas existentes, como as Ethereum, -Zerocash, Bitcoin, CryptoNote e assim por diante, possam ser usados com o Tendermint Core, -que é um motor de consenso de maior desempenho, em uma rede comum, abrindo -oportunidade de interoperabilidade entre as plataformas. Além disso, como -multi-asset blockchain, uma única transação pode conter vários -onde cada entrada pode ser qualquer tipo de token, permitindo a Cosmos -ser uma plataforma para a exchange descentralizada, mesmo que as ordens sejam -para outras plataformas. Alternativamente, uma zona pode servir como um -fault-tolerant (com livros de ordens), o que pode ser uma melhoria -nas exchanges centralizadas de criptomoeda que tendem a ser invadidas com -o tempo. - -As zonas também podem servir como versões bloqueadas de empresas e -sistemas, onde partes de um serviço particular da organização ou grupo de organizações -que são tradicionalmente executadas como um aplicativo TMSP -em uma certa zona, o que lhe permite receber a segurança e a interoperabilidade da -rede pública Cosmos sem sacrificar o controle sobre o serviço subjacente. -Assim, a Cosmos pode oferecer o melhor da tecnologia blockchain para ambos os mundos e -para as organizações, que se recusam a deixar completamente o controle -para um distribuidor terceirizado. - -### Redução de partição de rede - -Alguns afirmam que um grande problema da coerência-favorecendo algoritmos de consenso -como o Tendermint é que qualquer partição de rede que faz com que não haja uma única -partição com +⅔ de poder de votação (por exemplo, ⅓+ ficando offline) irá parar o consenso -completamente. A arquitetura Cosmos pode ajudar a mitigar esse problema usando umas -zonas regionais autônomas, onde o poder de voto para cada zona é -distribuído com base em uma região geográfica comum. Por exemplo, um -parâmetro pode ser para cidades individuais, ou regiões, para operar suas próprias zonas -de partilha com um centro em comum (por exemplo, o Cosmos Hub), permitindo que a -o hub possa parar devido a uma paralisação de rede temporária. -Observe que isso permite uma geologia real, política e rede-topológica, -que são recursos a serem considerados no projeto de sistemas robustos federados de fault-tolerant. - -### Sistema de Resolução de Nomes Federados - -NameCoin foi uma das primeiras blockchains a tentar resolver o -problema de resolução de nomes através de uma adaptação da blockchain do Bitcoin. Infelizmente -têm ocorrido várias questões com esta abordagem. - -Com a Namecoin, podemos verificar que, por exemplo, o nome <em>@satoshi</em> foi registrado como -particular, em algum momento do passado, mas não saberíamos se -a chave pública tinha sido atualizada recentemente, a menos que baixassemos todos os blocos -desde a última atualização desse nome. Isto é devido as limitações do modelo de -Merkle-ization de UTXO do Bitcoin, onde somente as transações (não -mutáveis) são Merkle-ized no hash do bloco. Isso nos permite -provar a existência, mas não a não-existência de atualizações posteriores a um nome. Assim, nós -não podemos saber com certeza o valor mais recente de um nome sem confiar em um -nó, ou recorrer a gastos significativos na largura de banda, baixando o -Blockchain. - -Mesmo se uma árvore de pesquisa Merkle-ized for implementada na NameCoin, sua dependência -sobre a prova-de-trabalho torna a verificação do cliente light problemática. Os clientes light devem -baixar uma cópia completa dos cabeçalhos para todos os blocos em toda a blockchain -(ou pelo menos todos os cabeçalhos desde a última atualização de um nome). Isso significa que -os requisitos de largura de banda crescem linearmente com a o passar do tempo [\[21\]][21]. -Além disso, as mudanças de nome em um bloco de prova-de-trabalho requerem -a confirmação do trabalho, o que pode levar até uma hora -no Bitcoin. - -Com Tendermint, tudo o que precisamos é o hash de bloco mais recente assinado por um quorum de -validadores (por poder de voto), e uma prova Merkle para o valor atual associado -com o nome. Isto torna possível ter uma solução sucinta, rápida e segura -para a verificação de valores de nome no cliente light. - -Na Cosmos, podemos aplicar este conceito e estendê-lo ainda mais. Cada -zona de registro de nomes na Cosmos pode ter um domínio de nível superior (TLD) -associado, como o ".com" ou ".org", e cada zona de registro de nome pode ter -suas próprias regras de governança e registro. - -## Emissão e Incentivos - -### O Token Atom - -Enquanto o Cosmos Hub é um ledger de distribuíção multi-asset, há um token nativo -especial chamado _atom_. Os atoms são o únicos símbolos do Cosmos -Hub. Os atoms são uma licença para o titular votar, validar ou delegar -validadores. Como o ether da Ethereum, os atoms também podem ser usados para -reduzir o spam. Atoms inflacionários adicionais e as taxas do bloco de transação -são recompensadas pelos validadores e delegados que -o validarão. - -A transação `BurnAtomTx` pode ser usada para cobrir proporcionalmente a quantidade -de tokens reservados para a pool. - -#### Levantamento de Fundos - -A distribuição inicial dos tokens atom e validadores na Genesis vão para os -doadores do Levantamento de Fundos da Cosmos (75%), doadores pesados (5%), Fundação da Rede -Cosmos (10%), e a ALL IN BITS, Inc (10%). A partir da Genesis em diante, 1/3 da -quantidade total de atoms será recompensada aos validadores e delegados durante -todo o ano. - -Veja o [Plano Cosmos](https://github.com/cosmos/cosmos/blob/master/PLAN.md) -para detalhes adicionais. - -#### Investindo - -Para evitar que o levantamento de fundos atraia especuladores de curto prazo apenas interessados -em esquemas de pump and dump, os atoms da Genesis não serão transferíveis até -eles tenham investido. Cada conta irá adquirir atoms durante um período de 2 anos com -taxa constante a cada hora, determinada pelo número total de atoms da Genesis/(2* -365 * 24) horas. Os atoms ganhos pela recompensa do bloco são pré-investidos, -e podem ser transferidos imediatamente, de modo que os validadores e os delegados ligados possam ganhar -mais da metade de seus atoms da Genesis após o primeiro ano. - -### Limitações do Número de Validadores - -Diferentemente do Bitcoin ou de outros blockchains de prova-de-trabalho, o blockchain Tendermint será -mais lento com mais validadores devido ao aumento da complexidade da comunicação. -Felizmente, podemos oferecer suporte a validadores suficientes para a -distribuição na Blockchain com tempos de confirmação de transação muito mais rápidos e, através de -largura de banda, armazenamento e aumento da capacidade de computação paralela, seremos capazes de -ter mais validadores no futuro. - -No dia da Genesis, o número máximo de validadores será definido como 100, -o número aumentará a uma taxa de 13% durante 10 anos até atingir a marca de 300 -Validadores. - - Ano 0: 100 - Ano 1: 113 - Ano 2: 127 - Ano 3: 144 - Ano 4: 163 - Ano 5: 184 - Ano 6: 208 - Ano 7: 235 - Ano 8: 265 - Ano 9: 300 - Ano 10: 300 - ... - -### Tornando-se um Validador depois do dia da Genesis - -Os titulares de atoms que ainda não são capazes de se tornarem validadores assinados e -submeter uma transação `BondTx`. A quantidade de atoms fornecida como garantia -deve ser diferente de zero. Qualquer pessoa pode se tornar um validador a qualquer momento, exceto quando o -tamanho do conjunto de validadores atual é maior que o número máximo de -validadores permitidos. Nesse caso, a transação só é válida se o montante -de atoms é maior do que a quantidade de atoms efetivos mantidos pelo menor -validador, onde atoms eficazes incluem atoms delegados. Quando um novo validador -substitui um validador existente de tal forma, o validador existente torna-se -inativo e todos os atoms e atoms delegados entram no estado de unbonding. - -### Penalidades para Validadores - -Deve haver alguma penalidade imposta aos validadores por qualquer desvio intencional -ou não intencional do protocolo sancionado. Algumas evidências são imediatamente admissíveis, -como um double-sign na mesma altura e volta, ou uma violação de "prevote-the-lock" -(uma regra do protocolo de consenso Tendermint). Tais evidências resultarão em que o -validador perca sua boa reputação e seus átomos ligados, bem como sua proporção de tokens -na pool reserva - coletivamente chamados de "stake" - serão cortados. - -Às vezes, os validadores não estarão disponíveis, devido a interrupções na rede regional, -falha de energia ou outros motivos. Se, em qualquer ponto nos blocos `ValidatorTimeoutWindow` -anteriores, o voto de validação de um validador não estiver incluído na cadeia de -blocos mais do que `ValidatorTimeoutMaxAbsent` vezes, esse validador ficará inativo e -perderá `ValidatorTimeoutPenalty` (PADRÃO DE 1%) de sua participação. - -Alguns comportamentos "maliciosos" não produzem provas obviamente discerníveis sobre -a blockchain. Nesses casos, os validadores podem coordenar fora da banda para forçar -o tempo limite desses validadores maliciosos, se houver um consenso majoritário. - -Em situações em que o Cosmos Hub parar devido a uma coalizão de ⅓+ de poder de voto -offline, ou em situações onde uma coalizão de ⅓+ de poder de voto censurar evidências de -comportamento malicioso entrando na blockchain, o hub deve recuperar com um hard-fork -de proposta reorganizacional. (Link to "Forks and Censorship Attacks"). - -### Taxas de Transação - -Os validadores do Cosmos Hub podem aceitar qualquer tipo de token ou combinação -de tipos como taxas para processar uma transação. Cada validador pode fixar subjetivamente a -taxa de câmbio que quiser e escolher as transações que desejar, desde que o `BlockGasLimit` -não seja excedido. As taxas cobradas, menos quaisquer impostos especificados abaixo, -são redistribuídas aos stakeholders ligados em proporção aos seus átomos ligados, cada `ValidatorPayoutPeriod` (PADRÃO DE 1 hora). - -Das taxas de transação cobradas, `ReserveTax` (PADRÃO DE 2%) irá para a pool reserva -para aumentar a pool reserva e aumentar a segurança e o valor da rede Cosmos. Além disso, um -`CommonsTax` (PADRÃO DE 3%) irá para o financiamento de bens comuns. Estes fundos vão para o -`CustodianAddress` para ser distribuído de acordo com as decisões tomadas pelo sistema de governança. - -Os titulares de átomos que delegam o seu poder de voto a outros validadores pagam uma comissão -ao validador delegado. A comissão pode ser definida por cada validador. - -### Incentivando Hackers - -A segurança do Cosmos Hub é uma função da segurança dos validadores subjacentes e da escolha -da delegação pelos delegados. A fim de incentivar a descoberta e notificação precoce de vulnerabilidades -encontradas, o Cosmos Hub incentiva os hackers a publicar exploits bem sucedidos através de uma transação -`ReportHackTx` que diz," Este validador foi hackeado. Por favor, envie recompensa para este endereço". -Depois de tal exploração, o validador e os delegados ficarão inativos, `HackPunishmentRatio` (PADRÃO DE 5%) -dos átomos de todos serão cortados, e`HackRewardRatio` (PADRÃO DE 5%) dos átomos de todos -serão recompensado com o endereço de recompensa do hacker. O validador deve recuperar os átomos -restantes usando sua chave de backup. - -Para evitar que esse recurso seja abusado para transferir átomos não invadidos, -a porção de átomos adquirido vs relativo de validadores e delegados antes e depois do `ReportHackTx` -permanecerá o mesmo, e o bounty do hacker irá incluir alguns átomos relativos, se houver. - -### Específicação de Governança - -O Cosmos Hub é operado por uma organização distribuída que requer um mecanismo de -governança bem definido para coordenar várias mudanças na blockchain, como parâmetros -variáveis do sistema, bem como atualizações de software e emendas constitucionais. - -Todos os validadores são responsáveis por votar em todas as propostas. -Não votar em uma proposta em tempo hábil resultará na desativação automática do -validador por um período de tempo denominado `AbsenteeismPenaltyPeriod` (PADRÃO DE 1 semana). - -Os delegados herdam automaticamente o voto do validador delegado. -Este voto pode ser anulado manualmente. Os átomos não ligados obtêm nenhum voto. - -Cada proposta requer um depósito de tokens de `MinimumProposalDeposit`, -que pode ser uma combinação de um ou mais tokens incluindo átomos. -Para cada proposta, os eleitores podem votar para receber o depósito. -Se mais da metade dos eleitores optarem por receber o depósito (por exemplo, porque a proposta era spam), -o depósito vai para a pool reserva, exceto os átomos que são queimados. - -Para cada proposta, os eleitores podem votar nas seguintes opições: - -- Sim -- Com Certeza -- Não -- Nunca -- Abstenção - -É necessário uma maioria estrita de votos Yea(Sim) ou YeaWithForce(Com certeza) -(ou votos Nay(Não) ou NayWithForce(Nunca)) para que a proposta seja decidida como aceita -(ou decidida como falha), mas 1/3+ pode vetar a decisão da maioria votando "Com certeza". -Quando uma maioria estrita é vetada, todos são punidos com a perda de `VetoPenaltyFeeBlocks` -(PADRÃO DE no valor de um dia de blocos) de taxas (exceto os impostos que não serão afetados), -e a parte que vetou a decisão da maioria será adicionalmente punida com a perda de `VetoPenaltyAtoms` -(PADRÃO DE 0.1%) de seus átomos. - -### Parâmetro de Mudança de Proposta - -Qualquer um dos parâmetros aqui definidos pode ser alterado com a aceitação -de um `ParameterChangeProposal`. - -### Texto da Proposta - -Todas as outras propostas, como uma proposta de atualização do protocolo, serão coordenadas através do genérico `TextProposal`. - -## Roteiro - -Veja [o Plano Cosmos](https://github.com/cosmos/cosmos/blob/master/PLAN.md). - -## Trabalho Relacionado - -Houve muitas inovações no consenso da blockchain e na escalabilidade nos últimos dois anos. -Esta seção fornece um breve levantamento de um seleto número das mais importantes. - -### Sistemas de Consenso - -#### Classic Byzantine Fault Tolerance - -Consenso na presença de participantes maliciosos é um problema que remonta ao início dos anos 1980, -quando Leslie Lamport cunhou a frase "falha bizantina" para se referir ao comportamento do processo -arbitrário que se desvia do comportamento pretendido, que contraste com uma "falha acidental", -em que um processo simplesmente falha. Soluções iniciais foram descobertas para redes síncronas onde -há um limite superior na latência da mensagem, embora o uso prático fosse limitado a ambientes altamente controlados, -como controladores de avião e datacenters sincronizados via relógios atômicos. -Não foi até o final dos anos 90 que a Practical Byzantine Fault Tolerance (PBFT) foi introduzida como -um eficiente algoritmo de consenso parcialmente síncrono capaz de tolerar até ⅓ de processos -comportando-se arbitrariamente. PBFT tornou-se o algoritmo padrão, gerando muitas variações, -incluindo mais recentemente uma criada pela IBM como parte de sua contribuição para a Hyperledger. - -O principal benefício do consenso Tendermint sobre PBFT é que o Tendermint tem uma estrutura -subjacente melhorada e simplificada, um dos quais é um resultado de adotar o paradigma blockchain. -Blocos Tendermint devem confirmar em ordem, o que evita a complexidade e sobrecarga de comunicação -associada a alteração de visão do PBFT's. No Cosmos e muitas outras criptomoedas, -não há necessidade de permitir o bloco <em>N+i</em> onde <em>i >= 1</em> se confirmar, -quando o próprio bloco <em>N</em> ainda não se confirmou. Se a largura de banda é a razão -pela qual o bloco <em>N</em> não se confirmou em uma zona do Cosmos, então isso não ajuda -a usar os votos de compartilhamento de largura de banda para blocos <em>N+i</em>. -Se uma partição de rede ou nós offline for a razão pela qual o bloco <em>N</em> não foi confirmado, -<em>N+i</em> não se comprometerá de qualquer maneira. - -Além disso, o lote de transações em blocos permite que o Merkle-hashing regule o estado da aplicação, -ao invés de resumos periódicos com esquemas de pontos de verificação como PBFT faz. -Isso permite confirmações de transações mais rápidas para clientes leves e uma comunicação mais rápida entre a blockchain. - -Tendermint Core também inclui muitas otimizações e recursos que vão acima e além do que é especificado no PBFT. -Por exemplo, os blocos propostos pelos validadores são divididos em partes, -Merkleized e inútilizados de tal forma que melhora o desempenho da transmissão -(ver LibSwift [\[19\]][19] para inspiração). Além disso, Tendermint Core não faz qualquer suposição sobre -a conectividade ponto-a-ponto, e funciona durante o tempo que a rede P2P está fracamente conectada. - -#### Participação delegada do BitShares - -Apesar de não serem os primeiros a implementar a prova-de-participação (Proof-of-Stake - PoS), -o BitShares [\[12\]][12] contribuiu consideravelmente para a pesquisa e adoção das blockchains que usam o PoS, -particularmente aqueles conhecidos como PoS "delegados". No BitShares, as partes interessadas elegem "testemunhas", -responsáveis por ordenar e confirmar transações e "delegados", responsáveis pela coordenação -de atualizações de software e alterações de parâmetros. Embora o BitShares atinja alto desempenho -(100k tx/s, 1s de latência) em condições ideais, ele está sujeito a ataques de duplo gasto por testemunhas -maliciosas que "forkem" a blockchain sem sofrer uma punição econômica explícita - ele sofre do problema -"nada a perder". O BitShares tenta suavizar o problema permitindo que as transações se refiram a -blocos-hashes recentes. Além disso, as partes interessadas podem remover ou substituir -testemunhas de má conduta diariamente, embora isso não faça nada para punir -explicitamente os ataques bem sucedidos de duplo gasto. - -#### Stellar - -Baseando-se em uma abordagem pioneira da Ripple, a Stellar [\[13\]][13] refinou um modelo do -Federated Byzantine Agreement em que os processos que participam do consenso não constituem -um conjunto fixo e globalmente conhecido. Em vez disso, cada nó de processo codifica uma ou mais -"fatias de quórum", cada uma constituindo um conjunto de processos confiáveis. Um "quórum" na -Stellar é definido como um conjunto de nós que contêm pelo menos uma fatia de quórum para cada -nó no conjunto, de modo que o acordo possa ser alcançado. - -A segurança do mecanismo Stellar baseia-se no pressuposto de que a intersecção de _qualquer_ dois -quóruns é não-vazia, enquanto a disponibilidade de um nó requer pelo menos uma das suas fatias de -quórum para consistir inteiramente de nós corretos, criando um troca externa entre o uso de grandes -ou pequenas fatias-quórum que podem ser difíceis de equilíbrar sem impor pressupostos significativos -sobre a confiança. Em última análise, os nós precisam, de alguma forma, escolher fatias de quórum adequadas -para que haja tolerância suficiente a falhas (ou qualquer "nó intacto" em geral, do qual muitos dos -resultados do trabalho dependem) e a única estratégia fornecida para garantir tal configuração é -hierárquica e similar ao Border Gateway Protocol (BGP), usado por ISPs de primeira linha na -internet para estabelecer tabelas de roteamento globais e usado pelos navegadores para gerenciar -certificados TLS; Ambos notórios por sua insegurança. - -A crítica sobre papel da Stellar nos sistemas PoS baseados em Tendermint é atenuada pela estratégia -de token descrita aqui, em que um novo tipo de token chamado _atom_ é emitido para representar -reivindicações para futuras porções de taxas e recompensas. A vantagem do PoS baseado em Tendermint, -portanto, é a sua relativa simplicidade, ao mesmo tempo que oferece garantias de segurança suficientes e prováveis. - -#### BitcoinNG - -O BitcoinNG é uma proposta de melhoria do Bitcoin que permitiria formas de escalabilidade vertical, -como o aumento do tamanho do bloco, sem as conseqüências econômicas negativas normalmente associadas a tal mudança, -como o impacto desproporcionalmente grande sobre os pequenos mineradores. Esta melhoria é conseguida separando -a eleição do líder da transmissão da transação: os líderes são eleitos pela primeira vez -por prova de trabalho(PoW) em "microblocos", e então são capazes de transmitir transações a -serem confirmadas até que um novo microbloco seja encontrado. Isso reduz os requisitos -de largura de banda necessários para vencer a corrida PoW, permitindo que os pequenos -mineiros possam competir mais justamente, e permitindo que as transações sejam confirmadas -com mais regularidade pelo último minerador para encontrar um micro-bloco. - -#### Casper - -Casper [\[16\]][16] é uma proposta de algoritmo de consenso PoS para o Ethereum. -Seu modo principal de operação é "consenso-por-aposta". Ao permitir que os validadores apostem -iterativamente em qual bloco eles acreditam que será confirmado na blockchain com base nas -outras apostas que eles têm visto até agora, a finalidade pode ser alcançada eventualmente. -[link](https://blog.ethereum.org/2015/12/28/understanding-serenity-part-2-casper/). Esta é uma área ativa -de pesquisa da equipe de Casper. O desafio está na construção de um mecanismo de apostas que pode ser -comprovado como uma estratégia evolutivamente estável. O principal benefício da Casper em relação à -Tendermint pode ser a oferta de "disponibilidade sobre a consistência" - consenso não requer -um quórum +⅔ de poder de voto - talvez ao custo de velocidade de confirmação ou complexidade de implementação. - -### Escala Horizontal - -#### Protocolo Interledger - -O Protocolo Interledger [\[14\]][14] não é estritamente uma solução de escalabilidade. -Ele fornece uma interoperabilidade ad hoc entre diferentes sistemas de ledger através de uma rede -de relações bilaterais livremente acopladas. Tal como a Lightning Network, a finalidade do -ILP é facilitar pagamentos, mas focaliza especificamente pagamentos em diferentes tipos de ledger, -estendendo o mecanismo de transações atômicas para incluir não apenas hash-locks, mas também um -quórum de notários (chamado de Atomic Transport Protocol). O último mecanismo para reforçar a -atomicidade em transacções entre-ledger é semelhante ao mecanismo SPV do cliente leve do Tendermint, -então uma ilustração da distinção entre ILP e Cosmos/IBC é garantida, e fornecida abaixo. - -1. Os notários de um conector em ILP não suportam mudanças de consentimento, e não permitem uma - pesagem flexível entre notários. Por outro lado, o IBC é projetado especificamente para blockchains, - onde os validadores podem ter diferentes pesos, e onde o consentimento pode mudar ao longo da cadeia de blocos. - -2. Como na Lightning Network, o receptor do pagamento em ILP deve estar on-line para enviar - uma confirmação de volta ao remetente. Em uma transferência de token sobre IBC, o conjunto - de validadores da blockchain do receptor é responsável por fornecer a confirmação, não o usuário receptor. - -3. A diferença mais notável é que os conectores do ILP não são responsáveis ou mantêm o estado - autoritário sobre os pagamentos, enquanto que no Cosmos, os validadores de um hub são a autoridade - do estado das transferências de tokens do IBC, bem como a autoridade da quantidade de tokens - mantidos por cada zona (mas não a quantidade de tokens mantidos por cada conta dentro de uma zona). - Esta é a inovação fundamental que permite a tranferência assimétrica segura de tokens de zona para - zona; O conector analógico do ILP no Cosmos é uma persistente e maximamente segura ledger de blockchain, o Cosmos Hub. - -4. Os pagamentos entre contas no ILP precisam ser suportados por uma ordem de compra/venda, uma - vez que não há transferência assimétrica de moedas de um ledger para outro, apenas a transferência - de valor ou equivalentes de mercado. - -#### Sidechains - -Sidechains [\[15\]][15] são um mecanismo proposto para dimensionar a rede Bitcoin através de -blockchains alternativas que são "atreladas" para a blockchain do Bitcoin. As Sidechains -permitem que bitcoins se movam efetivamente da blockchain do Bitcoin para a sidechain e retornarem, -e permitem a experimentação em novos recursos na sidechain. Como no Cosmos Hub, a sidechain e -Bitcoin servem como clientes leves uns dos outros, usando provas SPV para determinar quando as moedas -devem ser transferidas para a cadeia lateral e retornarem. Claro, como o Bitcoin usa PoW, sidechains -centradas em torno do Bitcoin sofrem dos muitos problemas e riscos do PoW como um mecanismo de consenso. -Além disso, esta é uma solução Bitcoin-maximalista que não suporta nativamente uma variedade de tokens e -topologia de rede entre-zona como o Cosmos faz. Dito isto, o mecanismo de núcleo bidirecional atrelado é, -em princípio, o mesmo que o empregado pela rede Cosmos. - -#### Esforços de Escalabilidade do Ethereum - -Ethereum está atualmente pesquisando uma série de estratégias diferentes para fragmentar o -estado da blockchain do Ethereum para atender às necessidades de escalabilidade. -Esses esforços têm como objetivo manter a camada de abstração oferecida pela atual -Ethereum Virtual Machine através do espaço de estado compartilhado. Vários esforços de -pesquisa estão em andamento neste momento. [\[18\]][18][\[22\]][22] - -##### Cosmos vs Ethereum 2.0 Mauve - -Cosmos e Ethereum 2.0 Mauve [\[22\]][22] tem diferentes objetivos de projeto. - -- Cosmos é especificamente sobre tokens. Malva é sobre escalonamento de computação geral. -- O Cosmos não está ligado ao EVM, por isso mesmo VMs diferentes podem interoperar. -- Cosmos permite que o criador da zona determine quem valida a zona. -- Qualquer pessoa pode iniciar uma nova zona no Cosmos (a menos que a governança decida o contrário). -- O hub isola falhas de zonas de modo que tokens invariantes sejam preservados. - -### Escala Geral - -#### Lightning Network - -A Lightning Network é uma proposta de rede de transferência de token operando em uma camada acima -da blockchain do Bitcoin (e outras blockchains públicas), permitindo a melhoria de muitas ordens -de magnitude no processamento de transações movendo a maioria das transações fora da ledger de consenso -para o chamado "Canais de pagamento". Isso é possível graças a scripts de criptomoedas em cadeia, -que permitem que as partes entrem em contratos estatais bilaterais onde o estado pode ser atualizado -compartilhando assinaturas digitais, e os contratos podem ser fechados definitivamente publicando -evidências na blockchain, um mecanismo primeiramente popularizado por trocas atômicas de -cross-chains(cadeias cruzadas). Ao abrir canais de pagamento com muitas partes, os participantes -da Lightning Network podem se tornar pontos focais para encaminhar os pagamentos de outros, -levando a uma rede de canais de pagamento totalmente conectada, ao custo do capital estar ligado aos canais de pagamento. - -Enquanto a Lightning Network também pode facilmente se estender através de várias blockchains independentes -para permitir a transferência de _value_ através de um mercado de câmbio, não pode ser usado para -transferir assimetricamente _tokens_ de uma blockchain para outra. O principal benefício da rede Cosmos -descrita aqui é permitir tais transferências diretas de tokens. Dito isto, esperamos que os canais de -pagamento e a Lightning Network sejam amplamente adotados juntamente com nosso mecanismo de transferência -de token, por razões de economia de custos e privacidade. - -#### Segregated Witness - -Segregated Witness é uma proposta de melhoria do Bitcoin -[link](https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki) que visa aumentar em 2X ou 3X a -taxa de transferência por bloco, ao mesmo tempo que faz a sincronização de blocos ser mais rapida para -novos nós. O brilho desta solução é de como ele funciona dentro das limitações do protocolo atual do Bitcoin -e permite uma atualização de soft-fork (ou seja, os clientes com versões mais antigas do software -continuarão funcionando após a atualização). O Tendermint, sendo um novo protocolo, não tem restrições -de projeto, por isso tem prioridades diferentes de escalonamento. Sobretudo, o Tendermint usa um algoritmo -de rodízio BFT baseado em assinaturas criptográficas em vez de mineração, o que trivialmente permite escalonamento -horizontal através de múltiplas blockchains paralelas, enquanto que os regulares e mais frequentes blocos confirmam -a escala vertical também. - -<hr/> - -## Apêndice - -### Responsabilidade de Fork - -Um protocolo de consenso bem projetado deve fornecer algumas garantias no caso da capacidade de -tolerância ser excedida e o consenso falhar. Isto é especialmente necessário nos sistemas econômicos, -onde o comportamento Bizantino pode ter recompensa financeira substancial. A -garantia maisimportante é uma forma de _fork-accountability_, onde os processos que -fizeram com que o consenso falhasse (ou seja, clientes do protocolo -motivados para aceitar valores diferentes - um fork) podem ser identificados e punidos de acordo com as -regras do protocolo , Ou, possivelmente, o sistema jurídico. Quando o sistema jurídico não é confiável -ou é excessivamente caro para suplicar, os validadores podem ser forçados a fazerem depósitos de segurança -para participar, e esses depósitos podem ser revogados ou cortados, quando um comportamento malicioso é detectado [\[10\]][10]. - -Observe que isso é diferente do Bitcoin, onde o fork é uma ocorrência regular devido à assincronia de -rede e à natureza probabilística de encontrar colisões de hash parciais. Uma vez que, em muitos casos, -um fork malicioso é indistinguível de um fork devido à assincronia, o Bitcoin não pode implementar de -forma confiável a responsabilidade de um fork, com exceção do custo implícito pago por mineradores que -tem a oportunidade de minerarem um bloco órfão. - -### Consenso Tendermint - -Chamamos as fases de votação de _PreVote_ e _PreCommit_. Um voto pode ser para um bloco em particular ou -para _Nil_. Chamamos uma coleção de +⅔ PreVotes para um único bloco na mesma rodada de um _Polka_, e uma -coleção de +⅔ PreCommits para um único bloco na mesma rodada de um _Commit_. Se +⅔ PreCommit para Nil na -mesma rodada, eles passam para a próxima rodada. - -Observe que o determinismo estrito no protocolo incorre em uma suposição de sincronia fraca, pois os líderes -com falhas devem ser detectados e ignorados. Assim, os validadores aguardam algum tempo, _TimeoutPropose_, -antes de Prevote Nil, e o valor de TimeoutPropose aumenta a cada rodada. A progressão através do -resto de uma rodada é totalmente assincrôna, onde o progresso é feito somente quando um validador -ouve de +⅔ da rede. Na prática, seria necessário um adversário extremamente forte para impedir -indefinidamente a suposição de sincronia fraca (fazendo com que o consenso deixasse de confirmar um bloco), -e isso pode ser ainda mais difícil usando valores randomizados de TimeoutPropose em cada validador. - -Um conjunto adicional de restrições, ou Locking Rules(Regras de bloqueio), garante que a rede acabará -por confirmar apenas um bloco em cada altura. Qualquer tentativa maliciosa de confirmar de causar um -bloco a ser confirmado a uma determinada altura pode ser identificada. Primeiro, um PreCommit para um -bloco deve vir com justificação, na forma de um Polka para esse bloco. Se o validador já tiver PreCommit -um bloco na rodada <em>R*1</em>, nós dizemos que eles estão \_locked* nesse bloco, e o Polka usado -para justificar o novo PreCommit na rodada <em>R_2</em> deve vir de uma rodada <em>R_polka</em> -onde <em>R_1 < R_polka <= R_2</em>. Em segundo lugar, os validadores devem propor e/ou pré-votar -o bloco que eles estão travados. Juntas, essas condições garantem que um validador não PreCommit -sem evidência suficiente como justificativa, e que os validadores que já têm PreCommit não podem -contribuir para a evidência de PreCommit algo mais. Isso garante a segurança e a vivacidade do algoritmo de consenso. - -Os detalhes completos do protocolo são descritos -[aqui](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - -### Clientes Leves do Tendermint - -A necessidade de sincronizar todos os cabeçalhos de bloco é eliminada no Tendermint-PoS, como por exemplo -a existência de uma cadeia alternativa (um fork) significando que ⅓+ do stake ligado pode ser reduzido. -Naturalmente, a partir que dividir requer que _someone_ compartilhe evidência de um fork, clientes leves -devem armazenar qualquer bloco-hash comprometido que eles vêem. Além disso, os clientes leves podem -periodicamente ficarem sincronizados com as alterações no conjunto de validadores, para evitar -[ataques de longo alcance](#preventing-long-range-attacks) (mas outras soluções são possíveis). - -Em espírito semelhante do Ethereum, o Tendermint permite que os aplicativos incorporem um hash de raiz -Merkle global em cada bloco, permitindo verifícações fáceis de consultas de estado para fins como saldos -de contas, o valor armazenado em um contrato ou a existência de saída de uma transação não gasta, -dependendo da natureza da aplicação. - -### Prevenção de ataques de longo alcance - -Assumindo uma coleção suficientemente elástica de redes de difusão e um conjunto de validador -estático, qualquer fork na blockchain pode ser detectado e os depósitos dos validadores ofensivos cortados. -Esta inovação, sugerida pela primeira vez por Vitalik Buterin no início de 2014, resolve o problema do "nada a perder" de outras -criptomoedas de PoW (ver [Trabalho Relacionado](#related-work)). No entanto, uma vez que os conjuntos de -validadores devem ser capazes de mudar, durante um longo período de tempo, os validadores originais podem -tornar-se não ligados e, portanto, seriam livres para criar uma nova cadeia a partir do bloco gênese, -não incorrendo nenhum custo, visto que eles não tem depósitos trancados. Este ataque veio a ser conhecido -como Ataque de Longo Alcance (Long Range Attack - LRA), em contraste com um Ataque de Curto Alcance, -onde os validadores que estão atualmente ligados causam um fork e são, portanto, puníveis -(assumindo um algoritimo BFT de fork-responsável como o consenso Tendermint). -Ataques de longo alcance são muitas vezes pensados para serem um golpe crítico para o PoW. - -Felizmente, o LRA pode ser atenuado da seguinte forma. Em primeiro lugar, para que um validador se -desatar (assim recuperando seu depósito colateral e não mais ganhando taxas para participar no consenso), -o depósito deve ser tornado intransferível por um período de tempo conhecido como o "unbonding period" -(período de desatamento), que pode ser na ordem de semanas ou meses. Em segundo lugar, para um cliente -leve ser seguro, a primeira vez que ele se conecta à rede, ele deve verificar um hash de bloqueio recente -contra uma fonte confiável ou, preferencialmente, várias fontes. Esta condição é por vezes referida como -"subjetividade fraca". Finalmente, para permanecer seguro, ele deve sincronizar com o mais recente -validador definido, pelo menos, tão frequentemente quanto a duração do período de desatamento. -Isso garante que o cliente leve saiba sobre as alterações no conjunto de validação definido antes de -um validador não ter mais o seu capital ligado e, portanto, não mais em jogo, o que permitiria enganar -o cliente, executando um ataque de longo alcance, criando novos blocos re-começando em uma altura -a qual foi ligado (assumindo que tem controle de muitas das primeiras chaves privadas). - -Note que superar o LRA desta forma requer uma revisão do modelo de segurança original do PoW. No PoW, -presume-se que um cliente leve pode sincronizar com a altura atual do bloco gênese confiável a qualquer -momento simplesmente processando o PoW em cada cabeçalho de bloco. Para superar o LRA, entretanto, -exigimos que um cliente leve entre em linha com alguma regularidade para rastrear mudanças no conjunto -de validadores e que, na primeira vez em que eles fiquem on-line, eles devem ser particularmente cuidadosos -para autenticar o que ouvem da rede contra fontes confiáveis . Naturalmente, este último requisito é -semelhante ao do Bitcoin, onde o protocolo e o software também devem ser obtidos a partir de uma fonte confiável. - -O método acima para prevenir LRA é bem adequado para validadores e nós completos de uma blockchain alimentada -por Tendermint porque estes nós são destinados a permanecerem conectados à rede. O método também é adequado -para clientes leves que podem ser esperados para sincronizar com a rede com freqüência. No entanto, para -os clientes leves que não se espera ter acesso frequente à Internet ou à rede da blockchain, ainda pode -ser utilizada outra solução para superar o LRA. Os detentores de tokens não validadores podem publicar -os seus tokens como colaterais com um período de não ligação muito longo (por exemplo, muito mais longo -do que o período de não ligação para validadores) e servir clientes leves com um método secundário de -atestar a validade dos blocos atuais e hashes de blocos passados. Embora esses tokens não contam para a -segurança do consenso da blockchain, eles podem fornecer fortes garantias para clientes leves. Se a -consulta histórica de hash de blocos fosse suportada no Ethereum, qualquer pessoa poderia vincular -seus tokens em um contrato inteligente projetado especialmente para isso e fornecer serviços de -comprovação de pagamentos, efetivamente criando um mercado para a segurança contra LRA de cliente leve. - -### Superando Forks e Ataques de Censura - -Devido à definição de uma confimação de bloco, qualquer coalizão de poder de voto ⅓+ pode interromper a -blockchain ficando off-line ou não transmitir os seus votos. Tal coalizão também pode censurar transações -particulares rejeitando blocos que incluem essas transações, embora isso resultaria em uma proporção -significativa de propostas de blocos a serem rejeitadas, o que iria retardar a taxa de blocos -confirmados da blockchain, reduzindo sua utilidade e valor. A coalizão mal-intencionada também pode transmitir -votos em um fio de modo a triturar os blocos confirmados da blockchain para quase parar, ou se envolver em -qualquer combinação desses ataques. Finalmente, isso pode fazer com que a cadeia de blocos "forke" (bifurque), -por dupla assinatura ou violação as regras de bloqueio. - -Se um adversário globalmente ativo também estivesse envolvido, poderia dividir a rede de tal maneira que -possa parecer que o subconjunto errado de validadores era responsável pela desaceleração. Esta não é apenas -uma limitação do Tendermint, mas sim uma limitação de todos os protocolos de consenso cuja -rede é potencialmente controlada por um adversário ativo. - -Para estes tipos de ataques, um subconjunto de validadores deve coordenar através de meios externos -para assinar um proposta de reorganização que escolhe um fork (e qualquer prova disso) e o -subconjunto inicial de validadores com suas assinaturas. Os validadores que assinam tal -proposta de reorganização deixam seu colateral em todos os outros forks. Os clientes -devem verificar as assinaturas na proposta de reorganização, verificar qualquer -evidência e fazer um julgamento ou solicitar ao usuário final uma decisão. Por exemplo, -uma carteira para celular um aplicativo que pode alertar o usuário com um aviso de segurança, -enquanto um refrigerador pode aceitar qualquer proposta de reorganização assinada por -\+½ dos validadores originais por poder de voto. - -Nenhum algoritmo não-sincrônico tolerante a falhas Bizantino pode chegar a um consenso quando ⅓+ -de poder de voto for desonesto, mas um fork supõe que ⅓+ do poder de voto já foram desonestos por -dupla assinatura ou bloqueio de mudança sem justificativa. Portanto, assinar a proposta de -reorganização é um problema de coordenação que não pode ser resolvido por qualquer protocolo -não-sincronico (isto é, automaticamente e sem fazer suposições sobre a confiabilidade da rede subjacente). -Por enquanto, deixamos o problema da coordenação da proposta de reorganização para a coordenação -humana através do consenso social na mídia na internet. Os validadores devem ter cuidado para garantir -que não haja partições de rede remanescentes antes de assinar uma proposta de reorganização, -para evitar situações em que duas propostas de reorganização em conflito sejam assinadas. - -Assumindo que o meio de coordenação é externo e o protocolo é robusto, resulta-se que os forks são -uma preocupação menor do que os ataques de censura. - -Além de forks e censura, que exigem ⅓+ poder de votação Bizantina, uma coalizão de +⅔ poder de -voto pode ser pratica arbitrária, estado inválido. Esta é a característica de qualquer sistema -de consenso (BFT). Ao contrário da dupla assinatura, que cria forks com provas facilmente -verificáveis, a detecção de obrigatoriedade de um estado inválido requer que os pares não -validadores verifiquem blocos inteiros, o que implica que eles mantêm uma cópia local do estado -e executam cada transação, computando a raiz de estado de forma independente para eles mesmos. -Uma vez detectado, a única maneira de lidar com essa falha é através do consenso social. -Por exemplo, em situações em que o Bitcoin falhou, seja por causa de bugs de software -(como em março de 2013), ou praticar um estado inválido devido ao comportamento Bizantino -dos mineradores (como em julho de 2015), a comunidade bem conectada de negócios, desenvolvedores, -mineradores e outras organizações estabeleceu um consenso social sobre quais ações manuais se -faziam necessárias para curar a rede. Além disso, uma vez que se pode esperar que os validadores -de uma cadeia de blocos de Tendermint sejam identificáveis, o compromisso de um estado inválido -pode até ser punido por lei ou por alguma jurisprudência externa, se desejado. - -### Especificação TMSP - -TMSP consiste em 3 tipos de mensagens primárias que são entregues do núcleo para o aplicativo. -O aplicativo responde com mensagens de resposta correspondentes. - -A mensagem `AppendTx` é o cavalo de trabalho da aplicação. Cada transação na blockchain -é entregue com esta mensagem. O aplicativo precisa validar cada transação recebida com a -mensagem AppendTx contra o estado atual, o protocolo de aplicativo e as credenciais -criptográficas da transação. Uma transação validada precisa atualizar o estado do -aplicativo - vinculando um valor a um armazenamento de valores chave ou atualizando o banco de dados UTXO. - -A mensagem `CheckTx` é semelhante à AppendTx, mas é apenas para validar transações. O mempool do -Tendermint Core primeiro verifica a validade de uma transação com o CheckTx e apenas relata -transações válidas para seus pares. Os aplicativos podem verificar um nonce incremental na transação -e retornar um erro em CheckTx se o nonce é antigo. - -A mensagem `Commit` é usada para calcular uma obrigação criptográfica com o estado atual da aplicação, -para ser colocada no próximo cabeçalho do bloco. Isso tem algumas propriedades úteis. Inconsistências -na atualização desse estado agora aparecerão como forks do blockchain que captura uma classe inteira -de erros de programação. Isso também simplifica o desenvolvimento de clientes leves e seguros, -já que as provas de Merkle-hash podem ser provadas verificando o hash de blocos, -e o hash de blocos é assinado por um quórum de validadores (por poder de voto). - -Mensagens TMSP adicionais permitem que o aplicativo acompanhe e altere o conjunto -de validadores e que o aplicativo receba as informações do bloco, como a altura e os votos de confirmação. - -Pedidos/respostas TMSP são simples mensagens Protobuf. -Confira o [arquivo do esquema](https://github.com/tendermint/abci/blob/master/types/types.proto). - -##### AppendTx - -- **Arguments**: - - `Data ([]byte)`: Os bytes de transação solicitada -- **Returns**: - - `Code (uint32)`: Código de resposta - - `Data ([]byte)`: Bytes de resultado, se houver - - `Log (string)`: Debug ou mensagem de erro -- **Usage**:<br/> - Acrescentar e executar uma transação. Se a transação for válida, - CodeType.OK - -##### CheckTx - -- **Arguments**: - - `Data ([]byte)`: Os bytes de transação solicitados -- **Returns**: - - `Code (uint32)`: Código de resposta - - `Data ([]byte)`: Bytes de resultado, se houver - - `Log (string)`: Debug ou mensagem de erro -- **Usage**:<br/> - Validar uma transação. Esta mensagem não deve mutar o estado. - As transações são primeiro executadas através do CheckTx antes da transmissão para os pares na camada mempool. - Você pode fazer o CheckTx semi-stateful e limpar o estado após `Commit` ou - `BeginBlock`, - para permitir sequências dependentes de transações no mesmo bloco. - -##### Commit - -- **Returns**: - - `Data ([]byte)`: O hash Merkle raiz - - `Log (string)`: Debug ou erro de mensagem -- **Usage**:<br/> - Retorna um hash Merkle raiz do estado da aplicação. - -##### Query - -- **Arguments**: - - `Data ([]byte)`: Os bytes de solicitação consultada -- **Returns**: - - `Code (uint32)`: Código de resposta - - `Data ([]byte)`: Os bytes de resposta consultada - - `Log (string)`: Debug ou erro de mensagem - -##### Flush - -- **Usage**:<br/> - Limpar a fila de resposta. Aplicações que implementam `types.Application` - não precisa implementar esta mensagem - é tratada pelo projeto. - -##### Info - -- **Returns**: - - `Data ([]byte)`: Os bytes de informação -- **Usage**:<br/> - Retorna informações sobre o estado da aplicação. Aplicação específicão. - -##### SetOption - -- **Arguments**: - - `Key (string)`: Chave para definir - - `Value (string)`: Valor a definir para a chave -- **Returns**: - - `Log (string)`: Debug ou mensagem de erro -- **Usage**:<br/> - Define as opções do aplicativo. Exemplo Key="mode", Value="mempool" para uma conexão mempool - , ou Key="mode", Value="consensus" para uma conexão de consenso. - Outras opções são específicas da aplicação. - -##### InitChain - -- **Arguments**: - - `Validators ([]Validator)`: validadores de genesis iniciais -- **Usage**:<br/> - Chamado uma vez na genesis - -##### BeginBlock - -- **Arguments**: - - `Height (uint64)`: A altura do bloco que está começando -- **Usage**:<br/> - Sinaliza o início de um novo bloco. Chamado antes de qualquer AppendTxs. - -##### EndBlock - -- **Arguments**: - - `Height (uint64)`: A altura do bloco que terminou -- **Returns**: - - `Validators ([]Validator)`: Mudança de validadores com novos poderes de voto (0 - para remover) -- **Usage**:<br/> - Sinaliza o fim de um bloco. Chamado antes de cada Commit após todas as - transações - -Veja [o repositório TMSP](https://github.com/tendermint/abci) para mais detalhes. - -### Reconhecimento de entrega de pacotes IBC - -Há várias razões pelas quais um remetente pode querer o reconhecimento da entrega de um pacote -pela cadeia de recebimento. Por exemplo, o remetente pode não saber o status da cadeia de -destino, se for esperado que esteja com defeito. Ou, o remetente pode querer impor um tempo -limite no pacote (com o campo `MaxHeight`), enquanto qualquer cadeia de destino pode sofrer -de um ataque de negação de serviço com um aumento repentino no número de pacotes de entrada. - -Nesses casos, o remetente pode exigir confirmação de entrega configurando o status -do pacote inicial como `AckPending`. Em seguida, é a responsabilidade da -cadeia receptora confirmar a entrega, incluindo uma abreviada `IBCPacket` no app Merkle hash. - -![Figura da Zone1, Zone2, e Hub IBC com -reconhecimento](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack.png) - -Primeiro, um `IBCBlockCommit` e`IBCPacketTx` são postados no "Hub" que prova -a existência de um `IBCPacket` na "Zone1". Digamos que `IBCPacketTx` tem o seguinte valor: - -- `FromChainID`: "Zone1" -- `FromBlockHeight`: 100 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 (say) - - `Status`: `AckPending` - - `Type`: "moeda" - - `MaxHeight`: 350 (Dizer que "Hub" está atualmente na altura 300) - - `Payload`: <Os bytes de uma carga paga de "moeda"> - -Em seguida, um `IBCBlockCommit` e `IBCPacketTx` são publicados na "Zone2" que comprova -a existência de um `IBCPacket` em "Hub". Digamos que `IBCPacketTx` tem o seguinte valor: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 300 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckPending` - - `Type`: "moeda" - - `MaxHeight`: 350 - - `Payload`: <Os mesmos bytes de uma carga paga de "moeda"> - -Em seguida, "Zone2" deve incluir em seu app-hash um pacote abreviado que mostra o novo -status de `AckSent`. Um `IBCBlockCommit` e `IBCPacketTx` são colocados de volta no "Hub" -que comprova a existência de um `IBCPacket` abreviado na "Zone2". Digamos que `IBCPacketTx` tem o seguinte valor: - -- `FromChainID`: "Zone2" -- `FromBlockHeight`: 400 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckSent` - - `Type`: "moeda" - - `MaxHeight`: 350 - - `PayloadHash`: <Os bytes de hash da mesma carga paga de "moeda"> - -Finalmente, "Hub" deve atualizar o status do pacote de `AckPending` para`AckReceived`. -A evidência desse novo status finalizado deve voltar a "Zone2". Digamos que `IBCPacketTx` tem o seguinte valor: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 301 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckReceived` - - `Type`: "moeda" - - `MaxHeight`: 350 - - `PayloadHash`: <Os bytes de hash da mesma carga paga de "moeda"> - -Enquanto isso, "Zone1" pode assumir de maneira otimista a entrega bem-sucedida de um pacote -de "moeda", a menos que provas em contrário sejam comprovadas em "Hub". No exemplo acima, -se "Hub" não tivesse recebido um status `AckSent` de "Zone2" pelo bloco 350, ele teria -definido o status automaticamente para `Timeout`. Essa evidência de um tempo limite pode -ser postada novamente na "Zone1", e quaisquer tokens podem ser retornados. - -![Figura da Zone1, Zone2, e Hub IBC com reconhecimento e -timeout](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack_timeout.png) - -### Árvore Merkle e Especificação de Prova - -Existem dois tipos de árvores Merkle suportadas no ecossistema Tendermint / Cosmos: A Árvore Simples e a Árvore IAVL+. - -#### Árvore Simples - -A Árvore Simples é uma árvore Merkle para uma lista estática de elementos. Se o número de -itens não for um poder de dois, algumas folhas estarão em níveis diferentes. Árvore Simples -tenta manter ambos os lados da árvore da mesma altura, mas a esquerda pode ter um maior. -Esta árvore Merkle é usada para Merkle-lizar as transações de um bloco, e os elementos de -nível superior da raiz do estado do aplicativo. - - * - / \ - / \ - / \ - / \ - * * - / \ / \ - / \ / \ - / \ / \ - * * * h6 - / \ / \ / \ - h0 h1 h2 h3 h4 h5 - - Uma ÁrvoreSimples com sete elementos - -#### Árvore IAVL+ - -O objetivo da estrutura de dados IAVL+ é fornecer armazenamento persistente para pares de valores-chave -no estado do aplicativo, de modo que um hash determinista de raiz Merkle possa ser calculado -eficientemente. A árvore é balanceada usando uma variante do [algoritmo AVL](https://en.wikipedia.org/wiki/AVL_tree), e todas as operações são O(log(n)). - -Em uma árvore AVL, as alturas das duas subárvores filhas de qualquer nó diferem por no máximo um. -Sempre que esta condição for violada após uma atualização, a árvore é rebalanceada criando O(log(n)) -novos nós que apontam para nós não modificados da árvore antiga. No algoritmo AVL original, os nós -internos também podem conter pares de valores-chave. O algoritmo AVL + (observe o sinal de adição) -modifica o algoritmo AVL para manter todos os valores em folha de nós, enquanto -usando apenas nós de ramo para armazenar chaves. Isso simplifica o algoritmo, mantendo a trilha hash merkle curta. - -A Árvore AVL + é análoga à Ethereum [Patricia tries](https://en.wikipedia.org/wiki/Radix_tree). -Há compensações. Chaves não precisam ser hasheadas antes da inserção em árvores IAVL+, portanto, -isso fornece iteração mais rápida ordenada no espaço-chave que pode beneficiar algumas aplicações. -A lógica é mais simples de implementar, requerendo apenas dois tipos de nós - nós internos e nós de folhas. -A prova de Merkle é em média mais curta, sendo uma árvore binária equilibrada. Por outro lado, -a raiz Merkle de uma árvore IAVL+ depende da ordem das atualizações. - -Iremos apoiar outras árvores Merkle eficientes, como Patricia Trie, da Ethereum, quando a variante binária estiver disponível. - -### Tipos de Transação - -Na implementação canônica, as transações são transmitidas para o aplicativo Cosmos hub através da interface TMSP. - -O Cosmos Hub aceitará uma série de tipos de transações primárias, incluindo `SendTx`, -`BondTx`, `UnbondTx`, `ReportHackTx`, `SlashTx`, `BurnAtomTx`, `ProposalCreateTx` e `ProposalVoteTx`, -que são relativamente auto-explicativas e será documentado em uma futura revisão deste artigo. -Aqui documentamos os dois principais tipos de transação para IBC: `IBCBlockCommitTx` e `IBCPacketTx`. - -#### IBCBlockCommitTx - -Uma transação `IBCBlockCommitTx` é composta de: - -- `ChainID (string)`: O ID da blockchain -- `BlockHash ([]byte)`: Os bytes de hash de bloco, a raiz Merkle que inclui o app-hash -- `BlockPartsHeader (PartSetHeader)`: Os bytes de cabeçalho do conjunto de blocos, - apenas necessários para verificar assinaturas de voto -- `BlockHeight (int)`: A altura do commit -- `BlockRound (int)`: A rodada do commit -- `Commit ([]Vote)`: O +⅔ Tendermint `Precommit` de votos que compõem um bloco -- `ValidatorsHash ([]byte)`: O hash da raiz da árvore-Merkle do novo conjunto de validadores -- `ValidatorsHashProof (SimpleProof)`: Uma ÁrvoreSimples da prova-Merkle para provar o - `ValidatorsHash` contra o `BlockHash` -- `AppHash ([]byte)`: Um hash da raiz da árvore-Merkle da Árvore IAVL do estado de aplicação -- `AppHashProof (SimpleProof)`: Uma ÁrvoreSimples da prova-Merkle para provar o - `AppHash` contra o `BlockHash` - -#### IBCPacketTx - -Um `IBCPacket` é composto de: - -- `Header (IBCPacketHeader)`: O cabeçalho do pacote -- `Payload ([]byte)`: Os bytes da carga paga do pacote. _Optional_ -- `PayloadHash ([]byte)`: O hash para os bytes do pacote. _Optional_ - -Qualquer um dos `Payload` ou `PayloadHash` deve estar presente. O hash de um `IBCPacket` -é uma raiz Merkle simples dos dois itens, `Header` e `Payload`. Um `IBCPacket` sem a carga completa -é chamado de _abbreviated packet_. - -Um `IBCPacketHeader` é composto de: - -- `SrcChainID (string)`: O ID da blockchain fonte -- `DstChainID (string)`: O ID da blockchain destino -- `Number (int)`: Um número exclusivo para todos os pacotes -- `Status (enum)`: Pode ser um `AckPending`, `AckSent`, `AckReceived`, - `NoAck`, ou `Timeout` -- `Type (string)`: Os tipos são dependentes da aplicação. Cosmos reserva-se ao tipo de pacote "moeda" -- `MaxHeight (int)`: Se status não for `NoAckWanted` ou `AckReceived` por essa altura, o status se tornará `Timeout`. _Opcional_ - -Uma transação `IBCPacketTx` é composta de: - -- `FromChainID (string)`: O ID da blockchain que está fornecendo este pacote; Não necessariamente a fonte -- `FromBlockHeight (int)`: A altura da blockchain na qual o seguinte pacote é incluído (Merkle-izado) no hash da blockchain de origem -- `Packet (IBCPacket)`: Um pacote de dados, cujo estado pode ser um - `AckPending`, `AckSent`, `AckReceived`, `NoAck`, ou `Timeout` -- `PacketProof (IAVLProof)`: Uma prova-Merkle da Árvore IAVL para para provar o hash do pacote contra o \`AppHash' da cadeia de origem em determinada altura - -A seqüência para enviar um pacote da "Zone1" para a "Zone2" através do "Hub" é mostrada em {Figure X}. -Primeiro, um `IBCPacketTx` prova ao "Hub" que o pacote está incluído no estado da aplicação de "Zone1". -Em seguida, outro `IBCPacketTx` prova a "Zone2" que o pacote está incluído no estado da aplicação "Hub". -Durante esse procedimento, os campos `IBCPacket` são idênticos: o `SrcChainID` é sempre "Zone1", -e o `DstChainID` é sempre" Zone2 ". - -O `PacketProof` deve ter o caminho correto da prova-Merkle, da seguinte maneira: - - IBC/<SrcChainID>/<DstChainID>/<Number> - -Quando "Zone1" quer enviar um pacote para "Zone2" através do "Hub", os dados de `IBCPacket` -são idênticos se o pacote é Merkle-izado em "Zone1", no "Hub" ou "Zone2". O único campo mutável -é `Status` para acompanhar a entrega, conforme mostrado abaixo. - -## Agradecimentos - -Agradecemos aos nossos amigos e colegas por sua ajuda na conceituação, revisão e apoio no nosso trabalho com Tendermint e Cosmos. - -- [Zaki Manian](https://github.com/zmanian) da - [SkuChain](http://www.skuchain.com/) forneceu muita ajuda na formatação e redacção, especialmente sob a seção TMSP -- [Jehan Tremback](https://github.com/jtremback) da Althea and Dustin Byington - por ajudar com iterações iniciais -- [Andrew Miller](https://soc1024.com/) da [Honey - Badger](https://eprint.iacr.org/2016/199) pelo feedback sobre consenso -- [Greg Slepak](https://fixingtao.com/) pelo feedback sobre consenso e redação -- Também agradecemos ao [Bill Gleim](https://github.com/gleim) e [Seunghwan - Han](http://www.seunghwanhan.com) por várias contribuições. -- [Pedro Augusto](https://github.com/ShooterXD) pela tradução para - Português - -## Citações - -- [1] Bitcoin: <https://bitcoin.org/bitcoin.pdf> -- [2] ZeroCash: <http://zerocash-project.org/paper> -- [3] Ethereum: <https://github.com/ethereum/wiki/wiki/White-Paper> -- [4] TheDAO: <https://download.slock.it/public/DAO/WhitePaper.pdf> -- [5] Segregated Witness: <https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki> -- [6] BitcoinNG: <https://arxiv.org/pdf/1510.02037v2.pdf> -- [7] Lightning Network: <https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf> -- [8] Tendermint: <https://github.com/tendermint/tendermint/wiki> -- [9] FLP Impossibility: <https://groups.csail.mit.edu/tds/papers/Lynch/jacm85.pdf> -- [10] Slasher: <https://blog.ethereum.org/2014/01/15/slasher-a-punitive-proof-of-stake-algorithm/> -- [11] PBFT: <http://pmg.csail.mit.edu/papers/osdi99.pdf> -- [12] BitShares: <https://bitshares.org/technology/delegated-proof-of-stake-consensus/> -- [13] Stellar: <https://www.stellar.org/papers/stellar-consensus-protocol.pdf> -- [14] Interledger: <https://interledger.org/rfcs/0001-interledger-architecture/> -- [15] Sidechains: <https://blockstream.com/sidechains.pdf> -- [16] Casper: <https://blog.ethereum.org/2015/08/01/introducing-casper-friendly-ghost/> -- [17] TMSP: <https://github.com/tendermint/abci> -- [18] Ethereum Sharding: <https://github.com/ethereum/EIPs/issues/53> -- [19] LibSwift: <http://www.ds.ewi.tudelft.nl/fileadmin/pds/papers/PerformanceAnalysisOfLibswift.pdf> -- [20] DLS: <http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf> -- [21] Thin Client Security: <https://en.bitcoin.it/wiki/Thin_Client_Security> -- [22] Ethereum 2.0 Mauve Paper: <https://cdn.hackaday.io/files/10879465447136/Mauve%20Paper%20Vitalik.pdf> - -#### Links não classificados - -- <https://www.docdroid.net/ec7xGzs/314477721-ethereum-platform-review-opportunities-and-challenges-for-private-and-consortium-blockchains.pdf> diff --git a/docs/resources/whitepaper-zh-CN.md b/docs/resources/whitepaper-zh-CN.md deleted file mode 100644 index 126deabd4..000000000 --- a/docs/resources/whitepaper-zh-CN.md +++ /dev/null @@ -1,1013 +0,0 @@ -# Cosmos -**A Network of Distributed Ledgers** - -**分布式账本网络** - -Jae Kwon jae@tendermint.com <br/> -Ethan Buchman ethan@tendermint.com - -讨论[请加入Telegram](https://t.me/cosmosproject)! - -_注意:如果你能在github上阅读,我们仍然定时更新这个文档,请定期检查更新!_ - -## Table of Contents ########################################################### - * [介绍](#介绍) - * [Tendermint](#tendermint) - * [验证人](#验证人) - * [共识](#共识) - * [轻客户端](#轻客户端) - * [防止攻击](#防止攻击) - * [ABCI](#abci) - * [Cosmos 概述](#Cosmos-概述 ) - * [Tendermint-拜占庭容错](#Tendermint-拜占庭容错) - * [治理](#治理) - * [枢纽与分区](#枢纽与分区) - * [枢纽](#枢纽) - * [分区](#分区) - * [跨链通信 IBC](#跨链通信-IBC) - * [用例](#用例) - * [分布式交易所](#分布式交易所) - * [作为其他加密货币的纽带](#作为其他加密货币的纽带) - * [以太坊的扩展](#以太坊的扩展) - * [多用一体化](#多用一体化) - * [缓解网络分区问题](#缓解网络分区问题) - * [联邦式名称解析系统](#联邦式名称解析系统) - * [发行与激励](#发行与激励) - * [Atom 代币](#Atom-代币) - * [众筹](#众筹) - * [验证人的数量限制](#验证人的数量限制) - * [成为创世日后的验证人](#成为创世日后的验证人) - * [对验证人的惩罚](#对验证人的惩罚) - * [交易费用](#交易费用) - * [激励黑客](#激励黑客) - * [治理规范](#治理规范) - * [参数变更提案](#参数变更提案) - * [文本提案](#文本提案) - * [路线图](#路线图) - * [相关工作 ](#相关工作 ) - * [共识系统](#共识系统) - * [经典拜占庭容错](#经典拜占庭容错) - * [BitShare委托权益](#BitShare委托权益) - * [Stellar](#stellar) - * [BitcoinNG](#bitcoinng) - * [Casper](#casper) - * [水平扩展](#水平扩展) - * [Interledger协议](#Interledger协议) - * [侧链](#侧链) - * [以太坊扩展性的努力](#以太坊扩展性的努力) - * [普遍扩展](#普遍扩展) - * [闪电网络](#闪电网络) - * [隔离验证人](#隔离验证人) - * [附录](#附录) - * [分叉问责制](#分叉问责制) - * [Tendermint共识](#Tendermint共识) - * [Tendermint轻客户端](#Tendermint轻客户端) - * [远程攻击的防御](#远程攻击的防御) - * [克服分叉与审查攻击](#克服分叉与审查攻击) - * [ABCI说明](#ABCI说明) - * [IBC数据包交付确认](#IBC数据包交付确认) - * [默克尔树及默克尔证明的说明](#默克尔树及默克尔证明的说明) - * [交易类型](#交易类型) - * [IBCBlockCommitTx](#ibcblockcommittx) - * [IBCPacketTx](#ibcpackettx) - * [鸣谢](#鸣谢) - * [引用](#引用) - -## 介绍 ################################################################ - - -开源的生态系统、去中心化的文件共享、以及公共的加密货币,这一系列技术的成功使人们启发和理解,去中心化的互联网协议是可以从根本上改善社会经济基础架构的。我们已经见识过个有专长的区块链应用,诸如比特币[\[1\]][1](加密货币),ZCASH [\[2\]][2] (隐私加密货币),也看到了例如以太坊 [\[3\]][3] 的大众智能合约平台,还有无数基于 EVM (以太坊虚拟机)开发的分布式应用,例如 Augur(预测市场)和 TheDAO [\[4\]][4] (投资俱乐部) - - -然而,迄今为止,这些区块链已经暴露了各种缺陷,包括总体能效低下,性能不佳或受到限制和缺乏成熟的治理机制。为了扩大比特币交易吞吐量,已经研发了许多诸如隔离见证 [\[5\]][5](Segregated-Witness)和BitcoinNG [\[6\]][6](一种新的可扩展协议)这样的解决方案,但这些垂直扩展解决方案仍然受到单一物理机容量的限制,以确保完整的可审计性。闪电网络 [\[7\]][7] 可以通过部分交易完全记录在主链账本外来扩展比特币的交易容量,这种方法十分适用于微支付和隐私保护支付通道,但是无法适用于更通用的扩展需求。 - - -理想的解决方案是允许多个并行的区块链交互操作的同时保持其安全特性。事实证明,采用工作量证明很难做到这一点,但也并非不可能。例如合并挖矿,允许在工作完成的同时,确保母链在子链上被重复使用,但交易必须通过每个节点依次进行验证,而且如果母链上的大多数哈希算力没有积极地对子链进行合并挖矿,那么就容易遭受到攻击。关于[可替代区块链网络架构的学术回顾](http://vukolic.com/iNetSec_2015.pdf) 将在附件中展示,我们也会在[相关工作](#related-work)中对其他(技术)方案和缺陷进行概括。 - - -这里我们要介绍的 Cosmos,一个全新的区块链网络架构,能够解决所有这些问题。Cosmos 是由许多被称之为“分区”的独立区块链组成的网络。分区在 Tendermint Core [\[8\]][8]的支持下运行,Tendermint Core 是一个[类似拜占庭容错](http://tendermint.com/blog/tendermint-vs-pbft/)安全共识引擎,具有高性能、一致性的特性,并且在严格的[分叉追责](#fork-accountability) 机制下能够制止恶意破坏者的行为。Tendermint Core 的拜占庭容错共识算法十分适合用于扩展权益证明(PoS)机制下的公共区块链。使用其他共识模型的区块链, 包括类似基于权益证明(PoS)的以太坊,以及比特币也能够通过使用适配分区被 Cosmos 网络连接。 - - -Cosmos 的第一个分区称之为 Cosmos 枢纽。Cosmos 枢纽是一种多资产权益证明加密货币网络,它通过简单的治理机制能够对网络进行适配和升级。此外,Cosmos 枢纽可以通过链接其他分区来实现扩展。 - - -Cosmos 网络的枢纽及各个分区可以通过区块链间通信(IBC)协议进行通信,这种协议就是针对区块链的虚拟用户数据报协议(UDP)或者传输控制协议(TCP)。代币可以安全、快速地从一个分区转到其他分区,而无需在两个分区之间拥具有汇兑流动性。相反,所有跨分区的代币转移都会通过 Cosmos 枢纽,以此来追踪记录每个分区持有代币的总量。这个枢纽会将每个分区与其他故障分区隔离开。因为每个人都可以将新的分区连接到 Cosmos 枢纽,所以分区将可以向后兼容新的区块链技术。 - - -利用 Cosmos 可以实现区块链间的互操作。这是一个具有潜力的有价值的互联网络,其中的资产由不同的验证人发布和控制,并可以在不依靠需要信任的第三方的情况下实现跨链资产无缝的转移和交易。 - -## Tendermint - - -在这一部分我们将阐述Tendermint共识协议和用于建立其应用程序的接口。 更多信息,请参见[附录](#附录) - -### 验证人 - - - -在经典的拜占庭容错算法中,每个节点有相同的权重。在 Tendermint,节点有着不同数量(非负)的 _投票权_,而那些拥有相当数量投票权的节点称之为 _验证人_。验证人通过广播加密签名、投票或者对下一个区块表决同意来参与共识协议。 - -验证者的投票权是一开始就确定好了,或者根据应用程序由区块链来决定修改投票权。例如,在像Cosmos 枢纽的权益证明应用里,投票权可由绑定为押金的代币数量来决定。 - - -注意:像⅔和⅓这样的分数指的是占总投票权的分数,而不是总验证人,除非所有验证人拥有相同权重。而>⅔ 的意思是"超过⅔ ",≥⅓则是"⅓或者更多"的意思。 - -### 共识 - - -Tendermint 是部分同步运作的拜占庭容错共识协议,这种协议源自DLS共识算法 [\[20\]][20]。Tendermint以简易性、高性能以及[分叉问责制](#fork-accountability)而著称。协议要求这组验证人固定且被熟知,并且每个验证人都有其公钥验证身份。这些验证人试图同时在一个区块上达成共识,这些区块是一系列的交易记录。每个区块的共识轮流进行,每一轮都会有个领头人,或者提议人,由他们来发起区块。之后验证人分阶段对是否接受该区块,或者是否进入下一轮做出投票。每轮的提议人会从验证人顺序列表中按照其投票权比例来选择确定。 - - -更多协议的全部细节,请点击[这里](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - - -Tendermint 采用了使用大多数投票(超过三分之二)和锁定机制的最优拜占庭容错,来确保其安全性。这些能够保证: - -* 蓄意破坏者想要造成安全性问题,必须有三分之一以上的投票权,并且要提交超过两份以上的值。 -* 如果有一组验证人成功破坏了安全性,或者曾试图这么做,他们会被协议识别。协议包括对有冲突的区块进行投票和广播那些有问题的投票。 - - -除了其超强的安全性外,Tendermint还具备杰出的性能。以商用型云平台为例,Tendermint共识以分布在五大洲七个数据中心的64位节点为基准,其每秒可以处理成千上万笔交易,订单提交延迟时间为1-2秒。而值得关注的是,即使是在极其恶劣的敌对环境中,比如验证人崩溃了或者是广播恶意破坏的投票,也能维持这种每秒超过千笔交易的较高性能。详见下图。 - -![Figure of Tendermint throughput performance](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/tendermint_throughput_blocksize.png) - -### 轻客户端 - - -Tendermint 共识算法的主要好处是具有安全简易的客户端,使其成为手机和物联网用例的理想选择。比特币轻客户端必须同步运行区块头组成的链,并且找到工作量证明最多的那一条链,而Tendermint轻客戸端只需和验证组的变化保持一致,然后简单地验证最新区块中预先提交的>⅔,来确定最新情况。 - -这种简单的轻客戸端证明机制也可以实现[区块链之间的通信](#inter-blockchain-communication-ibc)。 - -### 防止攻击 - - -Tendermint 有各种各样的防御措施来防止一些明显的攻击,比如[远程无利害关系双花攻击](#preventing-long-range-attacks) 及[审查制度](#overcoming-forks-and-censorship-attacks)。 这些在[附录](#appendix)中有更详细的讨论。 - -### ABCI - - -Tendermint共识算法是在叫做 Tendermint Core 的程序中实现的。这个程序是独立于应用的“共识引擎”,可以将任何已经确定的黑盒应用转变为分布式、可复制的区块链。Tendermint Core 可以通过应用区块链接口(ABCI) [\[17\]][17]与其他区块链应用连接。而且,应用区块链接口(ABCI) 接口允许区块链应用以任何语言编程实现,而不仅仅是写这个共识引擎所使用的语言。此外,应用区块链接口(ABCI) 也让交换任何现有区块链栈的共识层成为可能。 - - -我们将其与知名加密货币比特币进行了类比。在比特币这种加密币区块链中,每个节点都维持着完整的审核过的 UTXO(未使用交易输出)数据库。如果您想要在应用区块链接口(ABCI)基础上,创建出类似比特币的系统,那么 Tendermint Core 可以做到: - -* 在节点间共享区块及交易 -* 创建规范或不可改变的交易顺序(区块链) - - -同时,ABCI应用也可以做到: - -* 维护 UTXO 数据库 -* 验证交易的加密签名 -* 防止出现不存在的余额被交易 -* 允许客户访问UTXO数据库 - - - -## Cosmos 概述 - - -Cosmos是一个独立平行的区块链网络,其中每条区块链通过 Tendermint[1](http://github.com/tendermint/tendermint)这样的经典拜占庭容错共识算法来运行。 - - -网络中第一条区块链将会是 Cosmos 枢纽。Cosmos 枢纽通过全新的区块链间通信协议来连接其他众多区块链(或将其称之为 _分区_)。Cosmos 枢纽可以追踪无数代币的种类,并且在各个连接的分区里记录各种代币总数。代币可以安全快速地从一个分区转移到另一个分区,两者之间无需体现汇兑流动性,因为所有分区之间的代币传输都会经过 Cosmos 枢纽。 - - -这一架构解决了当今区块链领域面临的许多问题,包括应用程序互操作性、可扩展性、以及可无缝升级的能力。比如,从Bitcoind、Go-Ethereum、CryptoNote、ZCash或其他区块链系统中衍生出来的分区,都能被锚定接入 Cosmos 枢纽。这些分区允许 Cosmos 实现无限扩展,从而满足全球交易的需求。此外,分区也完全适用于分布式交易所,反之交易所也支持分区运行。 - - - -Cosmos 不仅仅是单一的分布式账本,而 Cosmos 枢纽也不是封闭式庭院或宇宙的中心。我们正在为分布式账本的开放网络设计一套协议,这套协议将基于密码学、稳健经济学、共识理论、透明性及可追责制的原则,成为未来金融系统的全新基础。 - -### Tendermint-拜占庭容错 - - -Cosmos 枢纽是 Cosmos 网络中第一条公共区块链,通过 Tendermint 的拜占庭共识算法运行。Tendermint 开源项目创立于2014年,旨在解决比特币工作量证明共识算法的速度、可扩展性以及造成的环境问题。通过采用并提高已经过验证的拜占庭算法(1988年在麻省理工学院开发)[\[20\]][20],Tendermint 成为了首个在概念论证了权益证明加密货币的团队,这种机制可以解决 NXT 和 BitShares 这些第一代权益证明加密币面临的"无利害关系"的问题。 - - -如今,实际上所有比特币移动钱包都要使用可靠的服务器来进行交易验证。这是因为工作量证明机制需要在交易被认定为无法逆转前进行多次确认。而在 Coinbase 之类的服务中也已经出现双重支付攻击。 - - -和其他区块链共识系统不同,Tendermint 提供的是即时、可证明安全的移动客户端支付验证方式。因为 Tendermint 被设计为完全不分叉,所以移动钱包就可以实时接收交易确认,从而在智能手机上真正实现去信任的支付方式。这一点也大大影响了物联网应用程序。 - - -Cosmos 中的验证人角色类似比特币矿工,但是他们采用加密签名来进行投票。验证人是专门用来提交区块的安全机器。非验证人可以将权益代币(也叫做"atom"币)委托给任何验证人来赚取一定的区块费用以及atom奖励,但是如果验证人被黑客攻击或者违反协议规定,那么代币就会面临被惩罚(削减)的风险。Tendermint 拜占庭共识的可证明安全机制,以及利益相关方(验证人和委托人)的抵押品保证,为节点甚至是轻客户端提供了可证明、可量化的安全性。 - -### 治理 - -分布式公共账本应该要有一套章程与治理体系。比特币依靠比特币基金会以及挖矿来协作更新,但是这是一个反应缓慢的治理制度。以太坊在采用硬分叉成 ETH 和 ETC 来解决 The DAO 黑客,这主要是因为之前没有设定社会契约或机制来进行这类决定。 - - -Cosmos 枢纽的验证人与委托人可以对提案进行投票,从而改变预先默认设置好的系统参数(比如区块转账费用限制),协作更新,并对可读性的章程进行修订投票,从而治理 Cosmos 枢纽制度。这个章程允许权益相关者聚集到一起,来解决盗窃及漏洞等相关问题(比如The DAO事件),并得出更快更明确的解决方案。 - - - -每个分区也可以制定自己的一套章程及治理机制。比如,Cosmos 枢纽的章程可以设置为强制实现枢纽的不可改变性(不能回滚,除了 Cosmos 枢纽节点产生的漏洞),而每个分区则可设置自己的回滚政策。 - - -Cosmos 网络能够在制度不同的分区间实现互操作性,这一点给客户极高的自由度和潜力而无需许可即可实验(新技术)。 - -## 枢纽与分区 - - -这里我们将描述一个全新的去中心化与可扩展性模型。Cosmos 网络通过 Tendermint 机制来运行众多的区块链。虽然现存提案的目标是创建一个包含全球所有交易订单的"单一区块链",但是 Cosmos 允许众多区块链在并行运行的同时,保持可互操作性。 - - -在这个基础上,Cosmos枢纽负责管理称之为“分区”的众多独立区块链(有时也叫做"分片",参考自众所周知的数据库扩展技术"分片")。枢纽上的分片会源源不断地提交最新区块,这一点可以让枢纽同步每一个分区的状态。同样地,每个分区也会和枢纽的状态保持一致(不过分区之间不会同彼此的同步,除非间接通过枢纽来实现)。通过发布默克尔证明来证明消息被接受和发送,来让消息从一个分区传递到另一个分区。这种机制叫做"区块链间通信",或者简称为"IBC"机制。 - -![Figure of hub and zones acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/hub_and_zones.png) - - -任何分区都可以自行成为枢纽来建立非循环图表,但为了清楚起见,我们只描述这种只有一个枢纽和许多非枢纽的分区这样简单的配置 - - -### 枢纽 - - -Cosmos枢纽是承载多种分布式账本资产的区块链,其中代币可以由个人或分区自己持有。这些代币能够通过特殊的IBC数据包,即"代币数据包"(coin packet)从一个分区转移到另一个分区。枢纽负责保持各个分区中各类代币总量不变。IBC代币数据包交易必须由发送人、枢纽及区块接受者执行。 - - - -因为Cosmos枢纽在整个系统中扮演着中央代币账本的角色,其安全性极其重要。虽然每个分区可能都是一个Tendermint区块链——只需通过4个,(或者在无需拜占庭容错共识的情况下更少的验证人来保证安全),但是Cosmos枢纽必须通过全球去中心化验证组来保证安全,而且这个验证组要能够承受最严重的攻击,比如区域网络分裂或者由国家发起的攻击。 - -### 分区 - - -Cosmos分区是独立的区块链,能够和Cosmos枢纽进行IBC消息交换。从枢纽的角度上看,分区是一种多重资产、动态会员制的多重签名账户,可以通过IBC数据包用来发送和接受代币。就像加密币账户一样,分区不能转移超出其持有量的代币,不过可以从其他拥有代币的人那里接收代币。分区可能会被指定为一种或多种代币的"来源",从而赋予其增加代币供应量的权力。 - - -Cosmos 枢纽的 Atom 或可作为分区验证人连接到枢纽的筹码。虽然在Tendermint分叉责任制下,分区出现双重支付攻击会导致atom数量减少,但是如果分区中有超过⅔的选票都出现拜占庭问题的话,那这个分区就可以提交无效状态。Cosmos 枢纽不会验证或执行提交到其他分区的交易,因此将代币发送到可靠的分区间就是用户的责任了。未来 Cosmos 枢纽的管理系统可能会通过改善提案,来解决分区故障问题。比如,在检测到袭击时,可以将有些分区(或全部分区)发起的代币转账将被暂停,实现紧急断路(即暂时中止代币转账)。 - -## 跨链通信-IBC - -现在我们来介绍下枢纽与分区之间通信的方法。假如现在有三个区块链,分别是"分区1"、“分区2"以及"枢纽”,我们想要"分区1"生成一个数据包,通过"枢纽"发送给"分区2"。为了让数据包从一个区块链转移到另一个区块链,需要在接收方区块链上发布一个证明,来明确发送方已经发起了一个数据包到指定目的地。接收方要验证的这个证明,必须和发送方区块头保持一致。这种机制就类似与侧链采用的机制,它需要两个相互作用的链,通过双向传送存在证明数据元(交易),来"知晓"另一方的情况。 - -IBC协议可以自然定义为两种交易的使用:一种是IBCBlockCommitTx 交易,这种交易可以让区块链向任何观察员证明其最新区块哈希值;另一种是IBCPacketTx 交易,这种交易则可以证明某个数据包确实由发送者的应用程序,通过默克尔证明机制(Merkle-proof)传送到了最新区块的哈希值上。 - -通过将IBC机制分离成两个单独的交易,即IBCBlockCommitTx 交易与 IBCPacketTx 交易,我们可以让接收方链的本地费用市场机制,来决定承认哪个数据包,与此同时还能确保发送方的完全自由,让其自行决定能够传出的数据包数量。 - -![Figure of Zone1, Zone2, and Hub IBC without acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_without_ack.png) - - - - -在上述案例中,为了更新"枢纽"上"分区1"的区块哈希(或者说"分区2"上"枢纽"的区块哈希),必须将IBCBlockCommitTx交易的"分区1"区块哈希值发布到"枢纽"上(或者将该交易的"枢纽"区块哈希值发布到"分区2"中)。 - -_关于两种IBC交易类型,详细请参见 [IBCBlockCommitTx](#ibcblockcommittx) 和 [IBCPacketTx](#ibcpacketcommit) - -## 用例 - -### 分布式交易所 - -比特币借助大量复制来增加分布式账本的安全性。用类似的方式,我们可以在区块链上运行交易所,来降低其受内部及外部攻击的可能性。我们称之为去中心化交易所。 - -现今,加密货币社区认为的去中心化交易所基于"跨链原子事务"交易( AXC 交易)。通过AXC交易,两条不同链上的两个用户可以发起两笔转账交易,交易在两个账本上要么一起提交执行,或者两个账本都不执行(即交易的原子性)。比如,两位用户可以通过AXC交易来实现比特币和以太币之间的交易(或是在不同账本上的任意两种代币),即使比特币和以太坊的区块链之间并没有彼此连接。AXC 交易模式下的交易所用户双方不需要彼此信任,也不用依赖交易匹配服务。其弊端是,交易双方必须同时在线才能进行交易。 - -另一种去中心化交易所是进行大量复制的具有独立区块链的分布式交易所。该种交易所的用户可以提交限价订单并关闭他们的计算机,交易可以在用户离线状态下执行。区块链将会代表交易者去完成匹配和交易。 - -一个中心化的交易所可以构建一个有大交易量的限价交易的买卖盘账目,以此来吸引更多的交易者。在交易所领域,流动性会引发更多流动性,因此在交易所业务中,其具有的网络效应也愈发明显(或者说至少产生了"赢家通吃"效应)。目前加密货币交易所 Poloniex 以每24小时2,000万美元的交易量排名第一, Bitfinex 则每24小时500万美元的交易额位列第二。在这种强大的网络效应之下,基于AXC的去中心化交易所的成交量是不太可能超过中心化交易所。去中心化交易所要想和中心化交易所一争高下,就需要支持以限价订单构成的具有深度的交易买卖盘账目的运行。而只有基于区块链的去中心化交易所可以实现这一点。 - -Tendermint提供的快速交易执行是另一大优势。Cosmos的内部网络可以在不牺牲一致性的前提下优先快速的确定最终性,来实现交易的快速完成 —— 同时针对交易订单交易,以及IBC(跨区块链通信)代币与其他网络的交易。 - -综上,根据现有加密货币交易所的情况,Cosmos的一项重大应用就是去中心化交易所(称为 Cosmos DEX)。其交易吞吐能量和委托延时可以与那些中心化交易所媲美。交易者可以在各方离线的状态下提交限价订单。并且,基于Tendermint,Cosmos枢纽以及IBC的情况下,交易者可以快速地完成在交易所及其他网络的资金进出。 - -### 作为其他加密货币的纽带 - -特权分区可以作为和其他加密货币挂钩的代币来源。这种挂钩类似Cosmos枢纽与分区之间的关系,两者都必须及时更新彼此最新的区块链,从而验证代币已经从一方转移到另一方。Cosmos网络上挂钩的”桥接分区“要和中心以及其他加密币保持同步。这种间接通过”桥接分区“可以保持枢纽逻辑的简洁。并且不必要了解其他的链上共识战略,如比特币工作量证明挖矿机制。 - -#### 向Cosmos枢纽发送代币 - -每个挂钩桥接分区的验证人都会在基于Tendermint公式的区块链之上,运行带有特殊的ABCI桥接应用程序,但同时也会运行一个原有区块链的“全节点”。 - -在原有区块链挖出新区块时,桥接分区验证人员将通过签署和分享起始点区块链的提示,各自局部视角可以达成一致。当一个桥接分区收到原有区块链的支付时(如在以太坊或比特币等PoW机制的链上有足够数目的确认),则在该桥接分区上创建具有该对应账户的余额。 - -就以太坊而言,桥接分区可以和Cosmos枢纽共享相同的验证人。以太坊方面(原本区块链),一个桥接合约将允许以太拥有者通过将以太币发送到以太坊的桥接分区的桥接合约上。一旦挂桥接合约接收到以太币,以太币就不能被撤回,除非从桥接分区接收到对应的IBC数据包。桥接合约跟随桥接分区的验证组,它可能与Cosmos枢纽的验证人组相同。 - -就比特币而言,概念是相似,除了代替一个桥接合约,每个UTXO将由一个门限多重签名P2SH数据库限制。由于P2SH系统的限制,签名者不能与Cosmos枢纽的验证人组相同。 - -#### 从Cosmos枢纽提出代币 - -桥接分区上的以太币(“桥接以太币”)可以在枢纽间转进,转出,完成传送到特定以太坊提取地址后,转出的“桥接以太币”被彻底删除。一个IBC消息可以证明桥接分区上的交易,这个消息将被公布到以太坊桥接合约中,以便以太币被取出。 - -就比特币而言,严谨的交易脚本系统让IBC币的镜像转换机制很难实现。每个UTXO都有自己的特定的脚本,所以当比特币履约签名者发生变化时,每个UTXO都必须迁移到新的UTXO。一个解决方案是根据需要,压缩和解压缩UTXO-set,以保持UTXO的总数量下降。 - -#### 挂钩区完全责任制 - -这类挂钩合约存在风险的风险是,可能会出现恶意的验证人组。如果拜占庭投票权超过⅓,就会造成分叉,即从以太坊桥接合约中提取以太币的同时,还能保持桥接分区中的挂钩以太币不变。甚至,如果拜占庭投票权超过⅔,可能会有人直接通过脱离原始桥接分区的桥接逻辑,对发送以太币发到桥接合约中的帐户下手,盗取以太币。 - -如果将这个桥接方法完全设计成责任制,就有可能解决这一问题。比如,枢纽及起始点的全部IBC包裹可能需要先通过桥接分区的认可,即让枢纽或起始点中的桥接合约对桥接分区的所有状态转换进行有效验证。枢纽及起始点要允许桥接分区的验证人提供抵押物,而侨界合约的代币转出需要延时(且抵押品解绑时间也要足够长),从而让单独的审计人有时间发起任何的质询。我们会把这一系统的设计说明以及执行方式开放,作为未来Cosmos改善的提议,以待Cosmos枢纽的管理系统审批通过。 - -### 以太坊的扩展 - - -众所周知,扩展问题是一直困扰着以太坊的问题。目前以太坊节点会处理节点上每笔交易,并且存储所有的状态[参考](https://docs.google.com/presentation/d/1CjD0W4l4-CwHKUvfF5Vlps76fKLEC6pIwu1a_kC_YRQ/mobilepresent?slide=id.gd284b9333_0_28).。 - -Tendermint提交区块的速度比以太坊工作量证明要快,所以由Tendermint共识推动且使用桥接以太币运行的以太坊虚拟机分区能够强化太坊区块链的性能。此外,虽然Cosmos枢纽及IBC包裹机制不能实现每秒合约逻辑的执行,但是它可以用来协调不同分区中以太坊合约之间的代币流通,通过分片方式为以代币为中心的以太坊扩展奠定基础。 - -### 多用一体化 - -Cosmos分区可以运行任意的应用逻辑,应用在分区创建时设定好,可通过管理者可以不断更新。这种灵活度使得Cosmos分区可以作为其他加密币的挂钩载体,比如以太坊或比特币,并且它还能和这些区块链的衍生品挂钩,利用同样的代码库,而在验证程序及初始分配有所区分。这样就允许多种现有加密币框架得以运行,如以太坊、Zerocash、比特币、CryptoNote等等,将其同Tendermint Core结合,成为通用网络中性能更优的共识引擎,为平台之间提供更多的交互机会。此外,作为多资产区块链,每笔交易都有可能包含多个输入输出项,其中每个输入项都可以是任意代币,使Cosmos直接成为去中心化交易所,当然这里假设交易订单通过其他平台进行匹配。替代方案是,让分区作为分布式容错交易所(包含买卖盘账目),这算是对中心化加密币交易所之上的严格改进——现行交易所在过去时常发生被攻击的事件。 - -分区也可以作为区块链版的企业及政府系统,其原本由一个或多个组织运行的特定服务,现在作为ABCI应用在某个分区上运行,从而在不放弃对底层服务控制的前提下,维持公共Cosmos网络的安全性及交互性。所以,Cosmos或可为那些既想使用区块链技术,又不愿彻底放弃控制权给分布式第三方的人,提供绝佳的运行环境。 - -### 缓解网络分区问题 - -有人认为像Tendermint这种支持一致性的共识算法有一个重大问题,就是网络分区会导致没有任何一个分区会拥有超过⅔的投票权(比如超过⅓投票权在线下),这会中断共识。而Cosmos架构可以缓解这个问题,它可以使用全球中心,同时,各分区实行地区自治,然后让每个分区的投票权按照正常的地域位置进行分配。如,一般范例就有可能是针对个别城市或地区的,让他们各自运行自己分区的同时,还能共享共同的枢纽(比如Cosmos枢纽),并且在临时的网络分区导致的中断期间,也可以继续维持地区自治活动。注意,这样一来在设计稳健的联邦式容错系统过程中,就可以去考虑真实的地理、政治及网络拓扑的特征了。 - -### 联邦式名称解析系统 - -NameCoin是首批试图通过比特币技术解决名称解析问题的区块链之一。不过,这个方案存在一些不足。 - -例如,我们可以通过Namecoin来验证@satoshi(聪)这个号是在过去某个时间点用特定公钥进行注册的。但是,该公钥是否更新过我们就不得而知了,除非将该名称最后一次更新之前的所有全部下载。这一点是比特币UTXO交易模式中默克尔化模型的局限性导致的,这类模型中只有交易(而非可变的应用状态)会以默克尔化加入到区块哈希中。它让我们得在之后用更新证明名称的存在,而非不存在。因此,我们必须依靠全节点才能明确这个名称的最近的值,或者花费大量资源下载整个区块链。 - -即使在NameCoin上运用默克尔化的搜索树,其工作量证明的独立性还是会导致轻客戸端的验证出现问题。轻客戸端必须下载区块链中所有区块头的完整备份(或者至少是自其最后的名称更新的所有区块头)。这意味着带宽需要随着时间做线性的扩展。 [21]此外,在工作量证明制度使区块链上的名称更改需要等额外的工作量证明验证确认才能进行,它在比特币上可能要花费一个小时。 - - -有了Tendermint,我们只需用到由法定数量验证人签署(通过投票权)的区块哈希,以及与名称相关的当前值的默克尔证明。这点让简易、快速、安全的轻客戸端名称值验证成为可能。 - -在Cosmos中,我们可以利用这个概念并延伸。每一个在Cosmos上的名称注册都能有一个相关的最高级别域名(TLD),比如".com"或者".org"等,而且每个名称注册分区都有自己的管理和登记规则。 - -## 发行与激励 - -### Atom 代币 - -Cosmos 枢纽是多资产分布式账本,它有自己的代币,是 Atom 。Atom 是 Cosmos 枢纽唯一的权益代币。Atom是持有人投票、验证或委托给其他验证人的许可证明,就像以太坊上的以太币一样,Atom也可以用来支付交易费以减少电子垃圾。额外的通胀 Atom 和区块交易费用就作为激励分给验证人及委托验证人。 - -`BurnAtomTx`交易可以用来恢复储蓄池中任意比例的代币。 - -#### 众筹 - -创世块上的 Atom 代币及验证人的初次分发是 Cosmos 众售参与者持有75%,预售参与者持有5%,Cosmos网络基金会持有10%,ALL IN BITS, 集团持有10%。从创世块开始,Atom 总量的1/3将作为奖励发放给每年担保持有的验证人以及委托人。 - -更多细节见 [Cosmos Plan](https://github.com/cosmos/cosmos/blob/master/PLAN.md) - -### 验证人的数量限制 - -与比特币或其他工作量证明区块链不同的是, 由于通信的复杂性增加, Tendermint 区块链会随着验证人的增加而变慢。幸运的是, 我们可以支持足够多的验证人来实现可靠的全球化分布式区块链, 并具有非常快的交易确认时间。 而且随着带宽、存储和并行计算容量的增加, 我们将来能够支持更多的验证人。 - -在创世日, 验证人的最大数量将设置为 100, 这个数字将以13% 的速度增长10年, 最终达到300位。 - -``` -Year 0: 100 -Year 1: 113 -Year 2: 127 -Year 3: 144 -Year 4: 163 -Year 5: 184 -Year 6: 208 -Year 7: 235 -Year 8: 265 -Year 9: 300 -Year 10: 300 -... -``` - -### 成为创世日后的验证人 - - -Atom 持有者可以通过签署和提交 `BondTx` 交易成为验证人。抵押的 atom 数量不能为零。任何人任何时候都成为验证人, 除非当前验证人组的数量超过了最大值。在这种情况下, 只有当持有 atom 的数量大于现有验证人中持有有效 atom 数量的最少者, 该交易才有效, 其中有效 atom 包括受委托的 atom。当一个新的验证人以这种方式替换现有的验证人时, 现有的验证人将离线, 其所有的 atom 和受委托的 atom 进入解绑状态。 - -### 对验证人的惩罚 - - -对于任何有意或无意的偏离认可协议的验证人, 必须对其施加一定的惩罚。有些证据立即可予受理, 比如在同样高度和回合的双重签名, 或违反 "预投票锁定" (Tendermint 协商一致议定书的规则)。这样的证据将导致验证人失去其良好的声誉, 其绑定的 atom 以及在储备池中的比例份额 – 统称为 “权益” – 将被大幅削减。 - -有时, 由于区域网络中断、电源故障或其他原因, 验证人将不可用。如果在过去任意时间点的 `ValidatorTimeoutWindow` 块中, 验证人的提交投票不包括在区块链中超过 `ValidatorTimeoutMaxAbsent` 次, 该验证人将离线, 并减少 `ValidatorTimeoutPenalty` (默认 1%) 的权益。 - - -一些 "恶意" 行为在区块链上并没有产生明显的证据。在这些情况下, 如果存在多数的协商一致, 则验证人可以在带外协调,强制将这些恶意验证人超时。 - - -如果 Cosmos 枢纽 因为超过⅓的投票权离线而出现了中止情况,或者说超过⅓的投票权审查到进入区块链的恶意行为,这时候枢纽就必须借助硬分叉重组协议来恢复。(详见“分叉与审查攻击”) - -### 交易费用 - - -Cosmos 枢纽验证人可以接受任何种类的代币或组合作为处理交易的费用。每个验证人可自行设置兑换率, 并选择其想要的交易, 只要不超过 `BlockGasLimit`, 每隔 `ValidatorPayoutPeriod` (默认为1小时) 时间会根据权益相关人绑定的 Atom 比例进行分配。 - - -在所收取的交易费用中, `ReserveTax` (默认 2%) 将存入储备池来增加储备量, 增加 Cosmos 枢纽的安全性和价值。这些资金也可以按照治理系统的决策进行分配。 - - -将投票权委托给其他验证人的 Atom 持有人会支付一定佣金给委托方,而这笔费用可以由每个验证人进行设置。 - -### 激励黑客 - -Cosmos 枢纽的安全取决于底层验证人的安全性和委托人的委托选择。为了鼓励发现和早期报告发现的漏洞, Cosmos 枢纽鼓励黑客通过 `ReportHackTx` 交易发布成功的漏洞, 说, "这个验证人被入侵了,请把赏金发送到这个地址"。这种情况下, 验证人和委托人将挂起闲置, 每个人 `HackPunishmentRatio` (默认 5%) 的 atom 将被削减, `HackRewardRatio` (默认 5%) 的 atom 将发送到黑客的赏金地址作为奖励。验证人必须使用其备份密钥来恢复剩余的 atom。 - -为了防止这一特性被滥用于转移未授权的 atom, 在 `ReportHackTx` 前后,Atom的比例(授权的与未授权的) 将保持不变, 而黑客的赏金将包括一些未授权的 atom (如果有的话)。 - -### 治理规范 - -Cosmos 枢纽是由一个分布式组织管理的, 需要一个明确的治理机制, 以协调对区块链的各种变化, 如系统的参数变量, 以及软件升级和宪法修订. - -所有验证人负责对所有提案进行表决。如果未能及时对提案进行表决, 将导致验证人被自动停用一段时间。 这段时间被称为 `AbsenteeismPenaltyPeriod` (默认1周)。 - - -委托人自动继承其委托的验证人的投票权。这一投票可以被手动覆盖掉。而未绑定的 Atom 是没有投票权的。 - - -每个提案都需要 `MinimumProposalDeposit` 代币的保证金, 这可能是一个或多个代币 (包括atom) 的组合。对于每项提案, 投票者可以投票表决取走保证金。如果超过半数的投票者选择取走保证金 (例如, 因为提案是垃圾信息), 那么保证金就会存入储备池, 除了被燃烧的 atoms。 - -对于每项提案, 投票人可以选择下列方案: - -* 同意 -* 强烈同意 -* 反对 -* 强烈反对 -* 弃权 - -决定采纳(或不采纳)提案需要严格的多数投“同意”或“强烈同意”(或者“反对”及“强烈反对”),但是超过1/3的人投“强烈反对”或“强烈支持”的话就可以否决大多数人的决定。如果大多数人票被否决,那么他们每个人都会失去 `VetoPenaltyFeeBlocks` (默认是一天的区块值 ,税费除外) 作为惩罚,而否决大多数决定的那一方还将额外失去 `VetoPenaltyAtoms` 默认为0.1%)的 Atom 作为惩罚。 - -### 参数变更提案 - - -这里定义的任何参数都可在 `ParameterChangeProposal` 通过后改变。 - -### 赏金提案 - - -通过 `BountyProposal` 后, Atom 可以增发和预留储备池资金作为赏金。 - -### 文本提案 - - -所有其他提案,比如用来更新协议的提案,都会通过通用的 TextProposal 来协调。 - - -## 路线图 - - -详见[计划](https://github.com/cosmos/cosmos/blob/master/PLAN.md). - -## 相关工作 - -过去几年涌现了很多区块链共识及扩展性方面的创新。在这一部分中将挑选一些重要的创新进行简单分析。 - -### 共识系统 - -#### 经典拜占庭容错 - -二十世纪八十年代早期就开始研究存在恶意参与者的共识机制,当时Leslie Lamport创造了”拜占庭容错”这个词,用来指那些图谋不轨参与者做出的恶意的行为,与”死机故障”不同,后者只是处理过程崩溃而已。早期针对同步网络也探索出了一些解决方案,网络信息滞后有一个上限,但实际使用是在高度受控的环境下进行,比如精密飞行仪器以及使用原子钟同步的数据中心。直到九十年代后期,实用拜占庭容错( Practical Byzantine Fault Tolerance ,PBFT)[\[11\]][11]才作为有效的、部分同步的共识算法被逐步推广。它可以容忍⅓参与者有恶意行为。PBFT成为标准算法,催生了各种版本,包括最近由IBM提出并使用于Hyperledger超级账本中的算法。 - -和PBFT相比,Tendermint共识的主要好处在于其改善且简化了的底层结构,其中有些是遵循了区块链典范的结果。Tendermint中,区块必须按顺序提交,这就消除复杂性,节省PBFT中状态变化相关的通信开支。在Cosmos和众多加密币中,如果区块N本身没有提交,那么就不能让它之后的区块N+i(i>=1)提交。如果是通信带宽限制导致了区块N未提交到Cosmos Zone上,那么将通信带宽用于分享选票给区块N+i是一种浪费。如果由于网络分区或者节点掉线导致的区块N未提交,那么N+i就无论如何也不能提交。 - -此外,将交易打包成块可以用默克尔哈希纪录应用程序的状态,而不是用PBFT检查机制进行定时摘要。这可以让轻客戸端更快的提交交易证明,以及更快的跨链通信。 - -Tendermint Core中也优化了很多PBFT特性以外的功能。比如,验证人提交的区块被分割多个部分,对其默克尔化后,然后在节点间广播。这种方式可以提高其广播性能(具体请查看LibSwift [19])。而且,Tendermint Core不会对点对点连接做任何假设,只要点对点间的网络不断开,那么它就能正常运行。 - -#### BitShare委托权益 - -BitShares [\[12\]][12]不是第一个采用权益证明机制(proof-of-stake,PoS)的区块链,但是其对PoS在区块链上的研究与推进做出了巨大的贡献,尤其是在DPoS,即受委托权益证明方面。在BitShares中,相关方选择”见证者”负责提交交易顺序并提交;相关方选择”委托人”负责协调软件更新与参数变化。尽管BitShare在理想环境下能达到很高的性能:100k tx/s,1秒的滞后。每一块只有一个单独的签名,得到交易的最终性的时间比区块时间略长。一个标准的协议仍在开发中。利益相关者可以每天去除或者替换有恶意行为的验证人,但是不同于Tendermint PoS的保证金机制,BitShares没有要求验证人或者代理人的提交押金,如果发生双花攻击的话,押金不会被削减。 - -#### Stellar - -Stellar [\[13\]][13]是以Ripple推行的解决方案为基础,它优化了联邦拜占庭协议模型,其中参与共识过程的并不构成一个固定的全局过程。 相反,每个进程节点组织一个或多个“仲裁片”,每个“仲裁片”构成一组可信进程。 Stellar中的“法定人数”被定义为一组包含至少个个节点的一个仲裁片。从而可达成一致。 - -恒星机制的安全性依赖于任何两个仲裁的交集都是非空的假设,同时,节点的可用性要求至少一个“仲裁片”完全由诚实节点组成。这就需要在“法定人数”的大小上作出妥协:人数过少难以获得共识,人数过多则难以信任所有人。可能难以平衡而不对信任做出重大假设。 除此之外,节点必须维护一定数量的仲裁片以获得足够的容错能力(或者任何“完整节点”,大部分结果的依赖),并且提供一种层级化的配置策略,类似于边界网关协议(Border Gateway Protocol,BGP)。BGP被互联网服务供应商(Internet Service Provider,ISP)用来建立全球路由表,也被浏览器用来管理传输层安全协议(Transport Layer Security,TLS)证书。他们都因为不安全而臭名昭着。 - - -对于Stellar论文中基于Tendermint的PoS批评可以通过本文描述的代币策略来缓解。本文提出了名为 _atom_ 的新的代币,它代表未来交易过程中产生的费用和奖励。 基于Tendermint PoS的优势在于其原理相对简单,同时仍可充分保证和证明安全性。 - -#### BitcoinNG - -BitcoinNG是对比特币的改进,允许垂直扩展,比如增加块的大小而避免带来负面的经济后果,例如对矿工造成的影响严重不成比例。 这种改进是通过将leader选举与交易广播分开来实现的:leader首先通过“微块micro-blocks”的PoW选举,然后leader能够广播交易直至下一个新的“微块”。 这减少了赢得PoW比赛所需的带宽要求,使矿主竞争更公平,并通过允许最后一名矿工提交微块以加快交易提交频率。 - -#### Casper - -Casper [\[16\]][16]是以太坊提出的PoS共识算法。 它的主要操作模式是“预测押注”的一致。 通过让验证者基于他们目前看到的其他投注来迭代地下注他们认为哪个块将提交入区块链。最终性可以立即实现。 - -[链接](https://blog.ethereum.org/2015/12/28/understanding-serenity-part-2-casper/). 这是Casper团队研究的一个热点领域。 挑战在于构建一个可以证明是一个演化稳定策略的投注机制。 与Tendermint相比,Casper的主要优势可能在于提供“可用性超越一致性” — 达成共识不需要超过50%的投票权 — 可能是以提交速度或实现复杂性为代价的。 - -### 水平扩展 - -#### Interledger协议 - -Interledger协议(The Interledger Protocol,ILP)[\[14\]][14]不是一种严格的扩展方案。 它通过一个松散耦合的双边关系网络,提供一种指定的跨不同账本系统交互操作。像闪电网络一样,ILP的目的是实现支付,但是它特别关注跨账本类型的支付,并扩展原子事务的处理机制,使得事务的处理不仅支持哈希锁,而且还包括法定人数的公证人(称为原子运输协议)。 后者在账本间交易中实施原子性的机制与Tendermint的轻客户SPV机制类似,因此比较ILP和Cosmos / IBC之间的区别是有必要的,具体见下文。 - -1.ILP不支持连接器公证员的变更,也不允许公证员之间有灵活的权重。 另一方面,IBC是专门为区块链设计的,验证人可以拥有不同的权重,并且随着区块链的发展,成员可以随时更改。 - -2.与闪电网络一样,ILP中的接收人必须在线才能向发起人发送确认。 在IBC代币传输中,接收者所在区块链的验证人集合负责提供确认,而不是接收用户本人。 - -3.最大的不同在于ILP连接器不需要负责对支付状态保持权威性,然而在Cosmos中,hub的验证人负责IBC代币传输状态和每个zone内持有代币数量的权威性。允许从zone之间的安全地不对称地交换代币是本质的创新。在cosmos中的ILP连接器可以看作是一个持久和最安全的区块链分类账:cosmos hub。 - -4.ILP内的跨账本支付需要一个交易所的指令集的支持。因为不存在从一个分类账到另一个分类账不对称的代币转移,只能做到市场等价物的转移。 - -#### 侧链 - -Sidechains [\[15\]][15]是一种通过使用与比特币区块链“双向挂钩”替代区块链来扩展比特币网络性能的机制。(双向挂钩相当于桥接,在cosmos中被称为“桥接”与市场挂钩区分)。侧链使得比特币可以方便的在比特币区块链和侧链间移动,并允许在侧链上实验新功能。在Cosmos Hub中,侧链和比特币是彼此的轻客户端,在比特币区块链和侧链间移动时使用SPV证明。当然,由于比特币使用PoW,以比特币为中心的侧链遭受许多由于PoW作为共识机制的引起的问题和风险。而且,这是一个比特币利益最大化的解决方案,并不像Cosmos那样本地支持各种代币和zone间网络拓扑结构。但是,双向挂钩的核心机制原则上与Cosmos所采用的机制相同。 - -#### 以太坊扩展性的努力 - - -以太坊目前正在研究许多不同的战略,将以太坊区块链的状态分区化,以解决可扩展性的需求。 这些努力的目标是在共享状态空间之上,维持当前以太坊虚拟机提供抽象层。 目前,多项研究工作正在进行。[\[18\]][18] [\[22\]][22] - -##### Cosmos vs Ethereum 2.0 Mauve - - -Cosmos 和 Ethereum 2.0 Mauve [\[22\]][[22] 有不同的设计理念。 - -* Cosmos是针对代币儿 Mauve 是关于扩大计算能力。 -* Cosmos不仅限于 EVM, 所以即使不同的VM也可以交互。 -* Cosmos 让zone的创建者决定验证人。 -* 任何人都可以在Cosmos中建立新的zone(除非管理者另做决定)。 -* hub与zone的失效隔离,所以全局的代币不变量可以保持。 - -### 普遍扩展 - -#### 闪电网络 - - -闪电网络被设计成一种代币传输网络,在比特币区块链(及其他公有区块链)上一层运行,通过把大部分交易从共识分类账之外转移到所谓的“ 付款渠道“。 这通过链上加密货币脚本来实现,这些脚本使双方能够进入双方持有的状态化合同,通过共享数字签名来更新状态,并且在合同结束后最终通过在区块链上发布证据,这种机制首先受到跨链原子互换交易的欢迎。 通过与多方开通支付渠道,闪电网络的参与者可以成为集中点,为其他人的支付提供路由,从而导致支付渠道网络的完全联通,其代价是绑定在支付渠道上的资金。 - -虽然闪电网络也可以轻松地跨越多个独立的区块链,并借助交易市场实现价值转移,但它不能实现从一个区块链到另一个区块链的非对称代币交易。 这里描述的Cosmos网络的主要优点是实现直接的代币交换。 也就是说,我们希望支付渠道和闪电网络将会与我们的代币传输机制一起被广泛采用,从而节省成本和保护隐私。 - -#### 隔离验证人 - - -隔离见证是一个比特币改进建议BIP,[链接](https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki) 旨在将每块交易吞吐量提高2倍或3倍,同时使新节点能更快同步区块。 这个解决方案的亮点在于它如何在比特币当前协议的局限下,允许软分叉升级(例如,具有旧版本软件的客户端将在升级后仍可继续运行)。 Tendermint作为一个新的协议没有设计的限制,所以它具有不同的扩展优先级。 TendermintBFT的循环算法,主要基于加密签名而不是采矿,这种算法允许通过多个并行区块链进行水平扩展,而更常规、更频繁的区块提交也允许垂直扩展。 - - - -## 附录 - -### 分叉问责制 - -经过精心设计的共识协议应该能够在超出容错能力或者共识出错的情况下为系统提供一定的保障。这在能够通过拜占庭行为获取实质经济回报的金融系统中显得尤为必要。_分叉问责制_ 就属于这种非常重要的保障机制,这种制度会使一个导致共识出错的进程(例如使协议客户端开始接受不同值——即分叉)被识别出来,并且根据协议规则对其进行惩罚,甚至将其移送至司法系统处置。但当司法系统不可靠或者诉讼费极其昂贵时,为了让验证人们都参与到这个机制当中,该制度会强制要求他们建立安全保证金,一旦检测到恶意行为,那么这些保证金将会被罚没或者削减[\[10\]][10]. - - - -注意这与比特币有所不同,由于网络非同步与局部哈希碰撞的概率特性,比特币的分叉是定期出现的。因为在很多情况下,恶意分叉与非同步引起的分叉是无法分辨的,除非让矿工为挖到的孤立区块支付隐性的机会成本,否则比特币就很难准确地执行分叉问责制。 - -### Tendermint共识 - - -我们把投票阶段分为 _预投票_ 和 _预提交_ 两个阶段。一个投票可以是用于特定区块,也可以用于 _Nil_。我们把同一轮超过⅔的单个区块的预投票总和称为 _Polka_ ,把同一轮超过⅔的单个区块的预提交总和称为 _Commit_。如果在同一轮中对Nil的预提交超过⅔,他们就进入到下一轮中。 - - -注意到协议中的严格的确定性会引发一个弱同步假设,因为错误的发起者必须被检测到并将其略过。验证人们在为Nil预投票之前会等待一段时间,称之为超时提议,这个超时提议的hi等待时间也会随着每一轮的进行而递增。每一轮的进行都是完全不同步的,在这个过程中,只有当验证人收听到超过⅔的网络投票才能进入到下一轮中。实际上,它需要极其强大的阻碍才能阻挠这个弱同步假设(导致无达成共识,无法提交区块),而且通过每个验证人超时提议(TimeoutPropose)的随机值还可更加提高这么做的难度。 - -另一个附加的约束条件,或者叫锁定条例,它能确保网络最终在每个高度只提交一个区块。任何试图在给定高度提交超过一个区块的恶意行为都会被识别出来。首先,一个区块的预提交必须被认为是正当的,并且以Polka的形式提交。如果验证人已经准备在R_1轮中预提交一个区块,我们称他们锁定了这个区块,并且然后用于验证R_2轮新预提交动作的Polka必须进入R_polka轮,其中 R_1 < R_polka <= R_2。其次,验证人们必须为他们锁定的区块提议并且(或者)预投票。这两个条件共同作用,确保了验证人在对其正当性没有充分论证的情况下不能进行预提交操作,并且保证已经完成预提交的验证人不能再为其他东西的预提交贡献证明投票。这样不但可以保证共识算法的安全性,还能保证它的活跃度。 - - -关于这一协议的全部细节参考[这里](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - -### Tendermint轻客户端 - -因为生成侧链(一个分叉)意味着至少⅓的担保权益被罚没,Tendermint权益证明(Tendermint-PoS)取消了同步所有区块头的要求。当然,罚没保证金也是需要有人共享分叉证据的,所以轻客戸端就要存储任何它见证的区块哈希提交。另外,轻客戸端可以定期地与验证人组的改变保持同步,以避免出现远程攻击(但是其他解决方案也具有可能性)。 - - -与以太坊类似,Tendermint能够让应用程序在每个区块中嵌入一个全局默克尔根的哈希值,可以简单方便的验证的状态查询,比如查询账户余额、在智能合约中的值,或者未使用交易输出(UTXO)的存在,具体由应用程序的特性决定。 - -### 远程攻击的防御 - - -假设有一个足够有弹性的广播网络采集与一组静态验证人组存在,那么任何区块链分叉都可以被检测到,而且发起攻击的验证人提交的保证金会被罚没。这个由Vitalik Buterin在2014年首次提出的新方法,解决了其他权益证明加密货币的"没有任何相关权益"问题(详见 [相关工作部分](#related-work))。但由于验证人组必须是能够变化的,在较长的时间段内最初的一些验证人会解除j押金绑定,这就使他们可以自由地从创世区块中创建新链,并且因为他们不再有被锁定的保证金,他们将不需要为这个行为支付任何费用。这类攻击被称为远程攻击(LRA),与短程攻击相比,后者对处在押金绑定中的验证人发起的分叉是可以对其进行惩罚的(假设有类似Tendermint共识这样的分叉问责制拜占庭容错算法)。所以远程攻击经常被认为是对权益证明机制的一种危险打击。 - - -幸运的是,远程攻击(LRA)可以用以下的途径来缓解。第一,对于解除绑定的验证人而言(取回抵押保证金并且不再从参与共识中获取费用),保证金在一定时间内不能转移,也可以称其为"解绑周期",这个周期可能长达数周或数月。第二,对于轻客户端的安全性而言,其首次连接到网络时必须根据可信源验证最新的一个区块哈希或者多个最好的区块哈希。这种情况有时被称为"弱主观性"。最后,为了保证安全,必须与最新的验证人组进行频繁的同步,时长与解绑周期一致。这样就确保了轻客戸端在因验证人解绑资金而失去任何权益之前,知道验证人组的变化情况,否则解绑的验证人就会在其绑定的高度后面开始创建新区块来实施远程攻击,以此来欺骗客户端(假设它可以控制足够多的早期私钥)。 - -注意到用这样的方式来对抗远程攻击(LRA)需要对工作量证明(proof-of-work)的原始安全模块进行彻底检查。在工作量证明中(PoW),一个轻客户端可以通过在每一个区块头中运行工作量证明,以此简便地在任何时候与可信任的创始区块的当前高度进行同步。但是,为了对抗远程攻击(LRA),我们需要轻客戸端定期上线追踪验证人组的变动,其中在首次上线时必须格外仔细地根据可靠源来验证从网络采集的信息。诚然,后面这个要求和比特币类似,其协议和软件也必须从可靠源获得。 - - -以上这些为了防止远程攻击的方法,比较好地适用于由Tendermint驱动下的区块链验证人节点以及全节点,因为这些节点需要保持与网络的连接。这些方法同样适用于希望频繁地与网络同步的轻客户端。但是,对于那些不希望频繁接入互联网或者区块链网络的轻客户端来说,还有另一种方法可以解决远程攻击的问题。非验证人节点可以在很长的解绑期内(比如比验证人的解绑期更久)使用代币作为保证金,并且为轻客戸端提供二级证明当前有效性以及过去区块哈希的解决方案。虽然这些代币对于区块链共识的安全性并没有价值,不过他们还是可以为轻客戸端提供强大的保障。如果在以太坊中支持历史区块哈希查询,那么任何人都可以用特定的智能合约来绑定他们的代币,并且提供付费证明服务,从而有效地针对轻客戸端LRA安全问题开发出一个市场。 - -### 克服分叉与审查攻击 - -由于提交区块流程的定义,任何联合后不少于⅓的投票权的节点都可以通过下线或者不广播选票来中止区块链运行。这样的联合也可以通过拒绝包含这些交易的区块来审查特定的交易,尽管这将导致大多数区块提案被拒绝,致使区块提交速率减缓,降低了它的实用性与价值。恶意的联合或许仍然会陆陆续续地广播选票,用阻挠区块链的区块提交来将其逼停,或者使用任何这些攻击的组合攻击。最终,它会通过双重签名或者违反锁定规则来造成区块链分叉。 - -如果一个全球活跃的作恶者也参与进来,就会用可能出现错误的验证组人子集导致速度降低的方法来分割网络。这不只是Tendermint面临的局限性,更确切地说是所有被活跃敌对者控制了网络的共识协议所面临的局限性1。 - - -对于这些类型的攻击,验证人的子集应该通过外部的方式进行协调,以签署选择一个分叉(及关系到的所有证据)与带有签名的验证人的初始子集的重组提案。签署了这样一份重组提案的验证者,将放弃在所有其他分叉上属于他们的保证金。客户端应在重组提案中验证签名以及任何相关的证据,并作出判断或提示终端用户作出决定。例如,一个手机钱包app应该在可能接受任何由一半以上的初始验证人们通过投票权利签署的重组提案时,给予用户安全警告提示。 - -当多于⅓的投票权益是不诚实的时候,一个非同步的拜占庭容错算法步能够达成共识,然而,分叉假设不少于⅓的投票权益已经因不正当的双重签名或者锁定改变而成为不诚实的。因此,签署重组提案是一个协调问题,任何非同步协议都无法解决这个问题(也就是自动的,并且不考虑底层网络的可靠性)。目前,我们通过互联网媒体的社会共识,把重组提案的协调问题留给了用户去协调。验证人必须在签署重组提案之前就确保没有出现网络分割的问题,以此来避免签署两个相冲突的重组提议的情况发生。 - - -假设外部协调媒介和协议是可靠的,对于分叉的担心会比审查攻击要少许多。 - - -除了需要大于⅓的拜占庭投票权益才能启动的分叉和审查制度以外,超过⅔的联合投票权益可能会提交任意、无效的状态。这是任何拜占庭容错算法的共识系统所特有的问题。与利用简单可验证证明来创建分叉的双重签名不同,检测无效状态的提交需要非验证节点来验证整个区块,这意味着非验证节点会保留一份本地的状态副本并执行每一笔交易,然后为他们自己独立计算出状态的根源。一旦检测出来,处理这类故障的唯一方法就是社会共识。打一个比方,在比特币出现问题的情况下,无论是由于软件漏洞造成的分叉(正如2013年3月),还是由于矿工拜占庭行为提交的无效状态(正如2015年7月),由商户、开发者、矿工和其他组织组成的联系紧密的社区所建立起来的社会共识会让他们按照分工来参与到修复网络的工作当中去。此外,由于Tendermint区块链的验证人身份是可识别的,那么如果需要的话,无效状态的提交实际上是可以被法律或其他外部法律体系惩治的。 - -### ABCI说明 - - -ABCI由3种主要的信息类型组成,这三类信息从共识引擎传递到应用程序上,然后应用程序用相应回复信息做出应答。 - - -`AppendTx` 信息是应用程序的主要传递媒介。区块链中的每一笔交易都通过这个信息来传递。应用程序需要验证每笔交易,这将通过接收针对当前状态、应用协议和交易密码证书的AppendTx信息来实现。验证过的交易将需要通过添加数值到键值存储或者更新UTXO数据库的方式来更新应用状态。 - - -`CheckTx`信息与AppendTx信息类似,但它只是为了交易验证。Tendermint Core的内存池会先用CheckTx验证交易有效性,并且只会将有效的交易传递给其他的节点。应用程序会检查交易序列号,如果序列号过期就会根据CheckTx返回一个错误。 - -`Commit`信息是用来计算之后会存入到下一区块头中的当前应用状态的加密提交项。这具有便利的特性。状态的前后矛盾性将会像引起程序错误,从而导致区块链分叉。这也简化了安全轻客戸端的开发,因为默克尔哈希证明可以通过检查区块哈希来加以验证,而区块哈希是由规定人数的验证人们签署的(通过投票权益)。 - - - -此外,ABCI信息允许应用程序保持追踪验证人组的改变,并让应用程序接收诸如高度和提交选票之类的区块信息。 - - -ABCI请求/响应是简单的Protobuf信息。请参考这里的[模式文件](https://github.com/tendermint/abci/blob/master/types/types.proto)。 - -##### AppendTx - * __命令行参数__: - * `Data ([]byte)`: 交易请求信息 - * __Returns__: - * `Code (uint32)`: 回复代码 - * `Data ([]byte)`: 结果字节,如果有的话 - * `Log (string)`: 错误信息 - * __使用__: - 提交并执行一笔交易,如果交易有效,那返回CodeType.OK - -##### CheckTx - * __命令行参数__: - * `Data ([]byte)`: 交易请求信息 - * __Returns__: - * `Code (uint32)`: 回复代码 - * `Data ([]byte)`: 结果字节,如果有的话 - * `Log (string)`: 错误信息 - * __使用__: - 验证一笔交易。这个信息不应该改变应用状态。交易在广播给其他节点前,首先通过CheckTx运行。你可以发起半状态化CheckTx,并在Commit or BeginBlock上清算状态,以允许序列执行同一区块中相关的交易。 - -##### Commit - * __返回值__: - * `Data ([]byte)`: 默克尔根值 - * `Log (string)`: 调试或出错信息 - * __使用__: - 返回当前应用状态。 - -##### Query - * __命令行参数__: - * `Data ([]byte)`: 请求数据 - * __返回值__: - * `Code (uint32)`: 回复代码 - * `Data ([]byte)`: 查询回复字节 - * `Log (string)`: 调试或出错信息 - -##### Flush - * __使用__: - 刷新回复队列。应用types.Application的应用程序无需实施这条信息——这个由项目进行处理。 - -##### Info - * __返回值__: - * `Data ([]byte)`: 信息字节串 - * __使用__: - 返回关于应用程序状态的信息. 应用指定。 - -##### SetOption - * __参数__: - * `Key (string)`: 设置参数 - * `Value (string)`: 参数值 - * __返回值__: - * `Log (string)`: Debug or error message - * __使用__: - 比如,针对内存池的连接可以将键设置为"mode"(模式),值为"mempool"(内存池)。或者针对共识连接,将键设置为"mode",值设置为"consensus"(共识)。其他选项根据可具体应用进行专门设置。 - -##### InitChain - * __参数__: - * `Validators ([]Validator)`: 初始化创世验证人 - * __使用__: - 在创世区块创建时调用 - -##### BeginBlock - * __参数__: - - - * * `Height (uint64)`: 区块刚开始的高度 - * __使用__: - 为新区块的开始提供信号。在附加交易(AppendTxs)前进行调用。 - -##### EndBlock - * __参数__: - * `Height (uint64)`: 结束时的区块高度 - * __返回值__: - * `Validators ([]Validator)`: 具有新选票的变动后的验证人(归零就去除) - * __使用__: - 为区块结束提供信号。在每次提交前所有交易后调用。 - -See [the ABCI repository](https://github.com/tendermint/abci#message-types) for more details - -更多细节请参考 [ABCI知识库](https://github.com/tendermint/abci#message-types)。 - -### IBC数据包交付确认 - - -发送者有很多需要接收链提供数据包交付确认的原因。比如,如果预计目的链会出错,那发送者就可能无法了解目的链的状态。或者,当目的链可能遇到因接收数据包猛烈增多而形成的拒绝服务攻击时,发送者会想要设定数据包超时时限(借助`MaxHeight` 即最大值包域)。 - - -在这些案例中,发送人可以通过在`AckPending`上设置初始数据包状态来要求提供交付确认。然后就由接收链通过包含一个简化的`IBCPacket`的应用默克尔哈希来确认交付。 - -![Figure of Zone1, Zone2, and Hub IBC with acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack.png) - - -首先,一个`IBCBlockCommit`和`IBCPacketTx`是被上传到“枢纽”上用来证明"分区1"上的`IBCPacket`的存在的。假设`IBCPacketTx`的值如下: - -- `FromChainID`: "Zone1" -- `FromBlockHeight`: 100 (假设) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 (say) - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 (假设 "枢纽" 当前高度为 300) - - `Payload`: <一个"代币"的有效负荷字节> - - -- `FromChainID`: "Zone1" -- `FromBlockHeight`: 100 (假设) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 (假设) - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 (假设"枢纽"目前的高度是300) - - `Payload`: <一个"代币"的有效负荷字节> - - -其次,一个`IBCBlockCommit` 和 `IBCPacketTx`被传输都“分区2”上用来证明`IBCPacket`在“枢纽”上的存在。假设`IBCPacketTx`的值如下: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 300 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 - - `Payload`: <一个"代币"相同的有效负荷字节> - - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 300 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 - - `Payload`: <一个"代币"相同的有效负荷字节> - - -接下来,"Zone2"必须将缩写的来显示`AckSent`的最新状态包添加到应用程序状态哈希中。 -`IBCBlockCommitand` 和`IBCPacketTx` 会传输到“枢纽"上来证明简化的`IBCPacket` 存在于"分区2"上。假设`IBCPacketTx` 的值如下: - -- `FromChainID`: "Zone2" -- `FromBlockHeight`: 400 (假设) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckSent` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <一个"代币"相同的有效负荷字节的哈希值> - - -- `FromChainID`: "Zone2" -- `FromBlockHeight`: 400 (假设) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckSent` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <一个"代币"相同的有效负荷字节的哈希值> - - - -最后,“枢纽”必须更新从`AckPending` 到`AckReceived`的数据包状态。这个新完成状态的证明应该返回到“分区2”上。假设`IBCPacketTx`的值如下: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 301 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckReceived` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <The hash bytes of the same "coin" payload> - - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 301 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckReceived` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <相同"代币"有效负荷的哈希字节> - - - -与此同时,“分区1”会假设“代币”包的交付已经成功,除非"枢纽"上有证据给出相反的证明。在上述例子中,如果"枢纽"没有从"分区2"接收到第350个区块的`AckSent` 状态,那么它就会自动将其设置为`Timeout`(超时)。这个超时的证据可以贴回到"Zone1"上,然后所有代币都会被返还。 - -![Figure of Zone1, Zone2, and Hub IBC with acknowledgement and timeout](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack_timeout.png) - -### 默克尔树及默克尔证明的说明 - -Tendermint/Cosmos生态支持的两种默克尔树:简单树和IAVL+树。 - -#### 简易版默克尔树 - - -简易版默克尔树针对基础的静态列表。如果项目的数量不是2的次方,那么有些树叶就会在不同的层上。简易树试图让树的两侧在同一高度,但是左边可能会稍大一点。这种默克尔树就是用于一个区块交易的默克尔化的,而顶层元素就是应用状态的根。 - -``` - * - / \ - / \ - / \ - / \ - * * - / \ / \ - / \ / \ - / \ / \ - * * * h6 - / \ / \ / \ - h0 h1 h2 h3 h4 h5 - - A SimpleTree with 7 elements -``` - -#### IAVL+树 - -IAVL+数据结构的目的是永久储存应用状态中的密钥对,这样就可以对确定的默克尔根哈希进行高效的运算。这个树的平衡通过 AVL算法的变体来实现,所有运行都是O(log(n))。 - -在AVL树中,任意节点的两个子树的高度至多有一处不同。无论在什么时候这种情况都是与更新相违背的,这个树都会通过创造O(log(n))新节点(指向旧树上未修改的节点)来再次达到平衡。在初始的AVL算法中,内部节点也可以保留密钥值对。AVL+算法(注意这里有个"+"号)对AVL算法进行了修改,来维持所有数值都在树叶节点上,同时还只需采用分支-节点来存储密钥。这样在维持较短的默克尔哈希轨迹对的同时,还简化了算法。 - -AVL+树类似于以太坊的[帕氏树](http://en.wikipedia.org/wiki/Radix_tree)。其中也有一定的折中。密钥不需要在嵌入到IAVL+树之前生成哈希,所以这就为密钥空间提供了较快的命令迭代,这会为很多应用程序带来好处。逻辑实现更简单,只需要内部节点和树叶节点这两种节点类型。作为一个平衡的二叉树,其默克尔证明平均更短。而另一方面,IAVL+树的默克尔根有取决于命令的更新。 -我们将支持额外有效的默克尔树,比如当二元变量可用时的以太坊帕氏树。 - - -### 交易类型 - - -在标准的执行中,交易通过ABCI界面涌入Cosmos Hub的应用程序中。 - -Cosmos Hub将接收几类主要的交易类型,包括`SendTx`, `BondTx`, `UnbondTx`, `ReportHackTx`, `SlashTx`, `BurnAtomTx`, -`ProposalCreateTx`,以及`ProposalVoteTx`(发送交易、绑定交易、解绑交易、攻击报告交易、削减交易、Atom燃烧交易,创建提案交易、以及提案投票交易),这些都不需要加以说明,会在未来版本的文档中加以备案。这里我们主要列举两个主要的IBC交易类型: `IBCBlockCommitTx` 以及`IBCPacketTx`(即IBC区块提交交易以及IBC数据包交易) - -#### IBCBlockCommitTx - - -IBCBlockCommitTx 交易主要由这些组成: - -- `ChainID (string)`: 区块链ID -- `BlockHash ([]byte)`: 区块哈希字节,就是包括应用程序哈希默克尔根 -- `BlockPartsHeader (PartSetHeader)`: 区块部分设置的头字节,只用于验证投票签名 -- `BlockHeight (int)`: 提交高度 -- `BlockRound (int)`: 提交回合 -- `Commit ([]Vote)`: 超过⅔的包括区块提交的Tendermint预提交投票 -- `ValidatorsHash ([]byte)`: 新验证组的默克尔树根哈希 -- `ValidatorsHashProof (SimpleProof)`: 在区块哈希中证明验证人哈希的简易树默克尔证明 -- `AppHash ([]byte)`: IAVL树,应用程序状态的默克尔树根哈希 -- `AppHashProof (SimpleProof)`: 在区块哈希中验证应用程序哈希的简易版默克尔树证明`AppHash` against the `BlockHash` - -#### IBCPacketTx - - -`IBCPacket` 由下列项组成: - -- `Header (IBCPacketHeader)`: 数据包头 -- `Payload ([]byte)`: 数据包有效负荷字节。可选择。 -- `PayloadHash ([]byte)`: 数据包字节哈希。可选择。 - - - -有效负荷`Payload`或有效负荷哈希`PayloadHash`必须存在一个。`IBCPacket` 的哈希就是两个项的简易版默克尔根,即头和有效负荷。没有完整有效负荷的`IBCPacket` 被称作 _缩写版包_ 。 - - - -`IBCPacketHeader`由下列项组成: - -`SrcChainID (string)`: 源区块链 ID -DstChainID (string)`: 目标区块链ID -Number (int)`: 所有数据包的唯一数字 -Status (enum)`:可以是AckPending,AckSent,AckReceived,NoAck,或Timeout任意一个 -Type (string)`: 种类根据应用程序决定。Cosmos保留"coin"(币)包种类。 -`MaxHeight (int)`: 如果状态不是这个高度给出的`NoAckWanted` 或者`AckReceived` ,那么状态就算超时。可选择。 - -An `IBCPacketTx` transaction is composed of: - -- `FromChainID (string)`: The ID of the blockchain which is providing this - packet; not necessarily the source -- `FromBlockHeight (int)`: The blockchain height in which the following packet - is included (Merkle-ized) in the block-hash of the source chain -- `Packet (IBCPacket)`: A packet of data, whose status may be one of - `AckPending`, `AckSent`, `AckReceived`, `NoAck`, or `Timeout` -- `PacketProof (IAVLProof)`: A IAVLTree Merkle-proof for proving the packet's - hash against the `AppHash` of the source chain at given height - -`IBCPacketTx` 交易有下列项组成: -- `FromChainID (string)`: 提供给这个数据包的区块链ID,不是源所必须的 -- `FromBlockHeight (int)`: 区块链高度,其中接下来的包会包含在(默克尔化的)源链的区块哈希中 -- `Packet (IBCPacket)`:数据包, 其状态可能是`AckPending`, `AckSent`, `AckReceived`, `NoAck`, 或者 `Timeout`其中的一个 -- `PacketProof (IAVLProof)`: IAVL树默克尔证明,用于验证在给定高度下的源链应用哈希中的数据包哈希 - - -通过"Hub",将数据包从"Zone1"发送到"Zone2"的序列,描述在{Figure X}函数中。首先,一个`IBCPacketTx`会向"Hub"证明数据包是包含在"Zone1"的应用程序状态中。然后,另一个`IBCPacketTx` 会向"Zone2"证明数据包包含在"Hub"的应用程序状态中。在这个过程中,`IBCPacketTx` 的字段是相同的:`SrcChainID`永远是"Zone1",而`DstChainID` 永远是"Zone2"。 - -The `PacketProof` must have the correct Merkle-proof path, as follows: - -`PacketProof` 必须有正确的默克尔证明路径,如下: - -``` -IBC/<SrcChainID>/<DstChainID>/<Number> - -``` - - -当“Zone1”要通过“Hub”将数据包传送到“Zone2”中,无论数据包是否在“Zone1”、“Hub”、或者“Zone2”中默克尔化了,`IBCPacket`数据都是相同的。唯一易变的字段是为追踪交付的`Status`。 - -## 鸣谢 ############################################################ - -我们为所有朋友欲同行们在概念成型与检查方面给予的帮助,以及对我们在Tendermint与Cosmos工作中的大力支持,表示衷心地感谢。 - -* [Zaki Manian](https://github.com/zmanian) of - [SkuChain](https://www.skuchain.com/) provided much help in formatting and -wording, especially under the ABCI section -* [Jehan Tremback](https://github.com/jtremback) of Althea and Dustin Byington - for helping with initial iterations -* [Andrew Miller](http://soc1024.com/) of [Honey - Badger](https://eprint.iacr.org/2016/199) for feedback on consensus -* [Greg Slepak](https://fixingtao.com/) for feedback on consensus and wording -* Also thanks to [Bill Gleim](https://github.com/gleim) and [Seunghwan - Han](http://www.seunghwanhan.com) for various contributions. -* __Your name and organization here for your contribution__ - -* [SkuChain](https://www.skuchain.com/)的[Zaki Manian](https://github.com/zmanian)在格式与措辞方面提供了很多帮助,尤其在ABCI部分。 -* Althea的[Jehan Tremback](https://github.com/jtremback)和Dustin Byington在初始迭代方面的帮助。 -* [Honey - Badger](https://eprint.iacr.org/2016/199)的 [Andrew Miller](http://soc1024.com/) 在共识部分给予的反馈。 -* [Greg Slepak](https://fixingtao.com/)对共识部分的反馈以及在措辞方面的帮助。 -* 同时还要感谢 [Bill Gleim](https://github.com/gleim)和 [Seunghwan - Han](http://www.seunghwanhan.com)在多方面的支持与贡献。 -* **此处还有您及您的组织对本文的贡献。 - -## 引用 - -[1]: https://bitcoin.org/bitcoin.pdf -[2]: http://zerocash-project.org/paper -[3]: https://github.com/ethereum/wiki/wiki/White-Paper -[4]: https://download.slock.it/public/DAO/WhitePaper.pdf -[5]: https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki -[6]: https://arxiv.org/pdf/1510.02037v2.pdf -[7]: https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf -[8]: https://github.com/tendermint/tendermint/wiki -[9]: https://groups.csail.mit.edu/tds/papers/Lynch/jacm85.pdf -[10]: https://blog.ethereum.org/2014/01/15/slasher-a-punitive-proof-of-stake-algorithm/ -[11]: http://pmg.csail.mit.edu/papers/osdi99.pdf -[12]: https://bitshares.org/technology/delegated-proof-of-stake-consensus/ -[13]: https://www.stellar.org/papers/stellar-consensus-protocol.pdf -[14]: https://interledger.org/rfcs/0001-interledger-architecture/ -[15]: https://blockstream.com/sidechains.pdf -[16]: https://blog.ethereum.org/2015/08/01/introducing-casper-friendly-ghost/ -[17]: https://github.com/tendermint/abci -[18]: https://github.com/ethereum/EIPs/issues/53 -[19]: http://www.ds.ewi.tudelft.nl/fileadmin/pds/papers/PerformanceAnalysisOfLibswift.pdf -[20]: http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf -[21]: https://en.bitcoin.it/wiki/Thin_Client_Security -[22]: http://vitalik.ca/files/mauve_paper.html - -* [1] Bitcoin: https://bitcoin.org/bitcoin.pdf -* [2] ZeroCash: http://zerocash-project.org/paper -* [3] Ethereum: https://github.com/ethereum/wiki/wiki/White-Paper -* [4] TheDAO: https://download.slock.it/public/DAO/WhitePaper.pdf -* [5] Segregated Witness: https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki -* [6] BitcoinNG: https://arxiv.org/pdf/1510.02037v2.pdf -* [7] Lightning Network: https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf -* [8] Tendermint: https://github.com/tendermint/tendermint/wiki -* [9] FLP Impossibility: https://groups.csail.mit.edu/tds/papers/Lynch/jacm85.pdf -* [10] Slasher: https://blog.ethereum.org/2014/01/15/slasher-a-punitive-proof-of-stake-algorithm/ -* [11] PBFT: http://pmg.csail.mit.edu/papers/osdi99.pdf -* [12] BitShares: https://bitshares.org/technology/delegated-proof-of-stake-consensus/ -* [13] Stellar: https://www.stellar.org/papers/stellar-consensus-protocol.pdf -* [14] Interledger: https://interledger.org/rfcs/0001-interledger-architecture/ -* [15] Sidechains: https://blockstream.com/sidechains.pdf -* [16] Casper: https://blog.ethereum.org/2015/08/01/introducing-casper-friendly-ghost/ -* [17] ABCI: https://github.com/tendermint/abci -* [18] Ethereum Sharding: https://github.com/ethereum/EIPs/issues/53 -* [19] LibSwift: http://www.ds.ewi.tudelft.nl/fileadmin/pds/papers/PerformanceAnalysisOfLibswift.pdf -* [20] DLS: http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf -* [21] Thin Client Security: https://en.bitcoin.it/wiki/Thin_Client_Security -* [22] Ethereum 2.0 Mauve Paper: http://vitalik.ca/files/mauve_paper.html - -#### 未分类链接 - -* https://www.docdroid.net/ec7xGzs/314477721-ethereum-platform-review-opportunities-and-challenges-for-private-and-consortium-blockchains.pdf.html diff --git a/docs/resources/whitepaper.md b/docs/resources/whitepaper.md deleted file mode 100644 index f77608072..000000000 --- a/docs/resources/whitepaper.md +++ /dev/null @@ -1,1590 +0,0 @@ -# Cosmos - -A Network of Distributed Ledgers - -Jae Kwon <mailto:jae@tendermint.com><br/> -Ethan Buchman <mailto:ethan@tendermint.com> - -For discussions, [join our community chat](https://riot.im/app/#/room/#cosmos:matrix.org)! - -_NOTE: If you can read this on GitHub, then we're still actively developing this -document. Please check regularly for updates!_ - -\[[toc]] - -## Introduction - -The combined success of the open-source ecosystem, decentralized -file-sharing, and public cryptocurrencies has inspired an understanding that -decentralized internet protocols can be used to radically improve socio-economic -infrastructure. We have seen specialized blockchain applications like Bitcoin -[\[1\]][1] (a cryptocurrency), Zerocash [\[2\]][2] (a cryptocurrency for -privacy), and generalized smart contract platforms such as Ethereum [\[3\]][3], -with countless distributed applications for the Ethereum Virtual Machine (EVM) such as Augur (a prediction -market) and TheDAO [\[4\]][4] (an investment club). - -To date, however, these blockchains have suffered from a number of drawbacks, -including their gross energy inefficiency, poor or limited performance, and -immature governance mechanisms. Proposals to scale -Bitcoin's transaction throughput, such as Segregated-Witness [\[5\]][5] and -BitcoinNG [\[6\]][6], are vertical scaling solutions that remain -limited by the capacity of a single physical machine, in order to ensure the -property of complete auditability. The Lightning Network [\[7\]][7] can help -scale Bitcoin transaction volume by leaving some transactions off the ledger -completely, and is well suited for micropayments and privacy-preserving payment -rails, but may not be suitable for more generalized scaling needs. - -An ideal solution is one that allows multiple parallel blockchains to -interoperate while retaining their security properties. This has proven -difficult, if not impossible, with proof-of-work. Merged mining, for instance, -allows the work done to secure a parent chain to be reused on a child chain, -but transactions must still be validated, in order, by each node, and a -merge-mined blockchain is vulnerable to attack if a majority of the hashing -power on the parent is not actively merge-mining the child. An academic review -of [alternative blockchain network -architectures](http://vukolic.com/iNetSec_2015.pdf) is provided for additional -context, and we provide summaries of other proposals and their drawbacks in -[Related Work](#related-work). - -Here we present Cosmos, a novel blockchain network architecture that addresses all -of these problems. Cosmos is a network of many independent blockchains, called -zones. The zones are powered by Tendermint Core [\[8\]][8], which provides a -high-performance, consistent, secure -[PBFT-like](https://blog.cosmos.network/tendermint-vs-pbft-12e9f294c9ab?gi=1777e47b6fc6) consensus engine, -where strict [fork-accountability](#fork-accountability) guarantees hold over -the behaviour of malicious actors. Tendermint Core's BFT consensus algorithm is -well suited for scaling public proof-of-stake blockchains. - -The first zone on Cosmos is called the Cosmos Hub. The Cosmos Hub is a -multi-asset proof-of-stake cryptocurrency with a simple governance mechanism -which enables the network to adapt and upgrade. In addition, the Cosmos Hub can be -extended by connecting other zones. - -The hub and zones of the Cosmos network communicate with each other via an -inter-blockchain communication (IBC) protocol, a kind of virtual UDP or TCP for -blockchains. Tokens can be transferred from one zone to another securely and -quickly without the need for exchange liquidity between zones. Instead, all -inter-zone token transfers go through the Cosmos Hub, which keeps track of the -total amount of tokens held by each zone. The hub isolates each zone from the -failure of other zones. Because anyone can connect a new zone to the Cosmos Hub, -zones allow for future-compatibility with new blockchain innovations. - -## Tendermint - -In this section we describe the Tendermint consensus protocol and the interface -used to build applications with it. For more details, see the [appendix](#appendix). - -### Validators - -In classical Byzantine fault-tolerant (BFT) algorithms, each node has the same -weight. In Tendermint, nodes have a non-negative amount of _voting power_, and -nodes that have positive voting power are called _validators_. Validators -participate in the consensus protocol by broadcasting cryptographic signatures, -or _votes_, to agree upon the next block. - -Validators' voting powers are determined at genesis, or are changed -deterministically by the blockchain, depending on the application. For example, -in a proof-of-stake application such as the Cosmos Hub, the voting power may be -determined by the amount of staking tokens bonded as collateral. - -_NOTE: Fractions like ⅔ and ⅓ refer to fractions of the total voting power, -never the total number of validators, unless all the validators have equal -weight. >⅔ means "more than ⅔", ≥⅓ means "at least ⅓"._ - -### Consensus - -Tendermint is a partially synchronous BFT consensus protocol derived from the -DLS consensus algorithm [\[20\]][20]. Tendermint is notable for its simplicity, -performance, and [fork-accountability](#fork-accountability). The protocol -requires a fixed known set of validators, where each validator is identified by -their public key. Validators attempt to come to consensus on one block at a time, -where a block is a list of transactions. Voting for consensus on a block proceeds in -rounds. Each round has a round-leader, or proposer, who proposes a block. The -validators then vote, in stages, on whether to accept the proposed block -or move on to the next round. The proposer for a round is chosen -deterministically from the ordered list of validators, in proportion to their -voting power. - -The full details of the protocol are described -[here](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - -Tendermint’s security derives from its use of optimal Byzantine fault-tolerance -via super-majority (>⅔) voting and a locking mechanism. Together, they ensure -that: - -- ≥⅓ voting power must be Byzantine to cause a violation of safety, where more - than two values are committed. -- if any set of validators ever succeeds in violating safety, or even attempts - to do so, they can be identified by the protocol. This includes both voting - for conflicting blocks and broadcasting unjustified votes. - -Despite its strong guarantees, Tendermint provides exceptional performance. In -benchmarks of 64 nodes distributed across 7 datacenters on 5 continents, on -commodity cloud instances, Tendermint consensus can process thousands of -transactions per second, with commit latencies on the order of one to two -seconds. Notably, performance of well over a thousand transactions per second -is maintained even in harsh adversarial conditions, with validators crashing or -broadcasting maliciously crafted votes. See the figure below for details. - -![Figure of Tendermint throughput performance](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/tendermint_throughput_blocksize.png) - -### Light Clients - -A major benefit of Tendermint's consensus algorithm is simplified light client -security, making it an ideal candidate for mobile and internet-of-things use -cases. While a Bitcoin light client must sync chains of block headers and find -the one with the most proof of work, Tendermint light clients need only to keep -up with changes to the validator set, and then verify the >⅔ PreCommits -in the latest block to determine the latest state. - -Succinct light client proofs also enable [inter-blockchain -communication](#inter-blockchain-communication-ibc). - -### Preventing Attacks - -Tendermint has protective measures for preventing certain notable -attacks, like [long-range-nothing-at-stake double -spends](#preventing-long-range-attacks) and -[censorship](#overcoming-forks-and-censorship-attacks). These are discussed more -fully in the [appendix](#appendix). - -### ABCI - -The Tendermint consensus algorithm is implemented in a program called Tendermint -Core. Tendermint Core is an application-agnostic "consensus engine" that can -turn any deterministic blackbox application into a distributedly replicated -blockchain. Tendermint Core connects to blockchain -applications via the Application Blockchain Interface (ABCI) [\[17\]][17]. Thus, ABCI -allows for blockchain applications to be programmed in any language, not just -the programming language that the consensus engine is written in. Additionally, -ABCI makes it possible to easily swap out the consensus layer of any existing -blockchain stack. - -We draw an analogy with the well-known cryptocurrency Bitcoin. Bitcoin is a -cryptocurrency blockchain where each node maintains a fully audited Unspent -Transaction Output (UTXO) database. If one wanted to create a Bitcoin-like -system on top of ABCI, Tendermint Core would be responsible for - -- Sharing blocks and transactions between nodes -- Establishing a canonical/immutable order of transactions (the blockchain) - -Meanwhile, the ABCI application would be responsible for - -- Maintaining the UTXO database -- Validating cryptographic signatures of transactions -- Preventing transactions from spending non-existent funds -- Allowing clients to query the UTXO database - -Tendermint is able to decompose the blockchain design by offering a very simple -API between the application process and consensus process. - -## Cosmos Overview - -Cosmos is a network of independent parallel blockchains that are each powered by -classical BFT consensus algorithms like Tendermint -[1](https://github.com/tendermint/tendermint). - -The first blockchain in this network will be the Cosmos Hub. The Cosmos Hub -connects to many other blockchains (or _zones_) via a novel inter-blockchain -communication protocol. The Cosmos Hub tracks numerous token types and keeps -record of the total number of tokens in each connected zone. Tokens can be -transferred from one zone to another securely and quickly without the need for -a liquid exchange between zones, because all inter-zone coin transfers go -through the Cosmos Hub. - -This architecture solves many problems that the blockchain space faces today, -such as application interoperability, scalability, and seamless upgradability. -For example, zones derived from Bitcoind, Go-Ethereum, CryptoNote, ZCash, or any -blockchain system can be plugged into the Cosmos Hub. These zones allow Cosmos -to scale infinitely to meet global transaction demand. Zones are also a great -fit for a distributed exchange, which will be supported as well. - -Cosmos is not just a single distributed ledger, and the Cosmos Hub isn't a -walled garden or the center of its universe. We are designing a protocol for -an open network of distributed ledgers that can serve as a new foundation for -future financial systems, based on principles of cryptography, sound economics, -consensus theory, transparency, and accountability. - -### Tendermint-BFT - -The Cosmos Hub is the first public blockchain in the Cosmos Network, powered by -Tendermint's BFT consensus algorithm. The Tendermint open-source project was -born in 2014 to address the speed, scalability, and environmental issues of -Bitcoin's proof-of-work consensus algorithm. By using and improving upon -proven BFT algorithms developed at MIT in 1988 [\[20\]][20], the Tendermint -team was the first to conceptually demonstrate a proof-of-stake cryptocurrency -that addresses the nothing-at-stake problem suffered by first-generation -proof-of-stake cryptocurrencies such as NXT and BitShares1.0. - -Today, practically all Bitcoin mobile wallets use trusted servers to provide -them with transaction verification. This is because proof-of-work requires -waiting for many confirmations before a transaction can be considered -irreversibly committed. Double-spend attacks have already been demonstrated on -services like CoinBase. - -Unlike other blockchain consensus systems, Tendermint offers instant and -provably secure mobile-client payment verification. Since the Tendermint is -designed to never fork at all, mobile wallets can receive instant transaction -confirmation, which makes trustless and practical payments a reality on -smartphones. This has significant ramifications for Internet of Things applications as well. - -Validators in Cosmos have a similar role to Bitcoin miners, but instead use -cryptographic signatures to vote. Validators are secure, dedicated machines -that are responsible for committing blocks. Non-validators can delegate their -staking tokens (called "atoms") to any validator to earn a portion of block fees -and atom rewards, but they incur the risk of getting punished (slashed) if the -delegate validator gets hacked or violates the protocol. The proven safety -guarantees of Tendermint BFT consensus, and the collateral deposit of -stakeholders--validators and delegators--provide provable, quantifiable -security for nodes and light clients. - -### Governance - -Distributed public ledgers should have a constitution and a governance system. -Bitcoin relies on the Bitcoin Foundation and mining to -coordinate upgrades, but this is a slow process. Ethereum split into ETH and -ETC after hard-forking to address TheDAO hack, largely because there was no -prior social contract nor mechanism for making such decisions. - -Validators and delegators on the Cosmos Hub can vote on proposals that can -change preset parameters of the system automatically (such as the block gas -limit), coordinate upgrades, as well as vote on amendments to the human-readable -constitution that govern the policies of the Cosmos Hub. The constitution -allows for cohesion among the stakeholders on issues such as theft -and bugs (such as TheDAO incident), allowing for quicker and cleaner resolution. - -Each zone can also have their own constitution and governance mechanism as well. -For example, the Cosmos Hub could have a constitution that enforces immutability -at the Hub (no roll-backs, save for bugs of the Cosmos Hub node implementation), -while each zone can set their own policies regarding roll-backs. - -By enabling interoperability among differing policy zones, the Cosmos network -gives its users ultimate freedom and potential for permissionless -experimentation. - -## The Hub and Zones - -Here we describe a novel model of decentralization and scalability. Cosmos is a -network of many blockchains powered by Tendermint. While existing proposals aim -to create a "single blockchain" with total global transaction ordering, Cosmos -permits many blockchains to run concurrently with one another while retaining -interoperability. - -At the basis, the Cosmos Hub manages many independent blockchains called "zones" -(sometimes referred to as "shards", in reference to the database scaling -technique known as "sharding"). A constant stream of recent block commits from -zones posted on the Hub allows the Hub to keep up with the state of each zone. -Likewise, each zone keeps up with the state of the Hub (but zones do not keep up -with each other except indirectly through the Hub). Packets of information are -then communicated from one zone to another by posting Merkle-proofs as evidence -that the information was sent and received. This mechanism is called -inter-blockchain communication, or IBC for short. - -![Figure of hub and zones -acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/images/hub_and_zones.png) - -Any of the zones can themselves be hubs to form an acyclic graph, but -for the sake of clarity we will only describe the simple configuration where -there is only one hub, and many non-hub zones. - -### The Hub - -The Cosmos Hub is a blockchain that hosts a multi-asset distributed ledger, -where tokens can be held by individual users or by zones themselves. These -tokens can be moved from one zone to another in a special IBC packet called a -"coin packet". The hub is responsible for preserving the global invariance of -the total amount of each token across the zones. IBC coin packet transactions -must be committed by the sender, hub, and receiver blockchains. - -Since the Cosmos Hub acts as the central ledger for the whole -system, the security of the Hub is of paramount importance. While each -zone may be a Tendermint blockchain that is secured by as few as 4 (or even -less if BFT consensus is not needed), the Hub must be secured by a globally -decentralized set of validators that can withstand the most severe attack -scenarios, such as a continental network partition or a nation-state sponsored -attack. - -### The Zones - -A Cosmos zone is an independent blockchain that exchanges IBC messages with the -Hub. From the Hub's perspective, a zone is a multi-asset dynamic-membership -multi-signature account that can send and receive tokens using IBC packets. Like -a cryptocurrency account, a zone cannot transfer more tokens than it has, but -can receive tokens from others who have them. A zone may be designated as an -"source" of one or more token types, granting it the power to inflate that token -supply. - -Atoms of the Cosmos Hub may be staked by validators of a zone connected to the -Hub. While double-spend attacks on these zones would result in the slashing of -atoms with Tendermint's fork-accountability, a zone where >⅔ of the voting power -are Byzantine can commit invalid state. The Cosmos Hub does not verify or -execute transactions committed on other zones, so it is the responsibility of -users to send tokens to zones that they trust. In the future, the Cosmos Hub's -governance system may pass Hub improvement proposals that account for zone -failures. For example, outbound token transfers from some (or all) zones may be -throttled to allow for the emergency circuit-breaking of zones (a temporary halt -of token transfers) when an attack is detected. - -## Inter-blockchain Communication (IBC) - -Now we look at how the Hub and zones communicate with each other. For example, if -there are three blockchains, "Zone1", "Zone2", and "Hub", and we wish for -"Zone1" to produce a packet destined for "Zone2" going through "Hub". To move a -packet from one blockchain to another, a proof is posted on the -receiving chain. The proof states that the sending chain published a packet for the alleged -destination. For the receiving chain to check this proof, it must be able keep -up with the sender's block headers. This mechanism is similar to that used by -sidechains, which requires two interacting chains to be aware of one another via a -bidirectional stream of proof-of-existence datagrams (transactions). - -The IBC protocol can naturally be defined using two types of transactions: an -`IBCBlockCommitTx` transaction, which allows a blockchain to prove to any -observer of its most recent block-hash, and an `IBCPacketTx` transaction, which -allows a blockchain to prove to any observer that the given packet was indeed -published by the sender's application, via a Merkle-proof to the recent -block-hash. - -By splitting the IBC mechanics into two separate transactions, we allow the -native fee market-mechanism of the receiving chain to determine which packets -get committed (i.e. acknowledged), while allowing for complete freedom on the -sending chain as to how many outbound packets are allowed. - -![Figure of Zone1, Zone2, and Hub IBC without -acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_without_ack.png) - -In the example above, in order to update the block-hash of -"Zone1" on "Hub" (or of "Hub" on "Zone2"), an `IBCBlockCommitTx` -transaction must be posted on "Hub" with the block-hash of "Zone1" (or on -"Zone2" with the block-hash of "Hub"). - -_See [IBCBlockCommitTx](#ibcblockcommittx) and [IBCPacketTx](#ibcpacketcommit) -for for more information on the two IBC transaction types._ - -## Use Cases - -### Distributed Exchange - -In the same way that Bitcoin is more secure by being a distributed, -mass-replicated ledger, we can make exchanges less vulnerable to external and -internal hacks by running it on the blockchain. We call this a distributed -exchange. - -What the cryptocurrency community calls a decentralized exchange today are -based on something called "atomic cross-chain" (AXC) transactions. With an AXC -transaction, two users on two different chains can make two transfer -transactions that are committed together on both ledgers, or none at all (i.e. -atomically). For example, two users can trade bitcoins for ether (or any two -tokens on two different ledgers) using AXC transactions, even though Bitcoin -and Ethereum are not connected to each other. The benefit of running an -exchange on AXC transactions is that neither users need to trust each other or -the trade-matching service. The downside is that both parties need to be -online for the trade to occur. - -Another type of decentralized exchange is a mass-replicated distributed -exchange that runs on its own blockchain. Users on this kind of exchange can -submit a limit order and turn their computer off, and the trade can execute -without the user being online. The blockchain matches and completes the trade -on behalf of the trader. - -A centralized exchange can create a deep orderbook of limit orders and thereby -attract more traders. Liquidity begets more liquidity in the exchange world, -and so there is a strong network effect (or at least a winner-take-most effect) -in the exchange business. The current leader for cryptocurrency exchanges -today is Poloniex with a 24-hour volume of $20M, and in second place is -Bitfinex with a 24-hour volume of $5M. Given such strong network effects, it -is unlikely for AXC-based decentralized exchanges to win volume over the -centralized exchanges. For a decentralized exchange to compete with a -centralized exchange, it would need to support deep orderbooks with limit -orders. Only a distributed exchange on a blockchain can provide that. - -Tendermint provides additional benefits of faster transaction commits. By -prioritizing fast finality without sacrificing consistency, zones in Cosmos can -finalize transactions fast -- for both exchange order transactions as well as -IBC token transfers to and from other zones. - -Given the state of cryptocurrency exchanges today, a great application for -Cosmos is the distributed exchange (aka the Cosmos DEX). The transaction -throughput capacity as well as commit latency can be comparable to those of -centralized exchanges. Traders can submit limit orders that can be executed -without both parties having to be online. And with Tendermint, the Cosmos hub, -and IBC, traders can move funds in and out of the exchange to and from other -zones with speed. - -### Bridging to Other Cryptocurrencies - -A privileged zone can act as the source of a bridged token of another -cryptocurrency. A bridge is similar to the relationship between a -Cosmos hub and zone; both must keep up with the latest blocks of the -other in order to verify proofs that tokens have moved from one to the other. A -"bridge-zone" on the Cosmos network keeps up with the Hub as well as the -other cryptocurrency. The indirection through the bridge-zone allows the logic of -the Hub to remain simple and agnostic to other blockchain consensus strategies -such as Bitcoin's proof-of-work mining. - -#### Sending Tokens to the Cosmos Hub - -Each bridge-zone validator would run a Tendermint-powered blockchain with a special -ABCI bridge-app, but also a full-node of the "origin" blockchain. - -When new blocks are mined on the origin, the bridge-zone validators will come -to agreement on committed blocks by signing and sharing their respective local -view of the origin's blockchain tip. When a bridge-zone receives payment on the -origin (and sufficient confirmations were agreed to have been seen in the case -of a PoW chain such as Ethereum or Bitcoin), a corresponding account is created -on the bridge-zone with that balance. - -In the case of Ethereum, the bridge-zone can share the same validator-set as the -Cosmos Hub. On the Ethereum side (the origin), a bridge-contract would allow -ether holders to send ether to the bridge-zone by sending it to the bridge-contract -on Ethereum. Once ether is received by the bridge-contract, the ether cannot be -withdrawn unless an appropriate IBC packet is received by the bridge-contract from -the bridge-zone. The bridge-contract tracks the validator-set of the bridge-zone, which -may be identical to the Cosmos Hub's validator-set. - -In the case of Bitcoin, the concept is similar except that instead of a single -bridge-contract, each UTXO would be controlled by a threshold multisignature P2SH -pubscript. Due to the limitations of the P2SH system, the signers cannot be -identical to the Cosmos Hub validator-set. - -#### Withdrawing Tokens from Cosmos Hub - -Ether on the bridge-zone ("bridged-ether") can be transferred to and from the -Hub, and later be destroyed with a transaction that sends it to a particular -withdrawal address on Ethereum. An IBC packet proving that the transaction -occurred on the bridge-zone can be posted to the Ethereum bridge-contract to -allow the ether to be withdrawn. - -In the case of Bitcoin, the restricted scripting system makes it difficult to -mirror the IBC coin-transfer mechanism. Each UTXO has its own independent -pubscript, so every UTXO must be migrated to a new UTXO when there is a change -in the set of Bitcoin escrow signers. One solution is to compress and -decompress the UTXO-set as necessary to keep the total number of UTXOs down. - -#### Total Accountability of Bridge Zones - -The risk of such a bridgeging contract is a rogue validator set. ≥⅓ Byzantine -voting power could cause a fork, withdrawing ether from the bridge-contract on -Ethereum while keeping the bridged-ether on the bridge-zone. Worse, >⅔ Byzantine -voting power can steal ether outright from those who sent it to the -bridge-contract by deviating from the original bridgeging logic of the bridge-zone. - -It is possible to address these issues by designing the bridge to be totally -accountable. For example, all IBC packets, from the hub and the origin, might -require acknowledgement by the bridge-zone in such a way that all state -transitions of the bridge-zone can be efficiently challenged and verified by -either the hub or the origin's bridge-contract. The Hub and the origin should -allow the bridge-zone validators to post collateral, and token transfers out of -the bridge-contract should be delayed (and collateral unbonding period -sufficiently long) to allow for any challenges to be made by independent -auditors. We leave the design of the specification and implementation of this -system open as a future Cosmos improvement proposal, to be passed by the Cosmos -Hub's governance system. - -### Ethereum Scaling - -Solving the scaling problem is an open issue for Ethereum. Currently, -Ethereum nodes process every single transaction and also store all the states. -[link](https://docs.google.com/presentation/d/1CjD0W4l4-CwHKUvfF5Vlps76fKLEC6pIwu1a_kC_YRQ/mobilepresent?slide=id.gd284b9333_0_28). - -Since Tendermint can commit blocks much faster than Ethereum's proof-of-work, -EVM zones powered by Tendermint consensus and operating on bridged-ether can -provide higher performance to Ethereum blockchains. Additionally, though the -Cosmos Hub and IBC packet mechanics does not allow for arbitrary contract logic -execution per se, it can be used to coordinate token movements between Ethereum -contracts running on different zones, providing a foundation for token-centric -Ethereum scaling via sharding. - -### Multi-Application Integration - -Cosmos zones run arbitrary application logic, which is defined at the beginning of the -zone's life and can potentially be updated over time by governance. Such flexibility -allows Cosmos zones to act as bridges to other cryptocurrencies such as Ethereum or -Bitcoin, and it also permits derivatives of those blockchains, utilizing the -same codebase but with a different validator set and initial distribution. This -allows many existing cryptocurrency frameworks, such as those of Ethereum, -Zerocash, Bitcoin, CryptoNote and so on, to be used with Tendermint Core, -which is a higher performance consensus engine, on a common network, opening tremendous -opportunity for interoperability across platforms. Furthermore, as a -multi-asset blockchain, a single transaction may contain multiple inputs and -outputs, where each input can be any token type, enabling Cosmos to serve -directly as a platform for decentralized exchange, though orders are assumed to -be matched via other platforms. Alternatively, a zone can serve as a distributed -fault-tolerant exchange (with orderbooks), which can be a strict improvement -over existing centralized cryptocurrency exchanges which tend to get hacked over -time. - -Zones can also serve as blockchain-backed versions of enterprise and government -systems, where pieces of a particular service that are traditionally run by an -organization or group of organizations are instead run as a ABCI application on -a certain zone, which allows it to inherit the security and interoperability of the -public Cosmos network without sacrificing control over the underlying service. -Thus, Cosmos may offer the best of both worlds for organizations looking to -utilize blockchain technology but who are wary of relinquishing control completely -to a distributed third party. - -### Network Partition Mitigation - -Some claim that a major problem with consistency-favouring consensus algorithms -like Tendermint is that any network partition which causes there to be no single -partition with >⅔ voting power (e.g. ≥⅓ going offline) will halt consensus -altogether. The Cosmos architecture can help mitigate this problem by using a global -hub with regional autonomous zones, where voting power for each zone are -distributed based on a common geographic region. For instance, a common -paradigm may be for individual cities, or regions, to operate their own zones -while sharing a common hub (e.g. the Cosmos Hub), enabling municipal activity to -persist in the event that the hub halts due to a temporary network partition. -Note that this allows real geological, political, and network-topological -features to be considered in designing robust federated fault-tolerant systems. - -### Federated Name Resolution System - -NameCoin was one of the first blockchains to attempt to solve the -name-resolution problem by adapting the Bitcoin blockchain. Unfortunately there -have been several issues with this approach. - -With Namecoin, we can verify that, for example, <em>@satoshi</em> was registered with a -particular public key at some point in the past, but we wouldn’t know whether -the public key had since been updated recently unless we download all the blocks -since the last update of that name. This is due to the limitation of Bitcoin's -UTXO transaction Merkle-ization model, where only the transactions (but not -mutable application state) are Merkle-ized into the block-hash. This lets us -prove existence, but not the non-existence of later updates to a name. Thus, we -can't know for certain the most recent value of a name without trusting a full -node, or incurring significant costs in bandwidth by downloading the whole -blockchain. - -Even if a Merkle-ized search tree were implemented in NameCoin, its dependency -on proof-of-work makes light client verification problematic. Light clients must -download a complete copy of the headers for all blocks in the entire blockchain -(or at least all the headers since the last update to a name). This means that -the bandwidth requirements scale linearly with the amount of time [\[21\]][21]. -In addition, name-changes on a proof-of-work blockchain requires waiting for -additional proof-of-work confirmation blocks, which can take up to an hour on -Bitcoin. - -With Tendermint, all we need is the most recent block-hash signed by a quorum of -validators (by voting power), and a Merkle proof to the current value associated -with the name. This makes it possible to have a succinct, quick, and secure -light-client verification of name values. - -In Cosmos, we can take this concept and extend it further. Each -name-registration zone in Cosmos can have an associated top-level-domain -(TLD) name such as ".com" or ".org", and each name-registration zone can have -its own governance and registration rules. - -## Issuance and Incentives - -### The Atom Token - -While the Cosmos Hub is a multi-asset distributed ledger, there is a special -native token called the _atom_. Atoms are the only staking token of the Cosmos -Hub. Atoms are a license for the holder to vote, validate, or delegate to other -validators. Like Ethereum's ether, atoms can also be used to pay for -transaction fees to mitigate spam. Additional inflationary atoms and block -transaction fees are rewarded to validators and delegators who delegate to -validators. - -The `BurnAtomTx` transaction can be used to recover any proportionate amount of -tokens from the reserve pool. - -#### Fundraiser - -The initial distribution of atom tokens and validators on Genesis will go to the -donors of the Cosmos Fundraiser (75%), lead donors (5%), Cosmos Network -Foundation (10%), and ALL IN BITS, Inc (10%). From genesis onward, 1/3 of the -total amount of atoms will be rewarded to bonded validators and delegators -every year. - -See the [Cosmos Plan](https://github.com/cosmos/cosmos/blob/master/PLAN.md) -for additional details. - -### Limitations on the Number of Validators - -Unlike Bitcoin or other proof-of-work blockchains, a Tendermint blockchain gets -slower with more validators due to the increased communication complexity. -Fortunately, we can support enough validators to make for a robust globally -distributed blockchain with very fast transaction confirmation times, and, as -bandwidth, storage, and parallel compute capacity increases, we will be able to -support more validators in the future. - -On genesis day, the maximum number of validators will be set to 100, and this -number will increase at a rate of 13% for 10 years, and settle at 300 -validators. - - Year 0: 100 - Year 1: 113 - Year 2: 127 - Year 3: 144 - Year 4: 163 - Year 5: 184 - Year 6: 208 - Year 7: 235 - Year 8: 265 - Year 9: 300 - Year 10: 300 - ... - -### Becoming a Validator After Genesis Day - -Atom holders who are not already can become validators by signing and -submitting a `BondTx` transaction. The amount of atoms provided as collateral -must be nonzero. Anyone can become a validator at any time, except when the -size of the current validator set is greater than the maximum number of -validators allowed. In that case, the transaction is only valid if the amount -of atoms is greater than the amount of effective atoms held by the smallest -validator, where effective atoms include delegated atoms. When a new validator -replaces an existing validator in such a way, the existing validator becomes -inactive and all the atoms and delegated atoms enter the unbonding state. - -### Penalties for Validators - -There must be some penalty imposed on the validators for any intentional -or unintentional deviation from the sanctioned protocol. Some evidence is -immediately admissible, such as a double-sign at the same height and round, or a -violation of "prevote-the-lock" (a rule of the Tendermint consensus protocol). -Such evidence will result in the validator losing its good standing and its -bonded atoms as well its proportionate share of tokens in the reserve pool -- -collectively called its "stake" -- will get slashed. - -Sometimes, validators will not be available, either due to regional network -disruptions, power failure, or other reasons. If, at any point in the past -`ValidatorTimeoutWindow` blocks, a validator's commit vote is not included in -the blockchain more than `ValidatorTimeoutMaxAbsent` times, that validator will -become inactive, and lose `ValidatorTimeoutPenalty` (DEFAULT 1%) of its stake. - -Some "malicious" behavior does not produce obviously discernable evidence on the -blockchain. In these cases, the validators can coordinate out of band to force -the timeout of these malicious validators, if there is a supermajority -consensus. - -In situations where the Cosmos Hub halts due to a ≥⅓ coalition of voting power -going offline, or in situations where a ≥⅓ coalition of voting power censor -evidence of malicious behavior from entering the blockchain, the hub must -recover with a hard-fork reorg-proposal. (Link to "Forks and Censorship -Attacks"). - -### Transaction Fees - -Cosmos Hub validators can accept any token type or combination of types as fees -for processing a transaction. Each validator can subjectively set whatever -exchange rate it wants, and choose whatever transactions it wants, as long as -the `BlockGasLimit` is not exceeded. The collected fees, minus any taxes -specified below, are redistributed to the bonded stakeholders in proportion to -their bonded atoms, every `ValidatorPayoutPeriod` (DEFAULT 1 hour). - -Of the collected transaction fees, `ReserveTax` (DEFAULT 2%) will go toward the -reserve pool to increase the reserve pool and increase the security and value of -the Cosmos network. These funds can also be distributed in accordance with the -decisions made by the governance system. - -Atom holders who delegate their voting power to other validators pay a -commission to the delegated validator. The commission can be set by each -validator. - -### Incentivizing Hackers - -The security of the Cosmos Hub is a function of the security of the underlying -validators and the choice of delegation by delegators. In order to encourage -the discovery and early reporting of found vulnerabilities, the Cosmos Hub -encourages hackers to publish successful exploits via a `ReportHackTx` -transaction that says, "This validator got hacked. Please send -bounty to this address". Upon such an exploit, the validator and delegators -will become inactive, `HackPunishmentRatio` (default 5%) of everyone's atoms -will get slashed, and `HackRewardRatio` (default 5%) of everyone's atoms will -get rewarded to the hacker's bounty address. The validator must recover the -remaining atoms by using their backup key. - -In order to prevent this feature from being abused to transfer unvested atoms, -the portion of vested vs unvested atoms of validators and delegators before and -after the `ReportHackTx` will remain the same, and the hacker bounty will -include some unvested atoms, if any. - -### Governance Specification - -The Cosmos Hub is operated by a distributed organization that requires a well-defined -governance mechanism in order to coordinate various changes to the blockchain, -such as the variable parameters of the system, as well as software upgrades and -constitutional amendments. - -All validators are responsible for voting on all proposals. Failing to vote on -a proposal in a timely manner will result in the validator being deactivated -automatically for a period of time called the `AbsenteeismPenaltyPeriod` -(DEFAULT 1 week). - -Delegators automatically inherit the vote of the delegated validator. This vote -may be overridden manually. Unbonded atoms get no vote. - -Each proposal requires a deposit of `MinimumProposalDeposit` tokens, which may -be a combination of one or more tokens including atoms. For each proposal, the -voters may vote to take the deposit. If more than half of the voters choose to -take the deposit (e.g. because the proposal was spam), the deposit goes to the -reserve pool, except any atoms which are burned. - -For each proposal, voters may vote with the following options: - -- Yea -- YeaWithForce -- Nay -- NayWithForce -- Abstain - -A strict majority of Yea or YeaWithForce votes (or Nay or NayWithForce votes) is -required for the proposal to be decided as passed (or decided as failed), but -1/3+ can veto the majority decision by voting "with force". When a strict -majority is vetoed, everyone gets punished by losing `VetoPenaltyFeeBlocks` -(DEFAULT 1 day's worth of blocks) worth of fees (except taxes which will not be -affected), and the party that vetoed the majority decision will be additionally -punished by losing `VetoPenaltyAtoms` (DEFAULT 0.1%) of its atoms. - -### Parameter Change Proposal - -Any of the parameters defined here can be changed with the passing of a -`ParameterChangeProposal`. - -### Bounty Proposal - -Atoms can be inflated and reserve pool funds spent with the passing of a `BountyProposal`. - -### Text Proposal - -All other proposals, such as a proposal to upgrade the protocol, will be -coordinated via the generic `TextProposal`. - -## Roadmap - -See [the Plan](https://github.com/cosmos/cosmos/blob/master/PLAN.md). - -## Related Work - -There have been many innovations in blockchain consensus and scalability in the -past couple of years. This section provides a brief survey of a select number -of important ones. - -### Consensus Systems - -#### Classic Byzantine Fault Tolerance - -Consensus in the presence of malicious participants is a problem dating back to -the early 1980s, when Leslie Lamport coined the phrase "Byzantine fault" to -refer to arbitrary process behavior that deviates from the intended behavior, -in contrast to a "crash fault", wherein a process simply crashes. Early -solutions were discovered for synchronous networks where there is an upper -bound on message latency, though practical use was limited to highly controlled -environments such as airplane controllers and datacenters synchronized via -atomic clocks. It was not until the late 90s that Practical Byzantine Fault -Tolerance (PBFT) [\[11\]][11] was introduced as an efficient partially -synchronous consensus algorithm able to tolerate up to ⅓ of processes behaving -arbitrarily. PBFT became the standard algorithm, spawning many variations, -including most recently one created by IBM as part of their contribution to -Hyperledger. - -The main benefit of Tendermint consensus over PBFT is that Tendermint has an -improved and simplified underlying structure, some of which is a result of -embracing the blockchain paradigm. Tendermint blocks must commit in order, -which obviates the complexity and communication overhead associated with PBFT's -view-changes. In Cosmos and many cryptocurrencies, there is no need to allow -for block <em>N+i</em> where <em>i >= 1</em> to commit, when block <em>N</em> -itself hasn't yet committed. If bandwidth is the reason why block <em>N</em> -hasn't committed in a Cosmos zone, then it doesn't help to use bandwidth sharing -votes for blocks <em>N+i</em>. If a network partition or offline nodes is the -reason why block <em>N</em> hasn't committed, then <em>N+i</em> won't commit -anyway. - -In addition, the batching of transactions into blocks allows for regular -Merkle-hashing of the application state, rather than periodic digests as with -PBFT's checkpointing scheme. This allows for faster provable transaction -commits for light-clients and faster inter-blockchain communication. - -Tendermint Core also includes many optimizations and features that go above and -beyond what is specified in PBFT. For example, the blocks proposed by -validators are split into parts, Merkle-ized, and gossipped in such a way that -improves broadcasting performance (see LibSwift [\[19\]][19] for inspiration). -Also, Tendermint Core doesn't make any assumption about point-to-point -connectivity, and functions for as long as the P2P network is weakly connected. - -#### BitShares delegated stake - -While not the first to deploy proof-of-stake (PoS), BitShares1.0 [\[12\]][12] -contributed considerably to research and adoption of PoS blockchains, -particularly those known as "delegated" PoS. In BitShares, stake holders elect -"witnesses", responsible for ordering and committing transactions, and -"delegates", responsible for coordinating software updates and parameter -changes. BitShares2.0 aims to achieve high performance (100k tx/s, 1s latency) -in ideal conditions, with each block signed by a single signer, and transaction -finality taking quite a bit longer than the block interval. A canonical -specification is still in development. Stakeholders can remove or replace -misbehaving witnesses on a daily basis, but there is no significant collateral -of witnesses or delegators in the likeness of Tendermint PoS that get slashed -in the case of a successful double-spend attack. - -#### Stellar - -Building on an approach pioneered by Ripple, Stellar [\[13\]][13] refined a -model of Federated Byzantine Agreement wherein the processes participating in -consensus do not constitute a fixed and globally known set. Rather, each -process node curates one or more "quorum slices", each constituting a set of -trusted processes. A "quorum" in Stellar is defined to be a set of nodes that -contain at least one quorum slice for each node in the set, such that agreement -can be reached. - -The security of the Stellar mechanism relies on the assumption that the -intersection of _any_ two quorums is non-empty, while the availability of a node -requires at least one of its quorum slices to consist entirely of correct nodes, -creating a trade-off between using large or small quorum-slices that may be -difficult to balance without imposing significant assumptions about trust. -Ultimately, nodes must somehow choose adequate quorum slices for there to be -sufficient fault-tolerance (or any "intact nodes" at all, of which much of the -results of the paper depend on), and the only provided strategy for ensuring -such a configuration is hierarchical and similar to the Border Gateway Protocol -(BGP), used by top-tier ISPs on the internet to establish global routing tables, -and by that used by browsers to manage TLS certificates; both notorious for -their insecurity. - -The criticism in the Stellar paper of the Tendermint-based proof-of-stake -systems is mitigated by the token strategy described here, wherein a new type of -token called the _atom_ is issued that represent claims to future portions of -fees and rewards. The advantage of Tendermint-based proof-of-stake, then, is its -relative simplicity, while still providing sufficient and provable security -guarantees. - -#### BitcoinNG - -BitcoinNG is a proposed improvement to Bitcoin that would allow for forms of -vertical scalability, such as increasing the block size, without the negative -economic consequences typically associated with such a change, such as the -disproportionately large impact on small miners. This improvement is achieved -by separating leader election from transaction broadcast: leaders are first -elected by proof-of-work in "micro-blocks", and then able to broadcast -transactions to be committed until a new micro-block is found. This reduces the -bandwidth requirements necessary to win the PoW race, allowing small miners to -more fairly compete, and allowing transactions to be committed more regularly by -the last miner to find a micro-block. - -#### Casper - -Casper [\[16\]][16] is a proposed proof-of-stake consensus algorithm for -Ethereum. Its prime mode of operation is "consensus-by-bet". By letting -validators iteratively bet on which block they believe will become committed -into the blockchain based on the other bets that they have seen so far, -finality can be achieved eventually. -[link](https://blog.ethereum.org/2015/12/28/understanding-serenity-part-2-casper/). -This is an active area of research by the Casper team. The challenge is in -constructing a betting mechanism that can be proven to be an evolutionarily -stable strategy. The main benefit of Casper as compared to Tendermint may be in -offering "availability over consistency" -- consensus does not require a >⅔ -quorum of voting power -- perhaps at the cost of commit speed or -implementation complexity. - -### Horizontal Scaling - -#### Interledger Protocol - -The Interledger Protocol [\[14\]][14] is not strictly a scalability solution. -It provides an ad hoc interoperation between different ledger systems through a -loosely coupled bilateral relationship network. Like the Lightning Network, -the purpose of ILP is to facilitate payments, but it specifically focuses on -payments across disparate ledger types, and extends the atomic transaction -mechanism to include not only hash-locks, but also a quorum of notaries (called -the Atomic Transport Protocol). The latter mechanism for enforcing atomicity -in inter-ledger transactions is similar to Tendermint's light-client SPV -mechanism, so an illustration of the distinction between ILP and Cosmos/IBC is -warranted, and provided below. - -1. The notaries of a connector in ILP do not support membership changes, and - do not allow for flexible weighting between notaries. On the other hand, - IBC is designed specifically for blockchains, where validators can have - different weights, and where membership can change over the course of the - blockchain. - -2. As in the Lightning Network, the receiver of payment in ILP must be online to - send a confirmation back to the sender. In a token transfer over IBC, the - validator-set of the receiver's blockchain is responsible for providing - confirmation, not the receiving user. - -3. The most striking difference is that ILP's connectors are not responsible or - keeping authoritative state about payments, whereas in Cosmos, the validators - of a hub are the authority of the state of IBC token transfers as well as the - authority of the amount of tokens held by each zone (but not the amount of - tokens held by each account within a zone). This is the fundamental innovation - that allows for secure asymmetric transfer of tokens from zone to zone; the - analog to ILP's connector in Cosmos is a persistent and maximally secure - blockchain ledger, the Cosmos Hub. - -4. The inter-ledger payments in ILP need to be backed by an exchange orderbook, - as there is no asymmetric transfer of coins from one ledger to another, only - the transfer of value or market equivalents. - -#### Sidechains - -Sidechains [\[15\]][15] are a proposed mechanism for scaling the Bitcoin -network via alternative blockchains that are "two-way pegged" to the Bitcoin -blockchain. (Two-way pegging is equivalent to bridging. In Cosmos we say -"bridging" to distinguish from market-pegging). Sidechains allow bitcoins to -effectively move from the Bitcoin blockchain to the sidechain and back, and -allow for experimentation in new features on the sidechain. As in the Cosmos -Hub, the sidechain and Bitcoin serve as light-clients of each other, using SPV -proofs to determine when coins should be transferred to the sidechain and back. -Of course, since Bitcoin uses proof-of-work, sidechains centered around Bitcoin -suffer from the many problems and risks of proof-of-work as a consensus -mechanism. Furthermore, this is a Bitcoin-maximalist solution that doesn't -natively support a variety of tokens and inter-zone network topology as Cosmos -does. That said, the core mechanism of the two-way peg is in principle the same -as that employed by the Cosmos network. - -#### Ethereum Scalability Efforts - -Ethereum is currently researching a number of different strategies to shard the -state of the Ethereum blockchain to address scalability needs. These efforts -have the goal of maintaining the abstraction layer offered by the current -Ethereum Virtual Machine across the shared state space. Multiple research -efforts are underway at this time. [\[18\]][18][\[22\]][22] - -##### Cosmos vs Ethereum 2.0 Mauve - -Cosmos and Ethereum 2.0 Mauve [\[22\]][22] have different design goals. - -- Cosmos is specifically about tokens. Mauve is about scaling general computation. -- Cosmos is not bound to the EVM, so even different VMs can interoperate. -- Cosmos lets the zone creator determine who validates the zone. -- Anyone can start a new zone in Cosmos (unless governance decides otherwise). -- The hub isolates zone failures so global token invariants are preserved. - -### General Scaling - -#### Lightning Network - -The Lightning Network is a proposed token transfer network operating at a layer -above the Bitcoin blockchain (and other public blockchains), enabling improvement of many -orders of magnitude in transaction throughput by moving the majority -of transactions outside of the consensus ledger into so-called "payment -channels". This is made possible by on-chain cryptocurrency scripts, which -enable parties to enter into bilateral stateful contracts where the state can -be updated by sharing digital signatures, and contracts can be closed by finally -publishing evidence onto the blockchain, a mechanism first popularized by -cross-chain atomic swaps. By opening payment channels with many parties, -participants in the Lightning Network can become focal points for routing the -payments of others, leading to a fully connected payment channel network, at the -cost of capital being tied up on payment channels. - -While the Lightning Network can also easily extend across multiple independent -blockchains to allow for the transfer of _value_ via an exchange market, it -cannot be used to asymmetrically transfer _tokens_ from one blockchain to -another. The main benefit of the Cosmos network described here is to enable -such direct token transfers. That said, we expect payment channels and the -Lightning Network to become widely adopted along with our token transfer -mechanism, for cost-saving and privacy reasons. - -#### Segregated Witness - -Segregated Witness is a Bitcoin improvement proposal -[link](https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki) that aims -to increase the per-block transaction throughput 2X or 3X, while simultaneously -making block syncing faster for new nodes. The brilliance of this solution is -in how it works within the limitations of Bitcoin's current protocol and allows -for a soft-fork upgrade (i.e. clients with older versions of the software will -continue to function after the upgrade). Tendermint, being a new protocol, has no -design restrictions, so it has a different scaling priorities. Primarily, -Tendermint uses a BFT round-robin algorithm based on cryptographic signatures -instead of mining, which trivially allows horizontal scaling through multiple -parallel blockchains, while regular, more frequent block commits allow for -vertical scaling as well. - -<hr/> - -## Appendix - -### Fork Accountability - -A well designed consensus protocol should provide some guarantees in the event that the tolerance -capacity is exceeded and the consensus fails. This is especially necessary in -economic systems, where Byzantine behaviour can have substantial financial -reward. The most important such guarantee is a form of _fork-accountability_, -where the processes that caused the consensus to fail (ie. caused clients of -the protocol to accept different values - a fork) can be identified and punished -according to the rules of the protocol, or, possibly, the legal system. When -the legal system is unreliable or excessively expensive to invoke, validators can be forced to make security -deposits in order to participate, and those deposits can be jailed, or slashed, -when malicious behaviour is detected [\[10\]][10]. - -Note this is unlike Bitcoin, where forking is a regular occurence due to -network asynchrony and the probabilistic nature of finding partial hash -collisions. Since in many cases a malicious fork is indistinguishable from a -fork due to asynchrony, Bitcoin cannot reliably implement fork-accountability, -other than the implicit opportunity cost paid by miners for mining an orphaned -block. - -### Tendermint Consensus - -We call the voting stages _PreVote_ and _PreCommit_. A vote can be for a -particular block or for _Nil_. We call a collection of >⅔ PreVotes for a single -block in the same round a _Polka_, and a collection of >⅔ PreCommits for a -single block in the same round a _Commit_. If >⅔ PreCommit for Nil in the same -round, they move to the next round. - -Note that strict determinism in the protocol incurs a weak synchrony assumption -as faulty leaders must be detected and skipped. Thus, validators wait some -amount of time, _TimeoutPropose_, before they Prevote Nil, and the value of -TimeoutPropose increases with each round. Progression through the rest of a -round is fully asynchronous, in that progress is only made once a validator hears -from >⅔ of the network. In practice, it would take an extremely strong -adversary to indefinitely thwart the weak synchrony assumption (causing the -consensus to fail to ever commit a block), and doing so can be made even more -difficult by using randomized values of TimeoutPropose on each validator. - -An additional set of constraints, or Locking Rules, ensure that the network will -eventually commit just one block at each height. Any malicious attempt to cause -more than one block to be committed at a given height can be identified. First, -a PreCommit for a block must come with justification, in the form of a Polka for -that block. If the validator has already PreCommit a block at round -<em>R*1</em>, we say they are \_locked* on that block, and the Polka used to -justify the new PreCommit at round <em>R_2</em> must come in a round -<em>R_polka</em> where <em>R_1 < R_polka <= R_2</em>. Second, validators -must Propose and/or PreVote the block they are locked on. Together, these -conditions ensure that a validator does not PreCommit without sufficient -evidence as justification, and that validators which have already PreCommit -cannot contribute to evidence to PreCommit something else. This ensures both -safety and liveness of the consensus algorithm. - -The full details of the protocol are described -[here](https://github.com/tendermint/tendermint/wiki/Byzantine-Consensus-Algorithm). - -### Tendermint Light Clients - -The need to sync all block headers is eliminated in Tendermint-PoS as the -existence of an alternative chain (a fork) means ≥⅓ of bonded stake can be -slashed. Of course, since slashing requires that _someone_ share evidence of a -fork, light clients should store any block-hash commits that it sees. -Additionally, light clients could periodically stay synced with changes to the -validator set, in order to avoid [long range -attacks](#preventing-long-range-attacks) (but other solutions are possible). - -In spirit similar to Ethereum, Tendermint enables applications to embed a -global Merkle root hash in each block, allowing easily verifiable state queries -for things like account balances, the value stored in a contract, or the -existence of an unspent transaction output, depending on the nature of the -application. - -### Preventing Long Range Attacks - -Assuming a sufficiently resilient collection of broadcast networks and a static -validator set, any fork in the blockchain can be detected and the deposits of -the offending validators slashed. This innovation, first suggested by Vitalik -Buterin in early 2014, solves the nothing-at-stake problem of other -proof-of-stake cryptocurrencies (see [Related Work](#related-work)). However, -since validator sets must be able to change, over a long range of time the -original validators may all become unbonded, and hence would be free to create a -new chain from the genesis block, incurring no cost as they no longer have -deposits locked up. This attack came to be known as the Long Range Attack (LRA), -in contrast to a Short Range Attack, where validators who are currently bonded -cause a fork and are hence punishable (assuming a fork-accountable BFT algorithm -like Tendermint consensus). Long Range Attacks are often thought to be a -critical blow to proof-of-stake. - -Fortunately, the LRA can be mitigated as follows. First, for a validator to -unbond (thereby recovering their collateral deposit and no longer earning fees -to participate in the consensus), the deposit must be made untransferable for an -amount of time known as the "unbonding period", which may be on the order of -weeks or months. Second, for a light client to be secure, the first time it -connects to the network it must verify a recent block-hash against a trusted -source, or preferably multiple sources. This condition is sometimes referred to -as "weak subjectivity". Finally, to remain secure, it must sync up with the -latest validator set at least as frequently as the length of the unbonding -period. This ensures that the light client knows about changes to the validator -set before a validator has its capital unbonded and thus no longer at stake, -which would allow it to deceive the client by carrying out a long range attack -by creating new blocks beginning back at a height where it was bonded (assuming -it has control of sufficiently many of the early private keys). - -Note that overcoming the LRA in this way requires an overhaul of the original -security model of proof-of-work. In PoW, it is assumed that a light client can -sync to the current height from the trusted genesis block at any time simply by -processing the proof-of-work in every block header. To overcome the LRA, -however, we require that a light client come online with some regularity to -track changes in the validator set, and that the first time they come online -they must be particularly careful to authenticate what they hear from the -network against trusted sources. Of course, this latter requirement is similar -to that of Bitcoin, where the protocol and software must also be obtained from a -trusted source. - -The above method for preventing LRA is well suited for validators and full nodes -of a Tendermint-powered blockchain because these nodes are meant to remain -connected to the network. The method is also suitable for light clients that -can be expected to sync with the network frequently. However, for light clients -that are not expected to have frequent access to the internet or the blockchain -network, yet another solution can be used to overcome the LRA. Non-validator -token holders can post their tokens as collateral with a very long unbonding -period (e.g. much longer than the unbonding period for validators) and serve -light clients with a secondary method of attesting to the validity of current -and past block-hashes. While these tokens do not count toward the security of -the blockchain's consensus, they nevertheless can provide strong guarantees for -light clients. If historical block-hash querying were supported in Ethereum, -anyone could bond their tokens in a specially designed smart contract and -provide attestation services for pay, effectively creating a market for -light-client LRA security. - -### Overcoming Forks and Censorship Attacks - -Due to the definition of a block commit, any ≥⅓ coalition of voting power can -halt the blockchain by going offline or not broadcasting their votes. Such a -coalition can also censor particular transactions by rejecting blocks that -include these transactions, though this would result in a significant proportion -of block proposals to be rejected, which would slow down the rate of block -commits of the blockchain, reducing its utility and value. The malicious -coalition might also broadcast votes in a trickle so as to grind blockchain -block commits to a near halt, or engage in any combination of these attacks. -Finally, it can cause the blockchain to fork, by double-signing or violating the -locking rules. - -If a globally active adversary were also involved, it could partition the network in -such a way that it may appear that the wrong subset of validators were -responsible for the slowdown. This is not just a limitation of Tendermint, but -rather a limitation of all consensus protocols whose network is potentially -controlled by an active adversary. - -For these types of attacks, a subset of the validators should coordinate through -external means to sign a reorg-proposal that chooses a fork (and any evidence -thereof) and the initial subset of validators with their signatures. Validators -who sign such a reorg-proposal forego their collateral on all other forks. -Clients should verify the signatures on the reorg-proposal, verify any evidence, -and make a judgement or prompt the end-user for a decision. For example, a -phone wallet app may prompt the user with a security warning, while a -refrigerator may accept any reorg-proposal signed by +½ of the original -validators by voting power. - -No non-synchronous Byzantine fault-tolerant algorithm can come to consensus when -≥⅓ of voting power are dishonest, yet a fork assumes that ≥⅓ of voting power -have already been dishonest by double-signing or lock-changing without -justification. So, signing the reorg-proposal is a coordination problem that -cannot be solved by any non-synchronous protocol (i.e. automatically, and -without making assumptions about the reliability of the underlying network). -For now, we leave the problem of reorg-proposal coordination to human -coordination via social consensus on internet media. Validators must take care -to ensure that there are no remaining network partitions prior to signing a -reorg-proposal, to avoid situations where two conflicting reorg-proposals are -signed. - -Assuming that the external coordination medium and protocol is robust, it -follows that forks are less of a concern than censorship attacks. - -In addition to forks and censorship, which require ≥⅓ Byzantine voting power, a -coalition of >⅔ voting power may commit arbitrary, invalid state. This is -characteristic of any (BFT) consensus system. Unlike double-signing, which -creates forks with easily verifiable evidence, detecting committment of an -invalid state requires non-validating peers to verify whole blocks, which -implies that they keep a local copy of the state and execute each transaction, -computing the state root independently for themselves. Once detected, the only -way to handle such a failure is via social consensus. For instance, in -situations where Bitcoin has failed, whether forking due to software bugs (as in -March 2013), or committing invalid state due to Byzantine behavior of miners (as -in July 2015), the well connected community of businesses, developers, miners, -and other organizations established a social consensus as to what manual actions -were required by participants to heal the network. Furthermore, since -validators of a Tendermint blockchain may be expected to be identifiable, -commitment of an invalid state may even be punishable by law or some external -jurisprudence, if desired. - -### ABCI Specification - -ABCI consists of 3 primary message types that get delivered from the core to the -application. The application replies with corresponding response messages. - -The `AppendTx` message is the work horse of the application. Each transaction in -the blockchain is delivered with this message. The application needs to validate -each transactions received with the AppendTx message against the current state, -application protocol, and the cryptographic credentials of the transaction. A -validated transaction then needs to update the application state — by binding a -value into a key values store, or by updating the UTXO database. - -The `CheckTx` message is similar to AppendTx, but it’s only for validating -transactions. Tendermint Core’s mempool first checks the validity of a -transaction with CheckTx, and only relays valid transactions to its peers. -Applications may check an incrementing nonce in the transaction and return an -error upon CheckTx if the nonce is old. - -The `Commit` message is used to compute a cryptographic commitment to the -current application state, to be placed into the next block header. This has -some handy properties. Inconsistencies in updating that state will now appear as -blockchain forks which catches a whole class of programming errors. This also -simplifies the development of secure lightweight clients, as Merkle-hash proofs -can be verified by checking against the block-hash, and the block-hash is signed -by a quorum of validators (by voting power). - -Additional ABCI messages allow the application to keep track of and change the -validator set, and for the application to receive the block information, such as -the height and the commit votes. - -ABCI requests/responses are simple Protobuf messages. Check out the [schema -file](https://github.com/tendermint/abci/blob/master/types/types.proto). - -##### AppendTx - -- **Arguments**: - - `Data ([]byte)`: The request transaction bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: Result bytes, if any - - `Log (string)`: Debug or error message -- **Usage**:<br/> - Append and run a transaction. If the transaction is valid, returns - CodeType.OK - -##### CheckTx - -- **Arguments**: - - `Data ([]byte)`: The request transaction bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: Result bytes, if any - - `Log (string)`: Debug or error message -- **Usage**:<br/> - Validate a transaction. This message should not mutate the state. - Transactions are first run through CheckTx before broadcast to peers in the - mempool layer. - You can make CheckTx semi-stateful and clear the state upon `Commit` or - `BeginBlock`, - to allow for dependent sequences of transactions in the same block. - -##### Commit - -- **Returns**: - - `Data ([]byte)`: The Merkle root hash - - `Log (string)`: Debug or error message -- **Usage**:<br/> - Return a Merkle root hash of the application state. - -##### Query - -- **Arguments**: - - `Data ([]byte)`: The query request bytes -- **Returns**: - - `Code (uint32)`: Response code - - `Data ([]byte)`: The query response bytes - - `Log (string)`: Debug or error message - -##### Flush - -- **Usage**:<br/> - Flush the response queue. Applications that implement `types.Application` - need not implement this message -- it's handled by the project. - -##### Info - -- **Returns**: - - `Data ([]byte)`: The info bytes -- **Usage**:<br/> - Return information about the application state. Application specific. - -##### SetOption - -- **Arguments**: - - `Key (string)`: Key to set - - `Value (string)`: Value to set for key -- **Returns**: - - `Log (string)`: Debug or error message -- **Usage**:<br/> - Set application options. E.g. Key="mode", Value="mempool" for a mempool - connection, or Key="mode", Value="consensus" for a consensus connection. - Other options are application specific. - -##### InitChain - -- **Arguments**: - - `Validators ([]Validator)`: Initial genesis-validators -- **Usage**:<br/> - Called once upon genesis - -##### BeginBlock - -- **Arguments**: - - `Height (uint64)`: The block height that is starting -- **Usage**:<br/> - Signals the beginning of a new block. Called prior to any AppendTxs. - -##### EndBlock - -- **Arguments**: - - `Height (uint64)`: The block height that ended -- **Returns**: - - `Validators ([]Validator)`: Changed validators with new voting powers (0 - to remove) -- **Usage**:<br/> - Signals the end of a block. Called prior to each Commit after all - transactions - -See [the ABCI repository](https://github.com/tendermint/abci#message-types) for more details. - -### IBC Packet Delivery Acknowledgement - -There are several reasons why a sender may want the acknowledgement of delivery -of a packet by the receiving chain. For example, the sender may not know the -status of the destination chain, if it is expected to be faulty. Or, the sender -may want to impose a timeout on the packet (with the `MaxHeight` packet field), -while any destination chain may suffer from a denial-of-service attack with a -sudden spike in the number of incoming packets. - -In these cases, the sender can require delivery acknowledgement by setting the -initial packet status to `AckPending`. Then, it is the receiving chain's -responsibility to confirm delivery by including an abbreviated `IBCPacket` in the -app Merkle hash. - -![Figure of Zone1, Zone2, and Hub IBC with -acknowledgement](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack.png) - -First, an `IBCBlockCommit` and `IBCPacketTx` are posted on "Hub" that proves -the existence of an `IBCPacket` on "Zone1". Say that `IBCPacketTx` has the -following value: - -- `FromChainID`: "Zone1" -- `FromBlockHeight`: 100 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 (say) - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 (say "Hub" is currently at height 300) - - `Payload`: <The bytes of a "coin" payload> - -Next, an `IBCBlockCommit` and `IBCPacketTx` are posted on "Zone2" that proves -the existence of an `IBCPacket` on "Hub". Say that `IBCPacketTx` has the -following value: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 300 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckPending` - - `Type`: "coin" - - `MaxHeight`: 350 - - `Payload`: <The same bytes of a "coin" payload> - -Next, "Zone2" must include in its app-hash an abbreviated packet that shows the -new status of `AckSent`. An `IBCBlockCommit` and `IBCPacketTx` are posted back -on "Hub" that proves the existence of an abbreviated `IBCPacket` on -"Zone2". Say that `IBCPacketTx` has the following value: - -- `FromChainID`: "Zone2" -- `FromBlockHeight`: 400 (say) -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckSent` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <The hash bytes of the same "coin" payload> - -Finally, "Hub" must update the status of the packet from `AckPending` to -`AckReceived`. Evidence of this new finalized status should go back to -"Zone2". Say that `IBCPacketTx` has the following value: - -- `FromChainID`: "Hub" -- `FromBlockHeight`: 301 -- `Packet`: an `IBCPacket`: - - `Header`: an `IBCPacketHeader`: - - `SrcChainID`: "Zone1" - - `DstChainID`: "Zone2" - - `Number`: 200 - - `Status`: `AckReceived` - - `Type`: "coin" - - `MaxHeight`: 350 - - `PayloadHash`: <The hash bytes of the same "coin" payload> - -Meanwhile, "Zone1" may optimistically assume successful delivery of a "coin" -packet unless evidence to the contrary is proven on "Hub". In the example -above, if "Hub" had not received an `AckSent` status from "Zone2" by block -350, it would have set the status automatically to `Timeout`. This evidence of -a timeout can get posted back on "Zone1", and any tokens can be returned. - -![Figure of Zone1, Zone2, and Hub IBC with acknowledgement and -timeout](https://raw.githubusercontent.com/gnuclear/atom-whitepaper/master/msc/ibc_with_ack_timeout.png) - -### Merkle Tree & Proof Specification - -There are two types of Merkle trees supported in the Tendermint/Cosmos -ecosystem: The Simple Tree, and the IAVL+ Tree. - -#### Simple Tree - -The Simple Tree is a Merkle tree for a static list of elements. If the number -of items is not a power of two, some leaves will be at different levels. Simple -Tree tries to keep both sides of the tree the same height, but the left may be -one greater. This Merkle tree is used to Merkle-ize the transactions of a -block, and the top level elements of the application state root. - - * - / \ - / \ - / \ - / \ - * * - / \ / \ - / \ / \ - / \ / \ - * * * h6 - / \ / \ / \ - h0 h1 h2 h3 h4 h5 - - A SimpleTree with 7 elements - -#### IAVL+ Tree - -The purpose of the IAVL+ data structure is to provide persistent storage for -key-value pairs in the application state such that a deterministic Merkle root -hash can be computed efficiently. The tree is balanced using a variant of the -[AVL algorithm](https://en.wikipedia.org/wiki/AVL_tree), and all operations are -O(log(n)). - -In an AVL tree, the heights of the two child subtrees of any node differ by at -most one. Whenever this condition is violated upon an update, the tree is -rebalanced by creating O(log(n)) new nodes that point to unmodified nodes of the -old tree. In the original AVL algorithm, inner nodes can also hold key-value -pairs. The AVL+ algorithm (note the plus) modifies the AVL algorithm to keep -all values on leaf nodes, while only using branch-nodes to store keys. This -simplifies the algorithm while keeping the merkle hash trail short. - -The AVL+ Tree is analogous to Ethereum's [Patricia -tries](https://en.wikipedia.org/wiki/Radix_tree). There are tradeoffs. Keys do -not need to be hashed prior to insertion in IAVL+ trees, so this provides faster -ordered iteration in the key space which may benefit some applications. The -logic is simpler to implement, requiring only two types of nodes -- inner nodes -and leaf nodes. The Merkle proof is on average shorter, being a balanced binary -tree. On the other hand, the Merkle root of an IAVL+ tree depends on the order -of updates. - -We will support additional efficient Merkle trees, such as Ethereum's Patricia -Trie when the binary variant becomes available. - -### Transaction Types - -In the canonical implementation, transactions are streamed to the Cosmos hub -application via the ABCI interface. - -The Cosmos Hub will accept a number of primary transaction types, including -`SendTx`, `BondTx`, `UnbondTx`, `ReportHackTx`, `SlashTx`, `BurnAtomTx`, -`ProposalCreateTx`, and `ProposalVoteTx`, which are fairly self-explanatory and -will be documented in a future revision of this paper. Here we document the two -primary transaction types for IBC: `IBCBlockCommitTx` and `IBCPacketTx`. - -#### IBCBlockCommitTx - -An `IBCBlockCommitTx` transaction is composed of: - -- `ChainID (string)`: The ID of the blockchain -- `BlockHash ([]byte)`: The block-hash bytes, the Merkle root which includes the - app-hash -- `BlockPartsHeader (PartSetHeader)`: The block part-set header bytes, only - needed to verify vote signatures -- `BlockHeight (int)`: The height of the commit -- `BlockRound (int)`: The round of the commit -- `Commit ([]Vote)`: The >⅔ Tendermint `Precommit` votes that comprise a block - commit -- `ValidatorsHash ([]byte)`: A Merkle-tree root hash of the new validator set -- `ValidatorsHashProof (SimpleProof)`: A SimpleTree Merkle-proof for proving the - `ValidatorsHash` against the `BlockHash` -- `AppHash ([]byte)`: A IAVLTree Merkle-tree root hash of the application state -- `AppHashProof (SimpleProof)`: A SimpleTree Merkle-proof for proving the - `AppHash` against the `BlockHash` - -#### IBCPacketTx - -An `IBCPacket` is composed of: - -- `Header (IBCPacketHeader)`: The packet header -- `Payload ([]byte)`: The bytes of the packet payload. _Optional_ -- `PayloadHash ([]byte)`: The hash for the bytes of the packet. _Optional_ - -Either one of `Payload` or `PayloadHash` must be present. The hash of an -`IBCPacket` is a simple Merkle root of the two items, `Header` and `Payload`. -An `IBCPacket` without the full payload is called an _abbreviated packet_. - -An `IBCPacketHeader` is composed of: - -- `SrcChainID (string)`: The source blockchain ID -- `DstChainID (string)`: The destination blockchain ID -- `Number (int)`: A unique number for all packets -- `Status (enum)`: Can be one of `AckPending`, `AckSent`, `AckReceived`, - `NoAck`, or `Timeout` -- `Type (string)`: The types are application-dependent. Cosmos reserves the - "coin" packet type -- `MaxHeight (int)`: If status is not `NoAckWanted` or `AckReceived` by this - height, status becomes `Timeout`. _Optional_ - -An `IBCPacketTx` transaction is composed of: - -- `FromChainID (string)`: The ID of the blockchain which is providing this - packet; not necessarily the source -- `FromBlockHeight (int)`: The blockchain height in which the following packet - is included (Merkle-ized) in the block-hash of the source chain -- `Packet (IBCPacket)`: A packet of data, whose status may be one of - `AckPending`, `AckSent`, `AckReceived`, `NoAck`, or `Timeout` -- `PacketProof (IAVLProof)`: A IAVLTree Merkle-proof for proving the packet's - hash against the `AppHash` of the source chain at given height - -The sequence for sending a packet from "Zone1" to "Zone2" through the -"Hub" is depicted in {Figure X}. First, an `IBCPacketTx` proves to -"Hub" that the packet is included in the app-state of "Zone1". Then, -another `IBCPacketTx` proves to "Zone2" that the packet is included in the -app-state of "Hub". During this procedure, the `IBCPacket` fields are -identical: the `SrcChainID` is always "Zone1", and the `DstChainID` is always -"Zone2". - -The `PacketProof` must have the correct Merkle-proof path, as follows: - - IBC/<SrcChainID>/<DstChainID>/<Number> - -When "Zone1" wants to send a packet to "Zone2" through "Hub", the -`IBCPacket` data are identical whether the packet is Merkle-ized on "Zone1", -the "Hub", or "Zone2". The only mutable field is `Status` for tracking -delivery. - -## Acknowledgements - -We thank our friends and peers for assistance in conceptualizing, reviewing, and -providing support for our work with Tendermint and Cosmos. - -- [Zaki Manian](https://github.com/zmanian) of - [SkuChain](http://www.skuchain.com/) provided much help in formatting and - wording, especially under the ABCI section -- [Jehan Tremback](https://github.com/jtremback) of Althea and Dustin Byington - for helping with initial iterations -- [Andrew Miller](https://soc1024.com/) of [Honey - Badger](https://eprint.iacr.org/2016/199) for feedback on consensus -- [Greg Slepak](https://fixingtao.com/) for feedback on consensus and wording -- Also thanks to [Bill Gleim](https://github.com/gleim) and [Seunghwan - Han](http://www.seunghwanhan.com) for various contributions. -- **Your name and organization here for your contribution** - -## Citations - -- [1] Bitcoin: <https://bitcoin.org/bitcoin.pdf> -- [2] ZeroCash: <http://zerocash-project.org/paper> -- [3] Ethereum: <https://github.com/ethereum/wiki/wiki/White-Paper> -- [4] TheDAO: <https://download.slock.it/public/DAO/WhitePaper.pdf> -- [5] Segregated Witness: <https://github.com/bitcoin/bips/blob/master/bip-0141.mediawiki> -- [6] BitcoinNG: <https://arxiv.org/pdf/1510.02037v2.pdf> -- [7] Lightning Network: <https://lightning.network/lightning-network-paper-DRAFT-0.5.pdf> -- [8] Tendermint: <https://github.com/tendermint/tendermint/wiki> -- [9] FLP Impossibility: <https://groups.csail.mit.edu/tds/papers/Lynch/jacm85.pdf> -- [10] Slasher: <https://blog.ethereum.org/2014/01/15/slasher-a-punitive-proof-of-stake-algorithm/> -- [11] PBFT: <http://pmg.csail.mit.edu/papers/osdi99.pdf> -- [12] BitShares: <https://bitshares.org/technology/delegated-proof-of-stake-consensus/> -- [13] Stellar: <https://www.stellar.org/papers/stellar-consensus-protocol.pdf> -- [14] Interledger: <https://interledger.org/rfcs/0001-interledger-architecture/> -- [15] Sidechains: <https://blockstream.com/sidechains.pdf> -- [16] Casper: <https://blog.ethereum.org/2015/08/01/introducing-casper-friendly-ghost/> -- [17] ABCI: <https://github.com/tendermint/abci> -- [18] Ethereum Sharding: <https://github.com/ethereum/EIPs/issues/53> -- [19] LibSwift: <http://www.ds.ewi.tudelft.nl/fileadmin/pds/papers/PerformanceAnalysisOfLibswift.pdf> -- [20] DLS: <http://groups.csail.mit.edu/tds/papers/Lynch/jacm88.pdf> -- [21] Thin Client Security: <https://en.bitcoin.it/wiki/Thin_Client_Security> -- [22] Ethereum 2.0 Mauve Paper: <https://cdn.hackaday.io/files/10879465447136/Mauve%20Paper%20Vitalik.pdf> - -#### Unsorted links - -- <https://www.docdroid.net/ec7xGzs/314477721-ethereum-platform-review-opportunities-and-challenges-for-private-and-consortium-blockchains.pdf> diff --git a/docs/sdk/cosmos-sdk-cli.md b/docs/sdk/cosmos-sdk-cli.md deleted file mode 100644 index 615d72252..000000000 --- a/docs/sdk/cosmos-sdk-cli.md +++ /dev/null @@ -1,34 +0,0 @@ -# cosmos-sdk-cli -Create a new blockchain project based on cosmos-sdk with a single command. - ---- - -# Installation - -```shell -$ go get github.com/cosmos/cosmos-sdk -$ cd $GOPATH/src/github.com/cosmos/cosmos-sdk -$ make install_cosmos-sdk-cli -``` - -This will install a binary cosmos-sdk-cli - -# Creating a new project - -**$cosmos-sdk-cli init** _Your-Project-Name_ - -This will initialize a project, the dependencies, directory structures with the specified project name. - -### Example: -```shell -$ cosmos-sdk-cli init testzone -p github.com/your_user_name/testzone -``` -`-p [remote-project-path]`. If this is not provided, it creates testzone under $GOPATH/src/ - - -```shell -$ cd $GOPATH/src/github.com/your_user_name/testzone -$ make -``` -This will create two binaries(testzonecli and testzoned) under bin folder. testzoned is the full node of the application which you can run, and testzonecli is your light client. - diff --git a/docs/sdk/modules.md b/docs/sdk/modules.md deleted file mode 100644 index 7df8fb2bb..000000000 --- a/docs/sdk/modules.md +++ /dev/null @@ -1,19 +0,0 @@ -# Modules - -::: tip Note -🚧 We are actively working on documentation for SDK modules. -::: - -The Cosmos SDK has all the necessary pre-built modules to add functionality on top of a `BaseApp`, which is the template to build a blockchain dApp in Cosmos. In this context, a `module` is a fundamental unit in the Cosmos SDK. - -Each module is an extension of the `BaseApp`'s functionalities that defines transactions, handles application state and manages the state transition logic. Each module also contains handlers for messages and transactions, queriers for handling query requests, as well as REST and CLI for secure user interactions. - -Some of the most important modules in the SDK are: - -- `Auth`: Defines a standard account structure (`BaseAccount`) and how transaction signers are authenticated. -- `Bank`: Defines how coins or tokens (i.e cryptocurrencies) are transferred. -- `Fees`: -- `Governance`: Governance related implementation including proposals and voting. -- `IBC`: Defines the intereoperability of blockchain zones according to the specifications of the [IBC Protocol](https://cosmos.network/whitepaper#inter-blockchain-communication-ibc). -- `Slashing`: -- `Staking`: Proof of Stake related implementation including bonding and delegation transactions, inflation, fees, unbonding, etc. diff --git a/docs/sdk/overview.md b/docs/sdk/overview.md deleted file mode 100644 index 70e826c60..000000000 --- a/docs/sdk/overview.md +++ /dev/null @@ -1,205 +0,0 @@ -# Cosmos SDK Overview - -The [Cosmos-SDK](https://github.com/cosmos/cosmos-sdk) is a framework for building multi-asset Proof-of-Stake (PoS) blockchains, like the Cosmos Hub, as well as Proof-Of-Authority (PoA) blockchains. - -The goal of the Cosmos-SDK is to allow developers to easily create custom interoperable blockchain applications within the Cosmos Network without having to recreate common blockchain functionality, thus removing the complexity of building a Tendermint ABCI application. We envision the SDK as the npm-like framework to build secure blockchain applications on top of Tendermint. - -In terms of its design, the SDK optimizes flexibility and security. The framework is designed around a modular execution stack which allows applications to mix and match elements as desired. In addition, all modules are sandboxed for greater application security. - -It is based on two major principles: - -- **Composability:** Anyone can create a module for the Cosmos-SDK and integrating the already-built modules is as simple as importing them into your blockchain application. - -- **Capabilities:** The SDK is inspired by capabilities-based security, and informed by years of wrestling with blockchain state-machines. Most developers will need to access other 3rd party modules when building their own modules. Given that the Cosmos-SDK is an open framework and that we assume that some of those modules may be malicious, we designed the SDK using object-capabilities (ocaps) based principles. In practice, this means that instead of having each module keep an access control list for other modules, each module implements special objects called keepers that can be passed to other modules to grant a pre-defined set of capabilities. For example, if an instance of module A's keepers is passed to module B, the latter will be able to call a restricted set of module A's functions. The capabilities of each keeper are defined by the module's developer, and it's the developer's job to understand and audit the safety of foreign code from 3rd party modules based on the capabilities they are passing into each 3rd party module. For a deeper look at capabilities, you can read this [article](http://habitatchronicles.com/2017/05/what-are-capabilities/). - -For an introduction to object-capabilities, see this [article](http://habitatchronicles.com/2017/05/what-are-capabilities/). - -## Languages - -The Cosmos-SDK is currently written in [Golang](https://golang.org/), though the -framework could be implemented similarly in other languages. -Contact us for information about funding an implementation in another language. - -## Directory Structure - -The SDK is laid out in the following directories: - -- `baseapp`: Defines the template for a basic [ABCI](https://cosmos.network/whitepaper#abci) application so that your Cosmos-SDK application can communicate with the underlying Tendermint node. -- `client`: CLI and REST server tooling for interacting with SDK application. -- `examples`: Examples of how to build working standalone applications. -- `server`: The full node server for running an SDK application on top of - Tendermint. -- `store`: The database of the SDK - a Merkle multistore supporting multiple types of underling Merkle key-value stores. -- `types`: Common types in SDK applications. -- `x`: Extensions to the core, where all messages and handlers are defined. - -## Object-Capability Model - -When thinking about security, it's good to start with a specific threat -model. Our threat model is the following: - -> We assume that a thriving ecosystem of Cosmos-SDK modules that are easy to compose into a blockchain application will contain faulty or malicious modules. - -The Cosmos-SDK is designed to address this threat by being the -foundation of an object capability system. - -> The structural properties of object capability systems favor -> modularity in code design and ensure reliable encapsulation in -> code implementation. -> -> These structural properties facilitate the analysis of some -> security properties of an object-capability program or operating -> system. Some of these — in particular, information flow properties -> — can be analyzed at the level of object references and -> connectivity, independent of any knowledge or analysis of the code -> that determines the behavior of the objects. -> -> As a consequence, these security properties can be established -> and maintained in the presence of new objects that contain unknown -> and possibly malicious code. -> -> These structural properties stem from the two rules governing -> access to existing objects: -> -> 1. An object A can send a message to B only if object A holds a -> reference to B. -> 2. An object A can obtain a reference to C only -> if object A receives a message containing a reference to C. As a -> consequence of these two rules, an object can obtain a reference -> to another object only through a preexisting chain of references. -> In short, "Only connectivity begets connectivity." - -See the [wikipedia -article](https://en.wikipedia.org/wiki/Object-capability_model) on the Object Capability Model for more -information. - -Strictly speaking, Golang does not implement object capabilities -completely, because of several issues: - -- pervasive ability to import primitive modules (e.g. "unsafe", "os") -- pervasive ability to override module vars <https://github.com/golang/go/issues/23161> -- data-race vulnerability where 2+ goroutines can create illegal interface values - -The first is easy to catch by auditing imports and using a proper -dependency version control system like Dep. The second and third are -unfortunate but it can be audited with some cost. - -Perhaps [Go2 will implement the object capability -model](https://github.com/golang/go/issues/23157). - -### What does it look like? - -Only reveal what is necessary to get the work done. - -For example, the following code snippet violates the object capabilities -principle: - -```go -type AppAccount struct {...} -var account := &AppAccount{ - Address: pub.Address(), - Coins: sdk.Coins{{"ATM", 100}}, -} -var sumValue := externalModule.ComputeSumValue(account) -``` - -The method `ComputeSumValue` implies a pure function, yet the implied -capability of accepting a pointer value is the capability to modify that -value. The preferred method signature should take a copy instead. - -```go -var sumValue := externalModule.ComputeSumValue(*account) -``` - -In the Cosmos SDK, you can see the application of this principle in the -basecoin examples folder. - -```go -// File: cosmos-sdk/examples/basecoin/app/init_handlers.go -package app - -import ( - "github.com/cosmos/cosmos-sdk/x/bank" - "github.com/cosmos/cosmos-sdk/x/sketchy" -) - -func (app *BasecoinApp) initRouterHandlers() { - - // All handlers must be added here. - // The order matters. - app.router.AddRoute("bank", bank.NewHandler(app.accountKeeper)) - app.router.AddRoute("sketchy", sketchy.NewHandler()) -} -``` - -In the Basecoin example, the sketchy handler isn't provided an account -mapper, which does provide the bank handler with the capability (in -conjunction with the context of a transaction run). - -## Application Architecture - -The SDK has multiple levels of "application": the ABCI app, the BaseApp, the BasecoinApp, and now your App. - -### ABCI App - -The basic ABCI interface allowing Tendermint to drive the applications state machine with transaction blocks. - -### BaseApp - -Implements an ABCI App using a MultiStore for persistence and a Router to handle transactions. -The goal is to provide a secure interface between the store and the extensible state machine -while defining as little about that state machine as possible (staying true to the ABCI). - -BaseApp requires stores to be mounted via capabilities keys - handlers can only access -stores they're given the key for. The BaseApp ensures all stores are properly loaded, cached, and committed. -One mounted store is considered the "main" - it holds the latest block header, from which we can find and load the most recent state. - -BaseApp distinguishes between two handler types - the `AnteHandler` and the `MsgHandler`. -The former is a global validity check (checking nonces, sigs and sufficient balances to pay fees, -e.g. things that apply to all transaction from all modules), the later is the full state transition function. -During CheckTx the state transition function is only applied to the checkTxState and should return -before any expensive state transitions are run (this is up to each developer). It also needs to return the estimated -gas cost. - -During DeliverTx the state transition function is applied to the blockchain state and the transactions -need to be fully executed. - -BaseApp is responsible for managing the context passed into handlers - it makes the block header available and provides the right stores for CheckTx and DeliverTx. BaseApp is completely agnostic to serialization formats. - -### Basecoin - -Basecoin is the first complete application in the stack. Complete applications require extensions to the core modules of the SDK to actually implement handler functionality. - -The native extensions of the SDK, useful for building Cosmos Zones, live under `x`. -Basecoin implements a `BaseApp` state machine using the `x/auth` and `x/bank` extensions, -which define how transaction signers are authenticated and how coins are transferred. -It should also use `x/ibc` and probably a simple staking extension. - -Basecoin and the native `x` extensions use go-amino for all serialization needs, -including for transactions and accounts. - -### Your Cosmos App - -Your Cosmos App is a fork of Basecoin - copy the `examples/basecoin` directory and modify it to your needs. -You may want to: - -- add fields to accounts -- copy and modify handlers -- add new handlers for new transaction types -- add new stores for better isolation across handlers - -The Cosmos Hub takes Basecoin and adds more stores and extensions to handle additional -transaction types and logic, like the advanced staking logic and the governance process. - -## Ethermint - -Ethermint is a new implementation of `BaseApp` that does not depend on Basecoin. -Instead of `cosmos-sdk/x/` it has its own `ethermint/x` based on `go-ethereum`. - -Ethermint uses a Patricia store for its accounts, and an IAVL store for IBC. -It has `x/ante`, which is quite similar to Basecoin's but uses RLP instead of go-amino. -Instead of `x/bank`, it has `x/eth`, which defines the single Ethereum transaction type -and all the semantics of the Ethereum state machine. - -Within `x/eth`, transactions sent to particular addresses can be handled in unique ways, -for instance to handle IBC and staking. diff --git a/docs/spec/README.md b/docs/spec/README.md index 49ea67b7a..047b1ca98 100644 --- a/docs/spec/README.md +++ b/docs/spec/README.md @@ -1,28 +1,31 @@ -# Cosmos Hub Spec +# Specifications -This directory contains specifications for the state transition machine of the -Cosmos Hub. +This directory contains specifications for the modules of the Cosmos SDK as well as Interchain Standards (ICS) and other specifications. -The Cosmos Hub holds all of its state in a Merkle store. Updates to +SDK applications hold this state in a Merkle store. Updates to the store may be made during transactions and at the beginning and end of every block. -While the first implementation of the Cosmos Hub is built using the Cosmos-SDK, -these specifications aim to be independent of any implementation details. That -said, they provide a detailed resource for understanding the Cosmos-SDK. +### SDK specifications: -- [Store](store) - The core Merkle store that holds the state. -- [Auth](auth) - The structure and authentication of accounts and transactions. -- [Bank](bank) - Sending tokens. -- [Governance](governance) - Proposals and voting. -- [Staking](staking) - Proof-of-stake bonding, delegation, etc. -- [Slashing](slashing) - Validator punishment mechanisms. -- [Distribution](distribution) - Fee distribution, and atom provision distribution -- [Inflation](inflation) - Atom provision creation -- [IBC](ibc) - Inter-Blockchain Communication (IBC) protocol. -- [Other](other) - Other components of the Cosmos Hub, including the reserve - pool, All in Bits vesting, etc. +- [Store](./store) - The core Merkle store that holds the state. +- [Bech32](./other/bech32.md) - Address format for Cosmos SDK applications. + +### Modules specifications: + +- [Auth](./auth) - The structure and authentication of accounts and transactions. +- [Bank](./bank) - Sending tokens. +- [Governance](./governance) - Proposals and voting. +- [Staking](./staking) - Proof-of-stake bonding, delegation, etc. +- [Slashing](./slashing) - Validator punishment mechanisms. +- [Distribution](./distribution) - Fee distribution, and staking token provision distribution . +- [Inflation](./inflation) - Staking token provision creation +- [IBC](./ibc) - Inter-Blockchain Communication (IBC) protocol. + +### Interchain standards + +- [ICS30](./ics/ics-030-signed-messages.md) - Signed messages standard. +- For details on the underlying blockchain and p2p protocols, see the [Tendermint specification](https://github.com/tendermint/tendermint/tree/develop/docs/spec). - diff --git a/docs/spec/distribution/overview.md b/docs/spec/distribution/overview.md index 28b1bbedd..8cbe9355a 100644 --- a/docs/spec/distribution/overview.md +++ b/docs/spec/distribution/overview.md @@ -2,71 +2,75 @@ ## Overview -This _simple_ distribution mechanism describes a functional way to passively -distribute rewards between validator and delegators. Note that this mechanism does -not distribute funds in as precisely as active reward distribution and will therefore -be upgraded in the future. +This _simple_ distribution mechanism describes a functional way to passively +distribute rewards between validators and delegators. Note that this mechanism does +not distribute funds in as precisely as active reward distribution mechanisms and +will therefore be upgraded in the future. The mechanism operates as follows. Collected rewards are pooled globally and divided out passively to validators and delegators. Each validator has the opportunity to charge commission to the delegators on the rewards collected on -behalf of the delegators by the validators. Fees are paid directly into a -global reward pool, and validator proposer-reward pool. Due to the nature of -passive accounting, whenever changes to parameters which affect the rate of reward -distribution occurs, withdrawal of rewards must also occur. +behalf of the delegators. Fees are collected directly into a global reward pool +and validator proposer-reward pool. Due to the nature of passive accounting, +whenever changes to parameters which affect the rate of reward distribution +occurs, withdrawal of rewards must also occur. - - Whenever withdrawing, one must withdraw the maximum amount they are entitled - too, leaving nothing in the pool. - - Whenever bonding, unbonding, or re-delegating tokens to an existing account, a +- Whenever withdrawing, one must withdraw the maximum amount they are entitled + to, leaving nothing in the pool. +- Whenever bonding, unbonding, or re-delegating tokens to an existing account, a full withdrawal of the rewards must occur (as the rules for lazy accounting change). - - Whenever a validator chooses to change the commission on rewards, all accumulated +- Whenever a validator chooses to change the commission on rewards, all accumulated commission rewards must be simultaneously withdrawn. The above scenarios are covered in `hooks.md`. -The distribution mechanism outlines herein is used to lazily distribute the +The distribution mechanism outlined herein is used to lazily distribute the following rewards between validators and associated delegators: - - multi-token fees to be socially distributed, - - proposer reward pool, - - inflated atom provisions, and - - validator commission on all rewards earned by their delegators stake + +- multi-token fees to be socially distributed +- proposer reward pool +- inflated atom provisions +- validator commission on all rewards earned by their delegators stake Fees are pooled within a global pool, as well as validator specific proposer-reward pools. The mechanisms used allow for validators and delegators to independently and lazily withdraw their rewards. -## Shortcomings +## Shortcomings As a part of the lazy computations, each delegator holds an accumulation term specific to each validator which is used to estimate what their approximate -fair portion of tokens held in the global fee pool is owed to them. +fair portion of tokens held in the global fee pool is owed to them. ``` entitlement = delegator-accumulation / all-delegators-accumulation ``` -Under the circumstance that there were constant and equal flow of incoming +Under the circumstance that there was constant and equal flow of incoming reward tokens every block, this distribution mechanism would be equal to the active distribution (distribute individually to all delegators each block). -However this is unrealistic so deviations from the active distribution will +However, this is unrealistic so deviations from the active distribution will occur based on fluctuations of incoming reward tokens as well as timing of -reward withdrawal by other delegators. +reward withdrawal by other delegators. -If you happen to know that incoming rewards are about significantly move up, +If you happen to know that incoming rewards are about to significantly increase, you are incentivized to not withdraw until after this event, increasing the -worth of your existing _accum_. +worth of your existing _accum_. See [#2764](https://github.com/cosmos/cosmos-sdk/issues/2764) +for further details. ## Affect on Staking Charging commission on Atom provisions while also allowing for Atom-provisions to be auto-bonded (distributed directly to the validators bonded stake) is -problematic within DPoS. Fundamentally these two mechanisms are mutually -exclusive. If there are Atom commissions and auto-bonding Atoms, the portion -of Atoms the reward distribution calculation would become very large as the Atom -portion for each delegator would change each block making a withdrawal of rewards -for a delegator require a calculation for every single block since the last -withdrawal. In conclusion, we can only have Atom commission and unbonded atoms +problematic within BPoS. Fundamentally, these two mechanisms are mutually +exclusive. If both commission and auto-bonding mechanisms are simultaneously +applied to the staking-token then the distribution of staking-tokens between +any validator and its delegators will change with each block. This then +necessitates a calculation for each delegation records for each block - +which is considered computationally expensive. + +In conclusion, we can only have Atom commission and unbonded atoms provisions or bonded atom provisions with no Atom commission, and we elect to implement the former. Stakeholders wishing to rebond their provisions may elect -to set up a script to periodically withdraw and rebond rewards. +to set up a script to periodically withdraw and rebond rewards. diff --git a/docs/spec/governance/README.md b/docs/spec/governance/README.md index 0ddbc7cfd..6cc1edb50 100644 --- a/docs/spec/governance/README.md +++ b/docs/spec/governance/README.md @@ -2,7 +2,7 @@ ## Abstract -This paper specifies the Governance module of the Cosmos-SDK, which was first described in the [Cosmos Whitepaper](https://cosmos.network/about/whitepaper) in June 2016. +This paper specifies the Governance module of the Cosmos-SDK, which was first described in the [Cosmos Whitepaper](https://cosmos.network/about/whitepaper) in June 2016. The module enables Cosmos-SDK based blockchain to support an on-chain governance system. In this system, holders of the native staking token of the chain can vote on proposals on a 1 token 1 vote basis. Next is a list of features the module currently supports: @@ -24,7 +24,7 @@ The following specification uses *Atom* as the native staking token. The module 1. **[Design overview](overview.md)** 2. **Implementation** 1. **[State](state.md)** - 1. Procedures + 1. Parameters 2. Proposals 3. Proposal Processing Queue 2. **[Transactions](transactions.md)** diff --git a/docs/spec/governance/state.md b/docs/spec/governance/state.md index ca130bd4c..3858ba06b 100644 --- a/docs/spec/governance/state.md +++ b/docs/spec/governance/state.md @@ -2,36 +2,35 @@ ## State -### Procedures and base types - -`Procedures` define the rule according to which votes are run. There can only -be one active procedure at any given time. If governance wants to change a -procedure, either to modify a value or add/remove a parameter, a new procedure -has to be created and the previous one rendered inactive. +### Parameters and base types +`Parameters` define the rules according to which votes are run. There can only +be one active parameter set at any given time. If governance wants to change a +parameter set, either to modify a value or add/remove a parameter field, a new +parameter set has to be created and the previous one rendered inactive. ```go -type DepositProcedure struct { - MinDeposit sdk.Coins // Minimum deposit for a proposal to enter voting period. - MaxDepositPeriod time.Time // Maximum period for Atom holders to deposit on a proposal. Initial value: 2 months +type DepositParams struct { + MinDeposit sdk.Coins // Minimum deposit for a proposal to enter voting period. + MaxDepositPeriod time.Time // Maximum period for Atom holders to deposit on a proposal. Initial value: 2 months } ``` ```go -type VotingProcedure struct { - VotingPeriod time.Time // Length of the voting period. Initial value: 2 weeks +type VotingParams struct { + VotingPeriod time.Time // Length of the voting period. Initial value: 2 weeks } ``` ```go -type TallyingProcedure struct { - Threshold sdk.Dec // Minimum propotion of Yes votes for proposal to pass. Initial value: 0.5 - Veto sdk.Dec // Minimum proportion of Veto votes to Total votes ratio for proposal to be vetoed. Initial value: 1/3 - GovernancePenalty sdk.Dec // Penalty if validator does not vote +type TallyParams struct { + Threshold sdk.Dec // Minimum proportion of Yes votes for proposal to pass. Initial value: 0.5 + Veto sdk.Dec // Minimum proportion of Veto votes to Total votes ratio for proposal to be vetoed. Initial value: 1/3 + GovernancePenalty sdk.Dec // Penalty if validator does not vote } ``` -Procedures are stored in a global `GlobalParams` KVStore. +Parameters are stored in a global `GlobalParams` KVStore. Additionally, we introduce some basic types: @@ -61,7 +60,7 @@ const ( ProposalStatusActive = 0x2 // MinDeposit is reachhed, participants can vote ProposalStatusAccepted = 0x3 // Proposal has been accepted ProposalStatusRejected = 0x4 // Proposal has been rejected - ProposalStatusClosed. = 0x5 // Proposal never reached MinDeposit + ProposalStatusClosed = 0x5 // Proposal never reached MinDeposit ) ``` @@ -76,7 +75,7 @@ const ( ### ValidatorGovInfo -This type is used in a temp map when tallying +This type is used in a temp map when tallying ```go type ValidatorGovInfo struct { @@ -87,7 +86,7 @@ This type is used in a temp map when tallying ### Proposals -`Proposals` are an item to be voted on. +`Proposals` are an item to be voted on. ```go type Proposal struct { @@ -99,7 +98,7 @@ type Proposal struct { SubmitTime time.Time // Time of the block where TxGovSubmitProposal was included DepositEndTime time.Time // Time that the DepositPeriod of a proposal would expire Submitter sdk.AccAddress // Address of the submitter - + VotingStartTime time.Time // Time of the block where MinDeposit was reached. time.Time{} if MinDeposit is not reached VotingEndTime time.Time // Time of the block that the VotingPeriod for a proposal will end. CurrentStatus ProposalStatus // Current status of the proposal @@ -135,7 +134,7 @@ For pseudocode purposes, here are the two function we will use to read or write ### Proposal Processing Queue **Store:** -* `ProposalProcessingQueue`: A queue `queue[proposalID]` containing all the +* `ProposalProcessingQueue`: A queue `queue[proposalID]` containing all the `ProposalIDs` of proposals that reached `MinDeposit`. Each `EndBlock`, all the proposals that have reached the end of their voting period are processed. To process a finished proposal, the application tallies the votes, compute the votes of @@ -145,8 +144,8 @@ For pseudocode purposes, here are the two function we will use to read or write And the pseudocode for the `ProposalProcessingQueue`: ```go - in EndBlock do - + in EndBlock do + for finishedProposalID in GetAllFinishedProposalIDs(block.Time) proposal = load(Governance, <proposalID|'proposal'>) // proposal is a const key @@ -171,7 +170,7 @@ And the pseudocode for the `ProposalProcessingQueue`: if (isVal) tmpValMap(voterAddress).Vote = vote - tallyingProcedure = load(GlobalParams, 'TallyingProcedure') + tallyingParam = load(GlobalParams, 'TallyingParam') // Update tally if validator voted they voted for each validator in validators @@ -182,14 +181,14 @@ And the pseudocode for the `ProposalProcessingQueue`: // Check if proposal is accepted or rejected totalNonAbstain := proposal.YesVotes + proposal.NoVotes + proposal.NoWithVetoVotes - if (proposal.Votes.YesVotes/totalNonAbstain > tallyingProcedure.Threshold AND proposal.Votes.NoWithVetoVotes/totalNonAbstain < tallyingProcedure.Veto) + if (proposal.Votes.YesVotes/totalNonAbstain > tallyingParam.Threshold AND proposal.Votes.NoWithVetoVotes/totalNonAbstain < tallyingParam.Veto) // proposal was accepted at the end of the voting period // refund deposits (non-voters already punished) proposal.CurrentStatus = ProposalStatusAccepted for each (amount, depositer) in proposal.Deposits depositer.AtomBalance += amount - else + else // proposal was rejected proposal.CurrentStatus = ProposalStatusRejected diff --git a/docs/spec/governance/transactions.md b/docs/spec/governance/transactions.md index f3636facd..68966bcfc 100644 --- a/docs/spec/governance/transactions.md +++ b/docs/spec/governance/transactions.md @@ -4,7 +4,7 @@ ### Proposal Submission -Proposals can be submitted by any Atom holder via a `TxGovSubmitProposal` +Proposals can be submitted by any Atom holder via a `TxGovSubmitProposal` transaction. ```go @@ -24,7 +24,7 @@ type TxGovSubmitProposal struct { * If `MinDeposit` is reached: * Push `proposalID` in `ProposalProcessingQueue` -A `TxGovSubmitProposal` transaction can be handled according to the following +A `TxGovSubmitProposal` transaction can be handled according to the following pseudocode. ```go @@ -32,31 +32,31 @@ pseudocode. // Check if TxGovSubmitProposal is valid. If it is, create proposal // upon receiving txGovSubmitProposal from sender do - - if !correctlyFormatted(txGovSubmitProposal) + + if !correctlyFormatted(txGovSubmitProposal) // check if proposal is correctly formatted. Includes fee payment. throw - - initialDeposit = txGovSubmitProposal.InitialDeposit - if (initialDeposit.Atoms <= 0) OR (sender.AtomBalance < initialDeposit.Atoms) + + initialDeposit = txGovSubmitProposal.InitialDeposit + if (initialDeposit.Atoms <= 0) OR (sender.AtomBalance < initialDeposit.Atoms) // InitialDeposit is negative or null OR sender has insufficient funds throw if (txGovSubmitProposal.Type != ProposalTypePlainText) OR (txGovSubmitProposal.Type != ProposalTypeSoftwareUpgrade) - + sender.AtomBalance -= initialDeposit.Atoms - depositProcedure = load(GlobalParams, 'DepositProcedure') - + depositParam = load(GlobalParams, 'DepositParam') + proposalID = generate new proposalID proposal = NewProposal() - + proposal.Title = txGovSubmitProposal.Title proposal.Description = txGovSubmitProposal.Description proposal.Type = txGovSubmitProposal.Type proposal.TotalDeposit = initialDeposit proposal.SubmitTime = <CurrentTime> - proposal.DepositEndTime = <CurrentTime>.Add(depositProcedure.MaxDepositPeriod) + proposal.DepositEndTime = <CurrentTime>.Add(depositParam.MaxDepositPeriod) proposal.Deposits.append({initialDeposit, sender}) proposal.Submitter = sender proposal.YesVotes = 0 @@ -64,15 +64,15 @@ upon receiving txGovSubmitProposal from sender do proposal.NoWithVetoVotes = 0 proposal.AbstainVotes = 0 proposal.CurrentStatus = ProposalStatusOpen - + store(Proposals, <proposalID|'proposal'>, proposal) // Store proposal in Proposals mapping return proposalID ``` ### Deposit -Once a proposal is submitted, if -`Proposal.TotalDeposit < ActiveProcedure.MinDeposit`, Atom holders can send +Once a proposal is submitted, if +`Proposal.TotalDeposit < ActiveParam.MinDeposit`, Atom holders can send `TxGovDeposit` transactions to increase the proposal's deposit. ```go @@ -89,7 +89,7 @@ type TxGovDeposit struct { * If `MinDeposit` is reached: * Push `proposalID` in `ProposalProcessingQueueEnd` -A `TxGovDeposit` transaction has to go through a number of checks to be valid. +A `TxGovDeposit` transaction has to go through a number of checks to be valid. These checks are outlined in the following pseudocode. ```go @@ -98,27 +98,27 @@ These checks are outlined in the following pseudocode. upon receiving txGovDeposit from sender do // check if proposal is correctly formatted. Includes fee payment. - - if !correctlyFormatted(txGovDeposit) + + if !correctlyFormatted(txGovDeposit) throw - + proposal = load(Proposals, <txGovDeposit.ProposalID|'proposal'>) // proposal is a const key, proposalID is variable - if (proposal == nil) + if (proposal == nil) // There is no proposal for this proposalID throw - + if (txGovDeposit.Deposit.Atoms <= 0) OR (sender.AtomBalance < txGovDeposit.Deposit.Atoms) OR (proposal.CurrentStatus != ProposalStatusOpen) - // deposit is negative or null + // deposit is negative or null // OR sender has insufficient funds // OR proposal is not open for deposit anymore throw - depositProcedure = load(GlobalParams, 'DepositProcedure') + depositParam = load(GlobalParams, 'DepositParam') - if (CurrentBlock >= proposal.SubmitBlock + depositProcedure.MaxDepositPeriod) + if (CurrentBlock >= proposal.SubmitBlock + depositParam.MaxDepositPeriod) proposal.CurrentStatus = ProposalStatusClosed else @@ -127,21 +127,21 @@ upon receiving txGovDeposit from sender do proposal.Deposits.append({txGovVote.Deposit, sender}) proposal.TotalDeposit.Plus(txGovDeposit.Deposit) - - if (proposal.TotalDeposit >= depositProcedure.MinDeposit) + + if (proposal.TotalDeposit >= depositParam.MinDeposit) // MinDeposit is reached, vote opens - + proposal.VotingStartBlock = CurrentBlock proposal.CurrentStatus = ProposalStatusActive - ProposalProcessingQueue.push(txGovDeposit.ProposalID) + ProposalProcessingQueue.push(txGovDeposit.ProposalID) store(Proposals, <txGovVote.ProposalID|'proposal'>, proposal) ``` ### Vote -Once `ActiveProcedure.MinDeposit` is reached, voting period starts. From there, -bonded Atom holders are able to send `TxGovVote` transactions to cast their +Once `ActiveParam.MinDeposit` is reached, voting period starts. From there, +bonded Atom holders are able to send `TxGovVote` transactions to cast their vote on the proposal. ```go @@ -157,25 +157,25 @@ vote on the proposal. *Note: Gas cost for this message has to take into account the future tallying of the vote in EndBlocker* -Next is a pseudocode proposal of the way `TxGovVote` transactions are +Next is a pseudocode proposal of the way `TxGovVote` transactions are handled: ```go // PSEUDOCODE // // Check if TxGovVote is valid. If it is, count vote// - + upon receiving txGovVote from sender do - // check if proposal is correctly formatted. Includes fee payment. - - if !correctlyFormatted(txGovDeposit) + // check if proposal is correctly formatted. Includes fee payment. + + if !correctlyFormatted(txGovDeposit) throw - + proposal = load(Proposals, <txGovDeposit.ProposalID|'proposal'>) - if (proposal == nil) + if (proposal == nil) // There is no proposal for this proposalID throw - + if (proposal.CurrentStatus == ProposalStatusActive) diff --git a/docs/ics/ics-030-signed-messages.md b/docs/spec/ics/ics-030-signed-messages.md similarity index 100% rename from docs/ics/ics-030-signed-messages.md rename to docs/spec/ics/ics-030-signed-messages.md diff --git a/docs/spec-proposals/f1-fee-distribution/f1_fee_distr.pdf b/docs/spec/spec-proposals/f1-fee-distribution/f1_fee_distr.pdf similarity index 100% rename from docs/spec-proposals/f1-fee-distribution/f1_fee_distr.pdf rename to docs/spec/spec-proposals/f1-fee-distribution/f1_fee_distr.pdf diff --git a/docs/spec-proposals/f1-fee-distribution/f1_fee_distr.tex b/docs/spec/spec-proposals/f1-fee-distribution/f1_fee_distr.tex similarity index 100% rename from docs/spec-proposals/f1-fee-distribution/f1_fee_distr.tex rename to docs/spec/spec-proposals/f1-fee-distribution/f1_fee_distr.tex diff --git a/scripts/Makefile b/scripts/Makefile new file mode 100644 index 000000000..9abc497e4 --- /dev/null +++ b/scripts/Makefile @@ -0,0 +1,54 @@ +### +# Find OS and Go environment +# GO contains the Go binary +# FS contains the OS file separator +### +ifeq ($(OS),Windows_NT) + GO := $(shell where go.exe 2> NUL) + FS := \\ +else + GO := $(shell command -v go 2> /dev/null) + FS := / +endif + +ifeq ($(GO),) + $(error could not find go. Is it in PATH? $(GO)) +endif + +GOPATH ?= $(shell $(GO) env GOPATH) +GITHUBDIR := $(GOPATH)$(FS)src$(FS)github.com + +### +# Functions +### + +go_get = $(if $(findstring Windows_NT,$(OS)),\ +IF NOT EXIST $(GITHUBDIR)$(FS)$(1)$(FS) ( mkdir $(GITHUBDIR)$(FS)$(1) ) else (cd .) &\ +IF NOT EXIST $(GITHUBDIR)$(FS)$(1)$(FS)$(2)$(FS) ( cd $(GITHUBDIR)$(FS)$(1) && git clone https://github.com/$(1)/$(2) ) else (cd .) &\ +,\ +mkdir -p $(GITHUBDIR)$(FS)$(1) &&\ +(test ! -d $(GITHUBDIR)$(FS)$(1)$(FS)$(2) && cd $(GITHUBDIR)$(FS)$(1) && git clone https://github.com/$(1)/$(2)) || true &&\ +)\ +cd $(GITHUBDIR)$(FS)$(1)$(FS)$(2) && git fetch origin && git checkout -q $(3) + +go_install = $(call go_get,$(1),$(2),$(3)) && cd $(GITHUBDIR)$(FS)$(1)$(FS)$(2) && $(GO) install + +### +# get_tools +### +all: get_tools +get_tools: dep gometalinter statik + +dep: + $(call go_get,golang,dep,22125cfaa6ddc71e145b1535d4b7ee9744fefff2) + cd $(GITHUBDIR)$(FS)golang$(FS)dep$(FS)cmd$(FS)dep && $(GO) install + +#v2.0.11 +gometalinter: + $(call go_install,alecthomas,gometalinter,17a7ffa42374937bfecabfb8d2efbd4db0c26741) + +statik: + $(call go_install,rakyll,statik,v0.1.5) + +.PHONY: all get_tools dep gometalinter statik + diff --git a/scripts/get_tools.sh b/scripts/get_tools.sh deleted file mode 100755 index 79ab32dec..000000000 --- a/scripts/get_tools.sh +++ /dev/null @@ -1,49 +0,0 @@ -#!/usr/bin/env bash -set -e - -# This file downloads all of the binary dependencies we have, and checks out a -# specific git hash. -# -# repos it installs: -# github.com/golang/dep/cmd/dep -# gopkg.in/alecthomas/gometalinter.v2 -# github.com/rakyll/statiik - -## check if GOPATH is set -if [ -z ${GOPATH+x} ]; then - echo "please set GOPATH (https://github.com/golang/go/wiki/SettingGOPATH)" - exit 1 -fi - -mkdir -p "$GOPATH/src/github.com" -cd "$GOPATH/src/github.com" || exit 1 - -installFromGithub() { - repo=$1 - commit=$2 - # optional - subdir=$3 - echo "--> Installing $repo ($commit)..." - if [ ! -d "$repo" ]; then - mkdir -p "$repo" - git clone "https://github.com/$repo.git" "$repo" - fi - if [ ! -z ${subdir+x} ] && [ ! -d "$repo/$subdir" ]; then - echo "ERROR: no such directory $repo/$subdir" - exit 1 - fi - pushd "$repo" && \ - git fetch origin && \ - git checkout -q "$commit" && \ - if [ ! -z ${subdir+x} ]; then cd "$subdir" || exit 1; fi && \ - go install && \ - if [ ! -z ${subdir+x} ]; then cd - || exit 1; fi && \ - popd || exit 1 - echo "--> Done" - echo "" -} - -installFromGithub golang/dep 22125cfaa6ddc71e145b1535d4b7ee9744fefff2 cmd/dep -## gometalinter v2.0.11 -installFromGithub alecthomas/gometalinter 17a7ffa42374937bfecabfb8d2efbd4db0c26741 -installFromGithub rakyll/statik v0.1.5 \ No newline at end of file diff --git a/scripts/import-export-sim.sh b/scripts/import-export-sim.sh new file mode 100755 index 000000000..8ace8e0d5 --- /dev/null +++ b/scripts/import-export-sim.sh @@ -0,0 +1,54 @@ +#!/bin/bash + +seeds=(1 2 4 7 9 20 32 123 124 582 1893 2989 3012 4728 37827 981928 87821 891823782 989182 89182391 \ +11 22 44 77 99 2020 3232 123123 124124 582582 18931893 29892989 30123012 47284728 37827) +blocks=$1 + +echo "Running multi-seed import-export simulation with seeds ${seeds[@]}" +echo "Running $blocks blocks per seed" +echo "Edit scripts/import-export-sim.sh to add new seeds. Keeping parameters in the file makes failures easy to reproduce." +echo "This script will kill all sub-simulations on SIGINT/SIGTERM (i.e. Ctrl-C)." + +trap 'kill $(jobs -pr)' SIGINT SIGTERM + +tmpdir=$(mktemp -d) +echo "Using temporary log directory: $tmpdir" + +sim() { + seed=$1 + echo "Running import/export Gaia simulation with seed $seed. This may take awhile!" + file="$tmpdir/gaia-simulation-seed-$seed-date-$(date -Iseconds -u).stdout" + echo "Writing stdout to $file..." + go test ./cmd/gaia/app -run TestGaiaImportExport -SimulationEnabled=true -SimulationNumBlocks=$blocks \ + -SimulationBlockSize=200 -SimulationCommit=true -SimulationSeed=$seed -v -timeout 24h > $file +} + +i=0 +pids=() +for seed in ${seeds[@]}; do + sim $seed & + pids[${i}]=$! + i=$(($i+1)) + sleep 10 # start in order, nicer logs +done + +echo "Simulation processes spawned, waiting for completion..." + +code=0 + +i=0 +for pid in ${pids[*]}; do + wait $pid + last=$? + seed=${seeds[${i}]} + if [ $last -ne 0 ] + then + echo "Import/export simulation with seed $seed failed!" + code=1 + else + echo "Import/export simulation with seed $seed OK" + fi + i=$(($i+1)) +done + +exit $code diff --git a/types/store.go b/types/store.go index 4b6e79a76..8fe0321f5 100644 --- a/types/store.go +++ b/types/store.go @@ -197,15 +197,16 @@ func DiffKVStores(a KVStore, b KVStore, prefixesToSkip [][]byte) (kvA cmn.KVPair kvB = cmn.KVPair{Key: iterB.Key(), Value: iterB.Value()} iterB.Next() } + if !bytes.Equal(kvA.Key, kvB.Key) { + return kvA, kvB, count, false + } compareValue := true for _, prefix := range prefixesToSkip { + // Skip value comparison if we matched a prefix if bytes.Equal(kvA.Key[:len(prefix)], prefix) { compareValue = false } } - if !bytes.Equal(kvA.Key, kvB.Key) { - return kvA, kvB, count, false - } if compareValue && !bytes.Equal(kvA.Value, kvB.Value) { return kvA, kvB, count, false } diff --git a/x/gov/client/cli/tx.go b/x/gov/client/cli/tx.go index 455bd58db..31780f68e 100644 --- a/x/gov/client/cli/tx.go +++ b/x/gov/client/cli/tx.go @@ -249,6 +249,30 @@ func GetCmdVote(cdc *codec.Codec) *cobra.Command { return cmd } +// GetCmdQueryProposal implements the query proposal command. +func GetCmdQueryParams(queryRoute string, cdc *codec.Codec) *cobra.Command { + cmd := &cobra.Command{ + Use: "param [param-type]", + Short: "Query the parameters (voting|tallying|deposit) of the governance process", + Args: cobra.ExactArgs(1), + RunE: func(cmd *cobra.Command, args []string) error { + paramType := args[0] + + cliCtx := context.NewCLIContext().WithCodec(cdc) + + res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/params/%s", queryRoute, paramType), nil) + if err != nil { + return err + } + + fmt.Println(string(res)) + return nil + }, + } + + return cmd +} + // GetCmdQueryProposal implements the query proposal command. func GetCmdQueryProposal(queryRoute string, cdc *codec.Codec) *cobra.Command { cmd := &cobra.Command{ @@ -504,7 +528,7 @@ func GetCmdQueryDeposits(queryRoute string, cdc *codec.Codec) *cobra.Command { return cmd } -// GetCmdQueryDeposits implements the command to query for proposal deposits. +// GetCmdQueryTally implements the command to query for proposal tally result. func GetCmdQueryTally(queryRoute string, cdc *codec.Codec) *cobra.Command { cmd := &cobra.Command{ Use: "tally", diff --git a/x/gov/client/rest/rest.go b/x/gov/client/rest/rest.go index f477f2ff9..930a39b02 100644 --- a/x/gov/client/rest/rest.go +++ b/x/gov/client/rest/rest.go @@ -18,7 +18,8 @@ import ( // REST Variable names // nolint const ( - RestProposalID = "proposalId" + RestParamsType = "type" + RestProposalID = "proposal-id" RestDepositer = "depositer" RestVoter = "voter" RestProposalStatus = "status" @@ -32,6 +33,11 @@ func RegisterRoutes(cliCtx context.CLIContext, r *mux.Router, cdc *codec.Codec) r.HandleFunc(fmt.Sprintf("/gov/proposals/{%s}/deposits", RestProposalID), depositHandlerFn(cdc, cliCtx)).Methods("POST") r.HandleFunc(fmt.Sprintf("/gov/proposals/{%s}/votes", RestProposalID), voteHandlerFn(cdc, cliCtx)).Methods("POST") + r.HandleFunc( + fmt.Sprintf("/gov/parameters/{%s}", RestParamsType), + queryParamsHandlerFn(cdc, cliCtx), + ).Methods("GET") + r.HandleFunc("/gov/proposals", queryProposalsWithParameterFn(cdc, cliCtx)).Methods("GET") r.HandleFunc(fmt.Sprintf("/gov/proposals/{%s}", RestProposalID), queryProposalHandlerFn(cdc, cliCtx)).Methods("GET") r.HandleFunc(fmt.Sprintf("/gov/proposals/{%s}/deposits", RestProposalID), queryDepositsHandlerFn(cdc, cliCtx)).Methods("GET") @@ -177,6 +183,21 @@ func voteHandlerFn(cdc *codec.Codec, cliCtx context.CLIContext) http.HandlerFunc } } +func queryParamsHandlerFn(cdc *codec.Codec, cliCtx context.CLIContext) http.HandlerFunc { + return func(w http.ResponseWriter, r *http.Request) { + vars := mux.Vars(r) + paramType := vars[RestParamsType] + + res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/gov/%s/%s", gov.QueryParams, paramType), nil) + if err != nil { + utils.WriteErrorResponse(w, http.StatusNotFound, err.Error()) + return + } + + utils.PostProcessResponse(w, cdc, res, cliCtx.Indent) + } +} + func queryProposalHandlerFn(cdc *codec.Codec, cliCtx context.CLIContext) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { vars := mux.Vars(r) diff --git a/x/gov/keeper.go b/x/gov/keeper.go index 862db8274..4f370ef53 100644 --- a/x/gov/keeper.go +++ b/x/gov/keeper.go @@ -250,7 +250,7 @@ func (keeper Keeper) GetDepositParams(ctx sdk.Context) DepositParams { return depositParams } -// Returns the current Voting Procedure from the global param store +// Returns the current VotingParams from the global param store // nolint: errcheck func (keeper Keeper) GetVotingParams(ctx sdk.Context) VotingParams { var votingParams VotingParams @@ -258,7 +258,7 @@ func (keeper Keeper) GetVotingParams(ctx sdk.Context) VotingParams { return votingParams } -// Returns the current Tallying Procedure from the global param store +// Returns the current TallyParam from the global param store // nolint: errcheck func (keeper Keeper) GetTallyParams(ctx sdk.Context) TallyParams { var tallyParams TallyParams diff --git a/x/gov/params.go b/x/gov/params.go index 4d8126d14..b9e1efa87 100644 --- a/x/gov/params.go +++ b/x/gov/params.go @@ -6,20 +6,20 @@ import ( sdk "github.com/cosmos/cosmos-sdk/types" ) -// Procedure around Deposits for governance +// Param around Deposits for governance type DepositParams struct { MinDeposit sdk.Coins `json:"min_deposit"` // Minimum deposit for a proposal to enter voting period. MaxDepositPeriod time.Duration `json:"max_deposit_period"` // Maximum period for Atom holders to deposit on a proposal. Initial value: 2 months } -// Procedure around Tallying votes in governance +// Param around Tallying votes in governance type TallyParams struct { Threshold sdk.Dec `json:"threshold"` // Minimum propotion of Yes votes for proposal to pass. Initial value: 0.5 Veto sdk.Dec `json:"veto"` // Minimum value of Veto votes to Total votes ratio for proposal to be vetoed. Initial value: 1/3 GovernancePenalty sdk.Dec `json:"governance_penalty"` // Penalty if validator does not vote } -// Procedure around Voting in governance +// Param around Voting in governance type VotingParams struct { VotingPeriod time.Duration `json:"voting_period"` // Length of the voting period. } diff --git a/x/gov/queryable.go b/x/gov/querier.go similarity index 86% rename from x/gov/queryable.go rename to x/gov/querier.go index 3e70d43aa..171ab469d 100644 --- a/x/gov/queryable.go +++ b/x/gov/querier.go @@ -1,6 +1,8 @@ package gov import ( + "fmt" + "github.com/cosmos/cosmos-sdk/codec" sdk "github.com/cosmos/cosmos-sdk/types" abci "github.com/tendermint/tendermint/abci/types" @@ -8,6 +10,7 @@ import ( // query endpoints supported by the governance Querier const ( + QueryParams = "params" QueryProposals = "proposals" QueryProposal = "proposal" QueryDeposits = "deposits" @@ -15,11 +18,17 @@ const ( QueryVotes = "votes" QueryVote = "vote" QueryTally = "tally" + + ParamDeposit = "deposit" + ParamVoting = "voting" + ParamTallying = "tallying" ) func NewQuerier(keeper Keeper) sdk.Querier { return func(ctx sdk.Context, path []string, req abci.RequestQuery) (res []byte, err sdk.Error) { switch path[0] { + case QueryParams: + return queryParams(ctx, path[1:], req, keeper) case QueryProposals: return queryProposals(ctx, path[1:], req, keeper) case QueryProposal: @@ -40,6 +49,31 @@ func NewQuerier(keeper Keeper) sdk.Querier { } } +func queryParams(ctx sdk.Context, path []string, req abci.RequestQuery, keeper Keeper) (res []byte, err sdk.Error) { + switch path[0] { + case ParamDeposit: + bz, err2 := codec.MarshalJSONIndent(keeper.cdc, keeper.GetDepositParams(ctx)) + if err2 != nil { + return nil, sdk.ErrInternal(sdk.AppendMsgToErr("could not marshal result to JSON", err2.Error())) + } + return bz, nil + case ParamVoting: + bz, err2 := codec.MarshalJSONIndent(keeper.cdc, keeper.GetVotingParams(ctx)) + if err2 != nil { + return nil, sdk.ErrInternal(sdk.AppendMsgToErr("could not marshal result to JSON", err2.Error())) + } + return bz, nil + case ParamTallying: + bz, err2 := codec.MarshalJSONIndent(keeper.cdc, keeper.GetTallyParams(ctx)) + if err2 != nil { + return nil, sdk.ErrInternal(sdk.AppendMsgToErr("could not marshal result to JSON", err2.Error())) + } + return bz, nil + default: + return res, sdk.ErrUnknownRequest(fmt.Sprintf("%s is not a valid query request path", req.Path)) + } +} + // Params for query 'custom/gov/proposal' type QueryProposalParams struct { ProposalID uint64 diff --git a/x/gov/querier_test.go b/x/gov/querier_test.go new file mode 100644 index 000000000..eee58d2ac --- /dev/null +++ b/x/gov/querier_test.go @@ -0,0 +1,306 @@ +package gov + +import ( + "strings" + "testing" + + "github.com/cosmos/cosmos-sdk/codec" + sdk "github.com/cosmos/cosmos-sdk/types" + "github.com/stretchr/testify/require" + abci "github.com/tendermint/tendermint/abci/types" +) + +func getQueriedParams(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier) (DepositParams, VotingParams, TallyParams) { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryParams, ParamDeposit}, "/"), + Data: []byte{}, + } + + bz, err := querier(ctx, []string{QueryParams, ParamDeposit}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var depositParams DepositParams + err2 := cdc.UnmarshalJSON(bz, &depositParams) + require.Nil(t, err2) + + query = abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryParams, ParamVoting}, "/"), + Data: []byte{}, + } + + bz, err = querier(ctx, []string{QueryParams, ParamVoting}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var votingParams VotingParams + err2 = cdc.UnmarshalJSON(bz, &votingParams) + require.Nil(t, err2) + + query = abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryParams, ParamTallying}, "/"), + Data: []byte{}, + } + + bz, err = querier(ctx, []string{QueryParams, ParamTallying}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var tallyParams TallyParams + err2 = cdc.UnmarshalJSON(bz, &tallyParams) + require.Nil(t, err2) + + return depositParams, votingParams, tallyParams +} + +func getQueriedProposal(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64) Proposal { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryProposal}, "/"), + Data: cdc.MustMarshalJSON(QueryProposalParams{ + ProposalID: proposalID, + }), + } + + bz, err := querier(ctx, []string{QueryProposal}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var proposal Proposal + err2 := cdc.UnmarshalJSON(bz, proposal) + require.Nil(t, err2) + return proposal +} + +func getQueriedProposals(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, depositer, voter sdk.AccAddress, status ProposalStatus, limit uint64) []Proposal { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryProposals}, "/"), + Data: cdc.MustMarshalJSON(QueryProposalsParams{ + Voter: voter, + Depositer: depositer, + ProposalStatus: status, + Limit: limit, + }), + } + + bz, err := querier(ctx, []string{QueryProposal}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var proposals []Proposal + err2 := cdc.UnmarshalJSON(bz, proposals) + require.Nil(t, err2) + return proposals +} + +func getQueriedDeposit(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64, depositer sdk.AccAddress) Deposit { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryDeposit}, "/"), + Data: cdc.MustMarshalJSON(QueryDepositParams{ + ProposalID: proposalID, + Depositer: depositer, + }), + } + + bz, err := querier(ctx, []string{QueryDeposits}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var deposit Deposit + err2 := cdc.UnmarshalJSON(bz, deposit) + require.Nil(t, err2) + return deposit +} + +func getQueriedDeposits(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64) []Deposit { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryDeposits}, "/"), + Data: cdc.MustMarshalJSON(QueryDepositsParams{ + ProposalID: proposalID, + }), + } + + bz, err := querier(ctx, []string{QueryDeposits}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var deposits []Deposit + err2 := cdc.UnmarshalJSON(bz, &deposits) + require.Nil(t, err2) + return deposits +} + +func getQueriedVote(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64, voter sdk.AccAddress) Vote { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryVote}, "/"), + Data: cdc.MustMarshalJSON(QueryVoteParams{ + ProposalID: proposalID, + Voter: voter, + }), + } + + bz, err := querier(ctx, []string{QueryVote}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var vote Vote + err2 := cdc.UnmarshalJSON(bz, &vote) + require.Nil(t, err2) + return vote +} + +func getQueriedVotes(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64) []Vote { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryVote}, "/"), + Data: cdc.MustMarshalJSON(QueryVotesParams{ + ProposalID: proposalID, + }), + } + + bz, err := querier(ctx, []string{QueryVotes}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var votes []Vote + err2 := cdc.UnmarshalJSON(bz, &votes) + require.Nil(t, err2) + return votes +} + +func getQueriedTally(t *testing.T, ctx sdk.Context, cdc *codec.Codec, querier sdk.Querier, proposalID uint64) TallyResult { + query := abci.RequestQuery{ + Path: strings.Join([]string{"custom", "gov", QueryTally}, "/"), + Data: cdc.MustMarshalJSON(QueryTallyParams{ + ProposalID: proposalID, + }), + } + + bz, err := querier(ctx, []string{QueryTally}, query) + require.Nil(t, err) + require.NotNil(t, bz) + + var tally TallyResult + err2 := cdc.UnmarshalJSON(bz, &tally) + require.Nil(t, err2) + return tally +} + +func testQueryParams(t *testing.T) { + cdc := codec.New() + mapp, keeper, _, _, _, _ := getMockApp(t, 1000) + querier := NewQuerier(keeper) + ctx := mapp.NewContext(false, abci.Header{}) + + getQueriedParams(t, ctx, cdc, querier) +} + +func testQueries(t *testing.T) { + cdc := codec.New() + mapp, keeper, _, addrs, _, _ := getMockApp(t, 1000) + querier := NewQuerier(keeper) + handler := NewHandler(keeper) + ctx := mapp.NewContext(false, abci.Header{}) + + depositParams, _, _ := getQueriedParams(t, ctx, cdc, querier) + + // addrs[0] proposes (and deposits) proposals #1 and #2 + res := handler(ctx, NewMsgSubmitProposal("title", "description", ProposalTypeText, addrs[0], sdk.Coins{sdk.NewInt64Coin("dummycoin", 1)})) + var proposalID1 uint64 + cdc.MustUnmarshalBinaryLengthPrefixed(res.Data, &proposalID1) + + res = handler(ctx, NewMsgSubmitProposal("title", "description", ProposalTypeText, addrs[0], sdk.Coins{sdk.NewInt64Coin("dummycoin", 1)})) + var proposalID2 uint64 + cdc.MustUnmarshalBinaryLengthPrefixed(res.Data, &proposalID2) + + // addrs[1] proposes (and deposits) proposals #3 + res = handler(ctx, NewMsgSubmitProposal("title", "description", ProposalTypeText, addrs[1], sdk.Coins{sdk.NewInt64Coin("dummycoin", 1)})) + var proposalID3 uint64 + cdc.MustUnmarshalBinaryLengthPrefixed(res.Data, &proposalID3) + + // addrs[1] deposits on proposals #2 & #3 + res = handler(ctx, NewMsgDeposit(addrs[1], proposalID2, depositParams.MinDeposit)) + res = handler(ctx, NewMsgDeposit(addrs[1], proposalID3, depositParams.MinDeposit)) + + // check deposits on proposal1 match individual deposits + deposits := getQueriedDeposits(t, ctx, cdc, querier, proposalID1) + require.Len(t, deposits, 1) + deposit := getQueriedDeposit(t, ctx, cdc, querier, proposalID1, addrs[0]) + require.Equal(t, deposit, deposits[0]) + + // check deposits on proposal2 match individual deposits + deposits = getQueriedDeposits(t, ctx, cdc, querier, proposalID2) + require.Len(t, deposits, 2) + deposit = getQueriedDeposit(t, ctx, cdc, querier, proposalID2, addrs[0]) + require.True(t, deposit.Equals(deposits[0])) + deposit = getQueriedDeposit(t, ctx, cdc, querier, proposalID2, addrs[1]) + require.True(t, deposit.Equals(deposits[1])) + + // check deposits on proposal3 match individual deposits + deposits = getQueriedDeposits(t, ctx, cdc, querier, proposalID3) + require.Len(t, deposits, 1) + deposit = getQueriedDeposit(t, ctx, cdc, querier, proposalID3, addrs[1]) + require.Equal(t, deposit, deposits[0]) + + // Only proposal #1 should be in Deposit Period + proposals := getQueriedProposals(t, ctx, cdc, querier, nil, nil, StatusDepositPeriod, 0) + require.Len(t, proposals, 1) + require.Equal(t, proposalID1, proposals[0].GetProposalID()) + // Only proposals #2 and #3 should be in Voting Period + proposals = getQueriedProposals(t, ctx, cdc, querier, nil, nil, StatusVotingPeriod, 0) + require.Len(t, proposals, 2) + require.Equal(t, proposalID2, proposals[0].GetProposalID()) + require.Equal(t, proposalID3, proposals[1].GetProposalID()) + + // Addrs[0] votes on proposals #2 & #3 + handler(ctx, NewMsgVote(addrs[0], proposalID2, OptionYes)) + handler(ctx, NewMsgVote(addrs[0], proposalID3, OptionYes)) + + // Addrs[1] votes on proposal #3 + handler(ctx, NewMsgVote(addrs[1], proposalID3, OptionYes)) + + // Test query voted by addrs[0] + proposals = getQueriedProposals(t, ctx, cdc, querier, nil, addrs[0], StatusNil, 0) + require.Equal(t, proposalID2, (proposals[0]).GetProposalID()) + require.Equal(t, proposalID3, (proposals[1]).GetProposalID()) + + // Test query votes on Proposal 2 + votes := getQueriedVotes(t, ctx, cdc, querier, proposalID2) + require.Len(t, votes, 1) + require.Equal(t, addrs[0], votes[0].Voter) + vote := getQueriedVote(t, ctx, cdc, querier, proposalID2, addrs[0]) + require.Equal(t, vote, votes[0]) + + // Test query votes on Proposal 3 + votes = getQueriedVotes(t, ctx, cdc, querier, proposalID3) + require.Len(t, votes, 2) + require.True(t, addrs[0].String() == votes[0].Voter.String()) + require.True(t, addrs[1].String() == votes[0].Voter.String()) + + // Test proposals queries with filters + + // Test query all proposals + proposals = getQueriedProposals(t, ctx, cdc, querier, nil, nil, StatusNil, 0) + require.Equal(t, proposalID1, (proposals[0]).GetProposalID()) + require.Equal(t, proposalID2, (proposals[1]).GetProposalID()) + require.Equal(t, proposalID3, (proposals[2]).GetProposalID()) + + // Test query voted by addrs[1] + proposals = getQueriedProposals(t, ctx, cdc, querier, nil, addrs[1], StatusNil, 0) + require.Equal(t, proposalID3, (proposals[0]).GetProposalID()) + + // Test query deposited by addrs[0] + proposals = getQueriedProposals(t, ctx, cdc, querier, addrs[0], nil, StatusNil, 0) + require.Equal(t, proposalID1, (proposals[0]).GetProposalID()) + + // Test query deposited by addr2 + proposals = getQueriedProposals(t, ctx, cdc, querier, addrs[1], nil, StatusNil, 0) + require.Equal(t, proposalID2, (proposals[0]).GetProposalID()) + require.Equal(t, proposalID3, (proposals[1]).GetProposalID()) + + // Test query voted AND deposited by addr1 + proposals = getQueriedProposals(t, ctx, cdc, querier, addrs[0], addrs[0], StatusNil, 0) + require.Equal(t, proposalID2, (proposals[0]).GetProposalID()) + + // Test Tally Query + tally := getQueriedTally(t, ctx, cdc, querier, proposalID2) + require.True(t, !tally.Equals(EmptyTallyResult())) +} diff --git a/x/gov/simulation/msgs.go b/x/gov/simulation/msgs.go index d075c86ae..b69d7c18c 100644 --- a/x/gov/simulation/msgs.go +++ b/x/gov/simulation/msgs.go @@ -96,7 +96,8 @@ func SimulateMsgSubmitProposal(k gov.Keeper) simulation.Operation { func simulateHandleMsgSubmitProposal(msg gov.MsgSubmitProposal, handler sdk.Handler, ctx sdk.Context, event func(string)) (action string, ok bool) { ctx, _ = ctx.CacheContext() - handler(ctx, msg) + result := handler(ctx, msg) + ok = result.IsOK() event(fmt.Sprintf("gov/MsgSubmitProposal/%v", ok)) action = fmt.Sprintf("TestMsgSubmitProposal: ok %v, msg %s", ok, msg.GetSignBytes()) return diff --git a/x/gov/test_common.go b/x/gov/test_common.go index 4e0ffd51a..a0bc80fc2 100644 --- a/x/gov/test_common.go +++ b/x/gov/test_common.go @@ -39,6 +39,7 @@ func getMockApp(t *testing.T, numGenAccs int) (*mock.App, Keeper, stake.Keeper, keeper := NewKeeper(mapp.Cdc, keyGov, pk, pk.Subspace("testgov"), ck, sk, DefaultCodespace) mapp.Router().AddRoute("gov", NewHandler(keeper)) + mapp.QueryRouter().AddRoute("gov", NewQuerier(keeper)) mapp.SetEndBlocker(getEndBlocker(keeper)) mapp.SetInitChainer(getInitChainer(mapp, keeper, sk)) diff --git a/x/mock/app.go b/x/mock/app.go index 627434a51..0b7b0ae16 100644 --- a/x/mock/app.go +++ b/x/mock/app.go @@ -1,9 +1,11 @@ package mock import ( + "bytes" "fmt" "math/rand" "os" + "sort" bam "github.com/cosmos/cosmos-sdk/baseapp" "github.com/cosmos/cosmos-sdk/codec" @@ -109,23 +111,64 @@ func (app *App) InitChainer(ctx sdk.Context, _ abci.RequestInitChain) abci.Respo return abci.ResponseInitChain{} } +// Type that combines an Address with the privKey and pubKey to that address +type AddrKeys struct { + Address sdk.AccAddress + PubKey crypto.PubKey + PrivKey crypto.PrivKey +} + +// implement `Interface` in sort package. +type AddrKeysSlice []AddrKeys + +func (b AddrKeysSlice) Len() int { + return len(b) +} + +// Sorts lexographically by Address +func (b AddrKeysSlice) Less(i, j int) bool { + // bytes package already implements Comparable for []byte. + switch bytes.Compare(b[i].Address.Bytes(), b[j].Address.Bytes()) { + case -1: + return true + case 0, 1: + return false + default: + panic("not fail-able with `bytes.Comparable` bounded [-1, 1].") + } +} + +func (b AddrKeysSlice) Swap(i, j int) { + b[j], b[i] = b[i], b[j] +} + // CreateGenAccounts generates genesis accounts loaded with coins, and returns // their addresses, pubkeys, and privkeys. func CreateGenAccounts(numAccs int, genCoins sdk.Coins) (genAccs []auth.Account, addrs []sdk.AccAddress, pubKeys []crypto.PubKey, privKeys []crypto.PrivKey) { + addrKeysSlice := AddrKeysSlice{} + for i := 0; i < numAccs; i++ { privKey := ed25519.GenPrivKey() pubKey := privKey.PubKey() addr := sdk.AccAddress(pubKey.Address()) - genAcc := &auth.BaseAccount{ + addrKeysSlice = append(addrKeysSlice, AddrKeys{ Address: addr, - Coins: genCoins, - } + PubKey: pubKey, + PrivKey: privKey, + }) + } - genAccs = append(genAccs, genAcc) - privKeys = append(privKeys, privKey) - pubKeys = append(pubKeys, pubKey) - addrs = append(addrs, addr) + sort.Sort(addrKeysSlice) + + for i := range addrKeysSlice { + addrs = append(addrs, addrKeysSlice[i].Address) + pubKeys = append(pubKeys, addrKeysSlice[i].PubKey) + privKeys = append(privKeys, addrKeysSlice[i].PrivKey) + genAccs = append(genAccs, &auth.BaseAccount{ + Address: addrKeysSlice[i].Address, + Coins: genCoins, + }) } return diff --git a/x/slashing/genesis.go b/x/slashing/genesis.go index 614bf41eb..1d4a44369 100644 --- a/x/slashing/genesis.go +++ b/x/slashing/genesis.go @@ -73,13 +73,13 @@ func ExportGenesis(ctx sdk.Context, keeper Keeper) (data GenesisState) { keeper.iterateValidatorSigningInfos(ctx, func(address sdk.ConsAddress, info ValidatorSigningInfo) (stop bool) { bechAddr := address.String() signingInfos[bechAddr] = info - array := []MissedBlock{} + localMissedBlocks := []MissedBlock{} keeper.iterateValidatorMissedBlockBitArray(ctx, address, func(index int64, missed bool) (stop bool) { - array = append(array, MissedBlock{index, missed}) + localMissedBlocks = append(localMissedBlocks, MissedBlock{index, missed}) return false }) - missedBlocks[bechAddr] = array + missedBlocks[bechAddr] = localMissedBlocks return false }) diff --git a/x/slashing/signing_info.go b/x/slashing/signing_info.go index 77c437174..291351742 100644 --- a/x/slashing/signing_info.go +++ b/x/slashing/signing_info.go @@ -84,10 +84,10 @@ func (k Keeper) setValidatorMissedBlockBitArray(ctx sdk.Context, address sdk.Con func (k Keeper) clearValidatorMissedBlockBitArray(ctx sdk.Context, address sdk.ConsAddress) { store := ctx.KVStore(k.storeKey) iter := sdk.KVStorePrefixIterator(store, GetValidatorMissedBlockBitArrayPrefixKey(address)) + defer iter.Close() for ; iter.Valid(); iter.Next() { store.Delete(iter.Key()) } - iter.Close() } // Construct a new `ValidatorSigningInfo` struct diff --git a/x/stake/client/cli/query.go b/x/stake/client/cli/query.go index 28d09df8e..24e449996 100644 --- a/x/stake/client/cli/query.go +++ b/x/stake/client/cli/query.go @@ -127,9 +127,7 @@ func GetCmdQueryValidatorUnbondingDelegations(queryRoute string, cdc *codec.Code } cliCtx := context.NewCLIContext().WithCodec(cdc) - params := stake.QueryValidatorParams{ - ValidatorAddr: valAddr, - } + params := stake.NewQueryValidatorParams(valAddr) bz, err := cdc.MarshalJSON(params) if err != nil { @@ -164,9 +162,7 @@ func GetCmdQueryValidatorRedelegations(queryRoute string, cdc *codec.Codec) *cob } cliCtx := context.NewCLIContext().WithCodec(cdc) - params := stake.QueryValidatorParams{ - ValidatorAddr: valAddr, - } + params := stake.NewQueryValidatorParams(valAddr) bz, err := cdc.MarshalJSON(params) if err != nil { @@ -290,6 +286,41 @@ func GetCmdQueryDelegations(storeName string, cdc *codec.Codec) *cobra.Command { return cmd } +// GetCmdQueryValidatorDelegations implements the command to query all the +// delegations to a specific validator. +func GetCmdQueryValidatorDelegations(queryRoute string, cdc *codec.Codec) *cobra.Command { + cmd := &cobra.Command{ + Use: "delegations-to [validator-addr]", + Short: "Query all delegations made to one validator", + Args: cobra.ExactArgs(1), + RunE: func(cmd *cobra.Command, args []string) error { + validatorAddr, err := sdk.ValAddressFromBech32(args[0]) + if err != nil { + return err + } + + params := stake.NewQueryValidatorParams(validatorAddr) + + bz, err := cdc.MarshalJSON(params) + if err != nil { + return err + } + + cliCtx := context.NewCLIContext().WithCodec(cdc) + + res, err := cliCtx.QueryWithData(fmt.Sprintf("custom/%s/validatorDelegations", queryRoute), bz) + if err != nil { + return err + } + + fmt.Println(string(res)) + return nil + }, + } + + return cmd +} + // GetCmdQueryUnbondingDelegation implements the command to query a single // unbonding-delegation record. func GetCmdQueryUnbondingDelegation(storeName string, cdc *codec.Codec) *cobra.Command { diff --git a/x/stake/client/cli/tx.go b/x/stake/client/cli/tx.go index 506499929..a42fcf6d0 100644 --- a/x/stake/client/cli/tx.go +++ b/x/stake/client/cli/tx.go @@ -211,26 +211,11 @@ func GetCmdDelegate(cdc *codec.Codec) *cobra.Command { return cmd } -// GetCmdRedelegate implements the redelegate validator command. +// GetCmdRedelegate the begin redelegation command. func GetCmdRedelegate(storeName string, cdc *codec.Codec) *cobra.Command { cmd := &cobra.Command{ Use: "redelegate", Short: "redelegate illiquid tokens from one validator to another", - } - - cmd.AddCommand( - client.PostCommands( - GetCmdBeginRedelegate(storeName, cdc), - )...) - - return cmd -} - -// GetCmdBeginRedelegate the begin redelegation command. -func GetCmdBeginRedelegate(storeName string, cdc *codec.Codec) *cobra.Command { - cmd := &cobra.Command{ - Use: "begin", - Short: "begin redelegation", RunE: func(cmd *cobra.Command, args []string) error { txBldr := authtxb.NewTxBuilderFromCLI().WithCodec(cdc) cliCtx := context.NewCLIContext(). diff --git a/x/stake/client/rest/query.go b/x/stake/client/rest/query.go index 935c0229f..8c669ee66 100644 --- a/x/stake/client/rest/query.go +++ b/x/stake/client/rest/query.go @@ -78,6 +78,12 @@ func registerQueryRoutes(cliCtx context.CLIContext, r *mux.Router, cdc *codec.Co validatorHandlerFn(cliCtx, cdc), ).Methods("GET") + // Get all delegations to a validator + r.HandleFunc( + "/stake/validators/{validatorAddr}/delegations", + validatorDelegationsHandlerFn(cliCtx, cdc), + ).Methods("GET") + // Get all unbonding delegations from a validator r.HandleFunc( "/stake/validators/{validatorAddr}/unbonding_delegations", @@ -227,6 +233,11 @@ func validatorHandlerFn(cliCtx context.CLIContext, cdc *codec.Codec) http.Handle return queryValidator(cliCtx, cdc, "custom/stake/validator") } +// HTTP request handler to query all unbonding delegations from a validator +func validatorDelegationsHandlerFn(cliCtx context.CLIContext, cdc *codec.Codec) http.HandlerFunc { + return queryValidator(cliCtx, cdc, "custom/stake/validatorDelegations") +} + // HTTP request handler to query all unbonding delegations from a validator func validatorUnbondingDelegationsHandlerFn(cliCtx context.CLIContext, cdc *codec.Codec) http.HandlerFunc { return queryValidator(cliCtx, cdc, "custom/stake/validatorUnbondingDelegations") diff --git a/x/stake/client/rest/utils.go b/x/stake/client/rest/utils.go index bb986c314..7f6edc193 100644 --- a/x/stake/client/rest/utils.go +++ b/x/stake/client/rest/utils.go @@ -62,10 +62,7 @@ func queryBonds(cliCtx context.CLIContext, cdc *codec.Codec, endpoint string) ht return } - params := stake.QueryBondsParams{ - DelegatorAddr: delegatorAddr, - ValidatorAddr: validatorAddr, - } + params := stake.NewQueryBondsParams(delegatorAddr, validatorAddr) bz, err := cdc.MarshalJSON(params) if err != nil { @@ -93,9 +90,7 @@ func queryDelegator(cliCtx context.CLIContext, cdc *codec.Codec, endpoint string return } - params := stake.QueryDelegatorParams{ - DelegatorAddr: delegatorAddr, - } + params := stake.NewQueryDelegatorParams(delegatorAddr) bz, err := cdc.MarshalJSON(params) if err != nil { @@ -123,9 +118,7 @@ func queryValidator(cliCtx context.CLIContext, cdc *codec.Codec, endpoint string return } - params := stake.QueryValidatorParams{ - ValidatorAddr: validatorAddr, - } + params := stake.NewQueryValidatorParams(validatorAddr) bz, err := cdc.MarshalJSON(params) if err != nil { diff --git a/x/stake/genesis.go b/x/stake/genesis.go index d44055ea8..7be8eff3a 100644 --- a/x/stake/genesis.go +++ b/x/stake/genesis.go @@ -29,17 +29,9 @@ func InitGenesis(ctx sdk.Context, keeper Keeper, data types.GenesisState) (res [ keeper.SetIntraTxCounter(ctx, data.IntraTxCounter) keeper.SetLastTotalPower(ctx, data.LastTotalPower) - // We only need to set this if we're starting from a list of validators, not a state export - setBondIntraTxCounter := true - for _, validator := range data.Validators { - if validator.BondIntraTxCounter != 0 { - setBondIntraTxCounter = false - } - } - for i, validator := range data.Validators { // set the intra-tx counter to the order the validators are presented, if necessary - if setBondIntraTxCounter { + if !data.Exported { validator.BondIntraTxCounter = int16(i) } keeper.SetValidator(ctx, validator) @@ -76,7 +68,15 @@ func InitGenesis(ctx sdk.Context, keeper Keeper, data types.GenesisState) (res [ keeper.InsertRedelegationQueue(ctx, red) } - res = keeper.ApplyAndReturnValidatorSetUpdates(ctx) + // don't need to run Tendermint updates if we exported + if data.Exported { + for _, lv := range data.LastValidatorPowers { + keeper.SetLastValidatorPower(ctx, lv.Address, lv.Power) + } + } else { + res = keeper.ApplyAndReturnValidatorSetUpdates(ctx) + } + return } @@ -100,16 +100,23 @@ func ExportGenesis(ctx sdk.Context, keeper Keeper) types.GenesisState { redelegations = append(redelegations, red) return false }) + var lastValidatorPowers []types.LastValidatorPower + keeper.IterateLastValidatorPowers(ctx, func(addr sdk.ValAddress, power sdk.Int) (stop bool) { + lastValidatorPowers = append(lastValidatorPowers, types.LastValidatorPower{addr, power}) + return false + }) return types.GenesisState{ Pool: pool, Params: params, IntraTxCounter: intraTxCounter, LastTotalPower: lastTotalPower, + LastValidatorPowers: lastValidatorPowers, Validators: validators, Bonds: bonds, UnbondingDelegations: unbondingDelegations, Redelegations: redelegations, + Exported: true, } } diff --git a/x/stake/keeper/delegation.go b/x/stake/keeper/delegation.go index d535419e1..32f63f5ed 100644 --- a/x/stake/keeper/delegation.go +++ b/x/stake/keeper/delegation.go @@ -37,6 +37,21 @@ func (k Keeper) GetAllDelegations(ctx sdk.Context) (delegations []types.Delegati return delegations } +// return all delegations to a specific validator. Useful for querier. +func (k Keeper) GetValidatorDelegations(ctx sdk.Context, valAddr sdk.ValAddress) (delegations []types.Delegation) { + store := ctx.KVStore(k.storeKey) + iterator := sdk.KVStorePrefixIterator(store, DelegationKey) + defer iterator.Close() + + for ; iterator.Valid(); iterator.Next() { + delegation := types.MustUnmarshalDelegation(k.cdc, iterator.Key(), iterator.Value()) + if delegation.GetValidatorAddr().Equals(valAddr) { + delegations = append(delegations, delegation) + } + } + return delegations +} + // return a given amount of all the delegations from a delegator func (k Keeper) GetDelegatorDelegations(ctx sdk.Context, delegator sdk.AccAddress, maxRetrieve uint16) (delegations []types.Delegation) { diff --git a/x/stake/keeper/delegation_test.go b/x/stake/keeper/delegation_test.go index 23dcbc1a6..3fa641fd2 100644 --- a/x/stake/keeper/delegation_test.go +++ b/x/stake/keeper/delegation_test.go @@ -105,6 +105,9 @@ func TestDelegation(t *testing.T) { resVal, err = keeper.GetDelegatorValidator(ctx, addrDels[1], addrVals[i]) require.Nil(t, err) require.Equal(t, addrVals[i], resVal.GetOperator()) + + resDels := keeper.GetValidatorDelegations(ctx, addrVals[i]) + require.Len(t, resDels, 2) } // delete a record diff --git a/x/stake/keeper/keeper.go b/x/stake/keeper/keeper.go index a74f86084..b6c973f98 100644 --- a/x/stake/keeper/keeper.go +++ b/x/stake/keeper/keeper.go @@ -112,6 +112,21 @@ func (k Keeper) SetLastValidatorPower(ctx sdk.Context, operator sdk.ValAddress, store.Set(GetLastValidatorPowerKey(operator), bz) } +// Iterate over last validator powers. +func (k Keeper) IterateLastValidatorPowers(ctx sdk.Context, handler func(operator sdk.ValAddress, power sdk.Int) (stop bool)) { + store := ctx.KVStore(k.storeKey) + iter := sdk.KVStorePrefixIterator(store, LastValidatorPowerKey) + defer iter.Close() + for ; iter.Valid(); iter.Next() { + addr := sdk.ValAddress(iter.Key()[len(LastValidatorPowerKey):]) + var power sdk.Int + k.cdc.MustUnmarshalBinaryLengthPrefixed(iter.Value(), &power) + if handler(addr, power) { + break + } + } +} + // Delete the last validator power. func (k Keeper) DeleteLastValidatorPower(ctx sdk.Context, operator sdk.ValAddress) { store := ctx.KVStore(k.storeKey) diff --git a/x/stake/querier/queryable.go b/x/stake/querier/queryable.go index bacc8d6ae..13ff97ef3 100644 --- a/x/stake/querier/queryable.go +++ b/x/stake/querier/queryable.go @@ -15,6 +15,7 @@ const ( QueryDelegatorDelegations = "delegatorDelegations" QueryDelegatorUnbondingDelegations = "delegatorUnbondingDelegations" QueryDelegatorRedelegations = "delegatorRedelegations" + QueryValidatorDelegations = "validatorDelegations" QueryValidatorUnbondingDelegations = "validatorUnbondingDelegations" QueryValidatorRedelegations = "validatorRedelegations" QueryDelegator = "delegator" @@ -34,6 +35,8 @@ func NewQuerier(k keep.Keeper, cdc *codec.Codec) sdk.Querier { return queryValidators(ctx, cdc, k) case QueryValidator: return queryValidator(ctx, cdc, req, k) + case QueryValidatorDelegations: + return queryValidatorDelegations(ctx, cdc, req, k) case QueryValidatorUnbondingDelegations: return queryValidatorUnbondingDelegations(ctx, cdc, req, k) case QueryValidatorRedelegations: @@ -73,6 +76,7 @@ type QueryDelegatorParams struct { // defines the params for the following queries: // - 'custom/stake/validator' +// - 'custom/stake/validatorDelegations' // - 'custom/stake/validatorUnbondingDelegations' // - 'custom/stake/validatorRedelegations' type QueryValidatorParams struct { @@ -88,6 +92,28 @@ type QueryBondsParams struct { ValidatorAddr sdk.ValAddress } +// creates a new QueryDelegatorParams +func NewQueryDelegatorParams(delegatorAddr sdk.AccAddress) QueryDelegatorParams { + return QueryDelegatorParams{ + DelegatorAddr: delegatorAddr, + } +} + +// creates a new QueryValidatorParams +func NewQueryValidatorParams(validatorAddr sdk.ValAddress) QueryValidatorParams { + return QueryValidatorParams{ + ValidatorAddr: validatorAddr, + } +} + +// creates a new QueryBondsParams +func NewQueryBondsParams(delegatorAddr sdk.AccAddress, validatorAddr sdk.ValAddress) QueryBondsParams { + return QueryBondsParams{ + DelegatorAddr: delegatorAddr, + ValidatorAddr: validatorAddr, + } +} + func queryValidators(ctx sdk.Context, cdc *codec.Codec, k keep.Keeper) (res []byte, err sdk.Error) { stakeParams := k.GetParams(ctx) validators := k.GetValidators(ctx, stakeParams.MaxValidators) @@ -119,6 +145,23 @@ func queryValidator(ctx sdk.Context, cdc *codec.Codec, req abci.RequestQuery, k return res, nil } +func queryValidatorDelegations(ctx sdk.Context, cdc *codec.Codec, req abci.RequestQuery, k keep.Keeper) (res []byte, err sdk.Error) { + var params QueryValidatorParams + + errRes := cdc.UnmarshalJSON(req.Data, ¶ms) + if errRes != nil { + return []byte{}, sdk.ErrUnknownAddress("") + } + + delegations := k.GetValidatorDelegations(ctx, params.ValidatorAddr) + + res, errRes = codec.MarshalJSONIndent(cdc, delegations) + if errRes != nil { + return nil, sdk.ErrInternal(sdk.AppendMsgToErr("could not marshal result to JSON", errRes.Error())) + } + return res, nil +} + func queryValidatorUnbondingDelegations(ctx sdk.Context, cdc *codec.Codec, req abci.RequestQuery, k keep.Keeper) (res []byte, err sdk.Error) { var params QueryValidatorParams diff --git a/x/stake/querier/queryable_test.go b/x/stake/querier/queryable_test.go index 925eae63f..eda5c6952 100644 --- a/x/stake/querier/queryable_test.go +++ b/x/stake/querier/queryable_test.go @@ -17,25 +17,6 @@ var ( pk1, pk2 = keep.PKs[0], keep.PKs[1] ) -func newTestDelegatorQuery(delegatorAddr sdk.AccAddress) QueryDelegatorParams { - return QueryDelegatorParams{ - DelegatorAddr: delegatorAddr, - } -} - -func newTestValidatorQuery(validatorAddr sdk.ValAddress) QueryValidatorParams { - return QueryValidatorParams{ - ValidatorAddr: validatorAddr, - } -} - -func newTestBondQuery(delegatorAddr sdk.AccAddress, validatorAddr sdk.ValAddress) QueryBondsParams { - return QueryBondsParams{ - DelegatorAddr: delegatorAddr, - ValidatorAddr: validatorAddr, - } -} - func TestNewQuerier(t *testing.T) { cdc := codec.New() ctx, _, keeper := keep.CreateTestInput(t, false, 1000) @@ -72,7 +53,7 @@ func TestNewQuerier(t *testing.T) { _, err = querier(ctx, []string{"parameters"}, query) require.Nil(t, err) - queryValParams := newTestValidatorQuery(addrVal1) + queryValParams := NewQueryValidatorParams(addrVal1) bz, errRes := cdc.MarshalJSON(queryValParams) require.Nil(t, errRes) @@ -82,13 +63,16 @@ func TestNewQuerier(t *testing.T) { _, err = querier(ctx, []string{"validator"}, query) require.Nil(t, err) + _, err = querier(ctx, []string{"validatorDelegations"}, query) + require.Nil(t, err) + _, err = querier(ctx, []string{"validatorUnbondingDelegations"}, query) require.Nil(t, err) _, err = querier(ctx, []string{"validatorRedelegations"}, query) require.Nil(t, err) - queryDelParams := newTestDelegatorQuery(addrAcc2) + queryDelParams := NewQueryDelegatorParams(addrAcc2) bz, errRes = cdc.MarshalJSON(queryDelParams) require.Nil(t, errRes) @@ -160,7 +144,7 @@ func TestQueryValidators(t *testing.T) { require.ElementsMatch(t, queriedValidators, validatorsResp) // Query each validator - queryParams := newTestValidatorQuery(addrVal1) + queryParams := NewQueryValidatorParams(addrVal1) bz, errRes := cdc.MarshalJSON(queryParams) require.Nil(t, errRes) @@ -189,13 +173,18 @@ func TestQueryDelegation(t *testing.T) { pool := keeper.GetPool(ctx) keeper.SetValidatorByPowerIndex(ctx, val1, pool) + val2 := types.NewValidator(addrVal2, pk2, types.Description{}) + keeper.SetValidator(ctx, val2) + pool = keeper.GetPool(ctx) + keeper.SetValidatorByPowerIndex(ctx, val2, pool) + keeper.Delegate(ctx, addrAcc2, sdk.NewCoin(types.DefaultBondDenom, sdk.NewInt(20)), val1, true) // apply TM updates keeper.ApplyAndReturnValidatorSetUpdates(ctx) // Query Delegator bonded validators - queryParams := newTestDelegatorQuery(addrAcc2) + queryParams := NewQueryDelegatorParams(addrAcc2) bz, errRes := cdc.MarshalJSON(queryParams) require.Nil(t, errRes) @@ -223,7 +212,7 @@ func TestQueryDelegation(t *testing.T) { require.NotNil(t, err) // Query bonded validator - queryBondParams := newTestBondQuery(addrAcc2, addrVal1) + queryBondParams := NewQueryBondsParams(addrAcc2, addrVal1) bz, errRes = cdc.MarshalJSON(queryBondParams) require.Nil(t, errRes) @@ -288,9 +277,32 @@ func TestQueryDelegation(t *testing.T) { _, err = queryDelegation(ctx, cdc, query, keeper) require.NotNil(t, err) + // Query validator delegations + + bz, errRes = cdc.MarshalJSON(NewQueryValidatorParams(addrVal1)) + require.Nil(t, errRes) + + query = abci.RequestQuery{ + Path: "custom/stake/validatorDelegations", + Data: bz, + } + + res, err = queryValidatorDelegations(ctx, cdc, query, keeper) + require.Nil(t, err) + + var delegationsRes []types.Delegation + errRes = cdc.UnmarshalJSON(res, &delegationsRes) + require.Nil(t, errRes) + + require.Equal(t, delegationsRes[0], delegation) + // Query unbonging delegation keeper.BeginUnbonding(ctx, addrAcc2, val1.OperatorAddr, sdk.NewDec(10)) + queryBondParams = NewQueryBondsParams(addrAcc2, addrVal1) + bz, errRes = cdc.MarshalJSON(queryBondParams) + require.Nil(t, errRes) + query = abci.RequestQuery{ Path: "/custom/stake/unbondingDelegation", Data: bz, @@ -356,7 +368,7 @@ func TestQueryRedelegations(t *testing.T) { require.True(t, found) // delegator redelegations - queryDelegatorParams := newTestDelegatorQuery(addrAcc2) + queryDelegatorParams := NewQueryDelegatorParams(addrAcc2) bz, errRes := cdc.MarshalJSON(queryDelegatorParams) require.Nil(t, errRes) @@ -375,7 +387,7 @@ func TestQueryRedelegations(t *testing.T) { require.Equal(t, redelegation, redsRes[0]) // validator redelegations - queryValidatorParams := newTestValidatorQuery(val1.GetOperator()) + queryValidatorParams := NewQueryValidatorParams(val1.GetOperator()) bz, errRes = cdc.MarshalJSON(queryValidatorParams) require.Nil(t, errRes) diff --git a/x/stake/stake.go b/x/stake/stake.go index 87087e59c..a922d9d72 100644 --- a/x/stake/stake.go +++ b/x/stake/stake.go @@ -84,7 +84,10 @@ var ( NewMsgBeginUnbonding = types.NewMsgBeginUnbonding NewMsgBeginRedelegate = types.NewMsgBeginRedelegate - NewQuerier = querier.NewQuerier + NewQuerier = querier.NewQuerier + NewQueryDelegatorParams = querier.NewQueryDelegatorParams + NewQueryValidatorParams = querier.NewQueryValidatorParams + NewQueryBondsParams = querier.NewQueryBondsParams ) const ( diff --git a/x/stake/types/genesis.go b/x/stake/types/genesis.go index f1673a376..1085e8f97 100644 --- a/x/stake/types/genesis.go +++ b/x/stake/types/genesis.go @@ -10,10 +10,18 @@ type GenesisState struct { Params Params `json:"params"` IntraTxCounter int16 `json:"intra_tx_counter"` LastTotalPower sdk.Int `json:"last_total_power"` + LastValidatorPowers []LastValidatorPower `json:"last_validator_powers"` Validators []Validator `json:"validators"` Bonds []Delegation `json:"bonds"` UnbondingDelegations []UnbondingDelegation `json:"unbonding_delegations"` Redelegations []Redelegation `json:"redelegations"` + Exported bool `json:"exported"` +} + +// Last validator power, needed for validator set update logic +type LastValidatorPower struct { + Address sdk.ValAddress + Power sdk.Int } func NewGenesisState(pool Pool, params Params, validators []Validator, bonds []Delegation) GenesisState { diff --git a/x/stake/types/params.go b/x/stake/types/params.go index dcd172982..4e9aba5ab 100644 --- a/x/stake/types/params.go +++ b/x/stake/types/params.go @@ -72,7 +72,7 @@ func (p Params) HumanReadableString() string { resp := "Params \n" resp += fmt.Sprintf("Unbonding Time: %s\n", p.UnbondingTime) - resp += fmt.Sprintf("Max Validators: %d: \n", p.MaxValidators) + resp += fmt.Sprintf("Max Validators: %d\n", p.MaxValidators) resp += fmt.Sprintf("Bonded Coin Denomination: %s\n", p.BondDenom) return resp }