diff --git a/.gitignore b/.gitignore index d93ff09e5a..5575d76998 100644 --- a/.gitignore +++ b/.gitignore @@ -5,7 +5,7 @@ release/ docs/tla/states/ *.out vendor/ -build/ +/build/ .vscode .idea __debug_* diff --git a/docs/build_deploy.sh b/docs/build_deploy.sh index a5cfb5676f..7f2567fb1c 100755 --- a/docs/build_deploy.sh +++ b/docs/build_deploy.sh @@ -5,10 +5,5 @@ echo "building docusaurus main docs" cp supported_versions.json versions.json npm ci && npm run build -# copy "legacy" docs directory into the final build directory -# the directory is in "docs/legacy" of the source branch (legacy-docs-page) -# the build environment must be in "./docs" for this to work as expected -git checkout origin/legacy-docs-page -- legacy -cp -r ./legacy ./build/ mv build ~/output echo "done building docusaurus main docs" diff --git a/docs/docs/build/_category_.json b/docs/docs/build/_category_.json new file mode 100644 index 0000000000..947929f840 --- /dev/null +++ b/docs/docs/build/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Build", + "position": 3 +} diff --git a/docs/docs/build/modules/01-overview.md b/docs/docs/build/modules/01-overview.md new file mode 100644 index 0000000000..4377496390 --- /dev/null +++ b/docs/docs/build/modules/01-overview.md @@ -0,0 +1,22 @@ +--- +sidebar_position: 1 +--- + +# Overview + +ICS consists of two main modules: + +* [x/provider](./02-provider.md) + * Provides to consumer chains updated information of opted in validators. + * Distributes ICS rewards to opted in validators. + * Jails and slashes validators that misbehave on consumer chains. +* [x/consumer](./03-consumer.md) + * Sends to the consensus engine the validator sets received from the provider chain. + * Splits consumer block rewards and sends ICS rewards to the provider chain. + * Notifies the provider chain of downtime infractions. + +Note that `x/types` contains types shared by both modules. + +In addition, the following modules are added to ICS to extend its functionality: + +* [x/democracy](04-democracy.md) \ No newline at end of file diff --git a/docs/docs/build/modules/02-provider.md b/docs/docs/build/modules/02-provider.md new file mode 100644 index 0000000000..2ddcf52419 --- /dev/null +++ b/docs/docs/build/modules/02-provider.md @@ -0,0 +1,3508 @@ +--- +sidebar_position: 2 +toc_min_heading_level: 2 +toc_max_heading_level: 5 +--- + +# x/ccv/provider + +## Overview + +The ICS provider module enables a proof-of-stake chain (known as the provider chain) +to share (parts of) its security with other chains (known as consumer chains). +This basically mean that consumer chains can run as proof-of-stake chains using +(parts of) the stake locked on the provider as collateral. + +The provider module has the following functionalities: + +- The permissionless creation of consumer chains. +- The customization of the consumer chains validator sets. +- The option for validators to opt in to validate the consumer chains they want. +- The distribution of rewards from consumer chains to the opted in validators. +- The slashing and jailing of validators commiting infractions on consumer chains based on cryptographic evidence. + +## State + +For clarity, the description of the the provider module state is split into features. +For a more accurate description, check out the `x/ccv/provider/types/keys.go` file, which contains the definitions of all the keys. + +### Consumer Lifecycle + +#### ConsumerId + +`ConsumerId` is the consumer ID of the next consumer chain to be created. + +Format: `byte(43) -> uint64` + +#### ConsumerIdToChainId + +`ConsumerIdToChainId` is the chain ID of a given consumer chain. + +Format: `byte(44) | len(consumerId) | []byte(consumerId) -> string` + +#### ConsumerIdToOwnerAddress + +`ConsumerIdToOwnerAddress` is the account address of the owner of a given consumer chain. + +Format: `byte(45) | len(consumerId) | []byte(consumerId) -> string` + +#### ConsumerIdToMetadataKey + +`ConsumerIdToMetadataKey` is the metadata of a given consumer chain. + +Format: `byte(46) | len(consumerId) | []byte(consumerId) -> ConsumerMetadata` + +#### ConsumerIdToPhase + +`ConsumerIdToPhase` is the phase of a given consumer chain. + +Format: `byte(49) | len(consumerId) | []byte(consumerId) -> ConsumerPhase`, where `ConsumerPhase` is defined as + +```proto +enum ConsumerPhase { + option (gogoproto.goproto_enum_prefix) = false; + + // UNSPECIFIED defines an empty phase. + CONSUMER_PHASE_UNSPECIFIED = 0; + // REGISTERED defines the phase in which a consumer chain has been assigned a unique consumer id. + // A chain in this phase cannot yet launch. + CONSUMER_PHASE_REGISTERED = 1; + // INITIALIZED defines the phase in which a consumer chain has set all the needed parameters to launch but + // has not yet launched (e.g., because the `spawnTime` of the consumer chain has not yet been reached). + CONSUMER_PHASE_INITIALIZED = 2; + // LAUNCHED defines the phase in which a consumer chain is running and consuming a subset of the validator + // set of the provider. + CONSUMER_PHASE_LAUNCHED = 3; + // STOPPED defines the phase in which a previously-launched chain has stopped. + CONSUMER_PHASE_STOPPED = 4; + // DELETED defines the phase in which the state of a stopped chain has been deleted. + CONSUMER_PHASE_DELETED = 5; +} +``` + +#### ConsumerIdToRemovalTime + +`ConsumerIdToRemovalTime` is the removal time of a given consumer chain in the stopped phase. + +Format: `byte(50) | len(consumerId) | []byte(consumerId) -> time.Time` + +#### SpawnTimeToConsumerIds + +`SpawnTimeToConsumerIds` are the IDs of initialized consumer chains ready to be launched at a timestamp `ts`. + +Format: `byte(51) | ts -> ConsumerIds`, where `ConsumerIds` is defined as + +```proto +message ConsumerIds { + repeated string ids = 1; +} +``` + +#### RemovalTimeToConsumerIds + +`RemovalTimeToConsumerIds` are the IDs of stopped consumer chains ready to be removed at a timestamp `ts`. + +Format: `byte(52) | ts -> ConsumerIds`, where `ConsumerIds` is defined as + +### Consumer Launch + +#### ConsumerIdToInitializationParameters + +`ConsumerIdToInitializationParameters` are the initialization parameters of a given consumer chain. + +Format: `byte(47) | len(consumerId) | []byte(consumerId) -> ConsumerInitializationParameters` + +#### ConsumerIdToChannelId + +`ConsumerIdToChannelId` is the ID of the CCV channel associated with a consumer chain. + +Format: `byte(5) | []byte(consumerId) -> string` + +#### ChannelIdToConsumerId + +`ChannelIdToConsumerId` is the consumer ID associated with a CCV channel. + +Format: `byte(6) | []byte(channelId) -> string` + +#### ConsumerIdToClientId + +`ConsumerIdToClientId` is the ID of the client associated with a consumer chain. +This is the underlying client of the corresponding CCV channel. + +Format: `byte(7) | []byte(consumerId) -> string` + +#### ClientIdToConsumerId + +`ClientIdToConsumerId` is the consumer ID associated with an IBC client (i.e., the underlying client of the corresponding CCV channel). + +Format: `byte(53) | len(clientId) | []byte(clientId) -> string` + +#### ConsumerGenesis + +`ConsumerGenesis` is the genesis state of the consumer module associated with a consumer chain. + +Format: `byte(14) | []byte(consumerId) -> ConsumerGenesisState` + + +### Key Assingment + +#### ConsumerValidators + +> TODO: `ConsumerValidators` and `ConsumerValidator` are too similar. + +`ConsumerValidators` is the public key assigned by a given validator with `addr` as its provider consensus address (i.e., `sdk.ConsAddress`) on a given consumer chain. + +Format: `byte(22) | len(consumerId) | []byte(consumerId) | addr -> crypto.PublicKey`, where `crypto` is `"github.com/cometbft/cometbft/proto/tendermint/crypto"`. + +#### ValidatorsByConsumerAddr + +`ValidatorsByConsumerAddr` is the consensus address on the provider chain of a validator with `addr` as its consensus address on a given consumer chain. + +Format: `byte(23) | len(consumerId) | []byte(consumerId) | addr -> sdk.ConsAddress`. + +#### ConsumerAddrsToPruneV2 + +`ConsumerAddrsToPruneV2` stores the list of consumer consensus addresses that can be prunned at a timestamp `ts` as they are no longer needed. + +Format: `byte(40) | len(consumerId) | []byte(consumerId) | ts -> AddressList`, where `AddressList` is defined as + +```proto +message AddressList { + repeated bytes addresses = 1; +} +``` + +### Power Shaping + +#### ConsumerIdToPowerShapingParameters + +`ConsumerIdToPowerShapingParameters` are the power-shaping parameters of a given consumer chain. + +Format: `byte(48) | len(consumerId) | []byte(consumerId) -> PowerShapingParameters` + +#### ConsumerValidator + +`ConsumerValidator` is the `ConsensusValidator` record of a provider validator on a given consumer chain, i.e., + +```proto +message ConsensusValidator { + // validator's consensus address on the provider chain + bytes provider_cons_addr = 1; + // voting power the validator has during this epoch + int64 power = 2; + // public key the validator uses on the consumer chain during this epoch + tendermint.crypto.PublicKey public_key = 3; + // height the validator had when it FIRST became a consumer validator + int64 join_height = 4; +} +``` + +Format: `byte(31) | len(consumerId) | []byte(consumerId) | addr -> ConsensusValidator`, with `addr` the validator's consensus address on the provider chain. + +#### OptedIn + +`OptedIn` is the list of provider validators that opted in to validate on a given consumer chain. +Note that opting in doesn't guarantee a spot in the consumer validator set. + +Format: `byte(32) | len(consumerId) | []byte(consumerId) | addr -> []byte{}`, with `addr` the validator's consensus address on the provider chain. + +#### Allowlist + +`Allowlist` is the list of provider validators that are eligible to validate a given consumer chain. + +Format: `byte(36) | len(consumerId) | []byte(consumerId) | addr -> []byte{}`, with `addr` the validator's consensus address on the provider chain. + +#### Denylist + +`Denylist` is the list of provider validators that are not eligible to validate a given consumer chain. +Note that validator can opt in regardless of whether they are eligible or not. + +Format: `byte(37) | len(consumerId) | []byte(consumerId) | addr -> []byte{}`, with `addr` the validator's consensus address on the provider chain. + +#### MinimumPowerInTopN + +`MinimumPowerInTopN` is the minimum voting power a provider validator must have to be required to validate a given TopN consumer chain. + +Format: `byte(40) | len(consumerId) | []byte(consumerId) -> uint64` + +### Validator Set Updates + +#### ValidatorSetUpdateId + +`ValidatorSetUpdateId` is an incrementing sequence number that is used as a unique identifier for validator set updates sent to the consumer chains. +The validator set update ID is incremented every epoch. + +Format: `byte(2) -> uint64` + +#### PendingVSCs + +`PendingVSCs` is the list of `VSCPackets` that are queued to be sent to a given consumer chain. + +Format: `byte(17) | []byte(consumerId) -> ValidatorSetChangePackets`, where `ValidatorSetChangePackets` is defined as + +```proto +message ValidatorSetChangePackets { + repeated ValidatorSetChangePacketData list = 1 + [ (gogoproto.nullable) = false ]; +} +``` + +#### LastProviderConsensusVals + +`LastProviderConsensusVals` is the last validator set sent to the consensus engine of the provider chain. + +Format: `byte(42) | addr -> ConsensusValidator`, with `addr` the validator's consensus address on the provider chain and `ConsensusValidator` defined as + +```proto +message ConsensusValidator { + // validator's consensus address on the provider chain + bytes provider_cons_addr = 1; + // voting power the validator has during this epoch + int64 power = 2; + // public key the validator uses on the consumer chain during this epoch + tendermint.crypto.PublicKey public_key = 3; + // height the validator had when it FIRST became a consumer validator + int64 join_height = 4; +} +``` + +### Reward Distribution + +#### ConsumerRewardDenoms + +`ConsumerRewardDenoms` is storing the list of whitelisted denoms that are accepted as ICS rewards. +Note that denoms that are not whitelisted can still be transfer to the `consumer_rewards_pool` account on the provider module, but they will not be distributed to validators and their delegators. + +Format: `byte(27) | []byte(denom) -> []byte{}` + +#### ConsumerRewardsAllocation + +`ConsumerRewardsAllocation` is the allocation of ICS rewards for a given consumer chain. +This is used to distribute ICS rewards only to the validators that are part of the consumer chain validator set. + +Format: `byte(38) | []byte(consumerId) -> ConsumerRewardsAllocation`, where `ConsumerRewardsAllocation` is defined as + +```proto +message ConsumerRewardsAllocation { + repeated cosmos.base.v1beta1.DecCoin rewards = 1 [ + (gogoproto.nullable) = false, + (amino.dont_omitempty) = true, + (gogoproto.castrepeated) = "github.com/cosmos/cosmos-sdk/types.DecCoins" + ]; +} +``` + +#### ConsumerCommissionRate + +`ConsumerCommissionRate` is the commission rate set by a provider validator for a given consumer chain. + +Format: `byte(39) | len(consumerId) | []byte(consumerId) | addr -> math.LegacyDec`, with `addr` the validator's consensus address on the provider chain and `math` is `"cosmossdk.io/math"`. + +### Consumer Infractions + +#### SlashMeter + +`SlashMeter` is the meter used for the throttling mechanism as the allowance of voting power that can be jailed over time. +It is decremented by the amount of voting power jailed whenever a validator is jailed for downtime, and periodically replenished as decided by on-chain params. +See [ADR 002](../../adrs/adr-002-throttle.md) for more details. + +Format: `byte(3) -> math.Int` + +#### SlashMeterReplenishTimeCandidate + +`SlashMeterReplenishTimeCandidate` is the next UTC time the `SlashMeter` could potentially be replenished. +Note that this value is the next time the `SlashMeter` will be replenished if and only if the `SlashMeter` is not full. +Otherwise this value will be updated in every future block until the slash meter becomes not full. + +Format: `byte(4) -> time.Time` + +#### ValsetUpdateBlockHeight + +`ValsetUpdateBlockHeight` is the block height associated with a validator set update ID `vscId`. +This is used for mapping infraction heights on consumer chains to heights on the provider chain via the validator set update IDs (together with [InitChainHeight](#initchainheight)). + +Format: `byte(13) | vscId -> uint64` + +#### InitChainHeight + +`InitChainHeight` is the block height on the provider when the CCV channel of a given consumer chain was established (i.e., the channel opening handshake was completed). +This is used for mapping infraction heights on consumer chains to heights on the provider chain (together with [ValsetUpdateBlockHeight](#valsetupdateblockheight)). + +Format: `byte(16) | []byte(consumerId) -> uint64` + +#### SlashAcks + +`SlashAcks` are addresses of validators for which `SlashPackets` for downtime infractions received from a given consumer chain were handled. +These addresses are sent together with the validator updates to the consumer chain as confirmation that the downtime infractions were dealt with. + +Format: `byte(15) | []byte(consumerId) -> SlashAcks`, where `SlashAcks` is defined as + +```proto +message SlashAcks { + repeated string addresses = 1; +} +``` + +#### EquivocationEvidenceMinHeight + +`EquivocationEvidenceMinHeight` is the minimum height of a valid evidence of equivocation on a given consumer chain. + +Format: `byte(29) | []byte(consumerId) -> uint64` + +## State Transitions + +### Consumer chain phases + +The following diagram describes the phases of a consumer chain from the perspective of the provider module: + +![Phases of a consumer chain](../../adrs/figures/adr19_phases_of_a_consumer_chain.png) + +## IBC Callbacks + +The consumer module is an IBC application that implements the [IBC module callback](https://ibc.cosmos.network/v8/ibc/apps/apps/#create-a-custom-ibc-application-module). + +### OnChanOpenInit + +`OnChanOpenInit` returns an error. `MsgChannelOpenInit` should be sent to the consumer. + +### OnChanOpenTry + +`OnChanOpenTry` validates the parameters of the _CCV channel_ -- an ordered IBC channel connected on the `provider` port +and with the counterparty port set to `consumer` -- and asserts that the counterparty version matches the expected version +(only verions `1` is supported). + +If the validation passes, the provider module verifies that the underlying client is the expected client of the consumer chain +(i.e., the client created during the consumer chain launch) and that no other CCV channel exists for this consumer chain. + +Finally, it sets the [ProviderFeePoolAddr](./03-consumer.md#providerfeepooladdrstr) as part of the metadata. + +### OnChanOpenAck + +`OnChanOpenAck` returns an error. `MsgChannelOpenAck` should be sent to the consumer. + +### OnChanOpenConfirm + +`OnChanOpenConfirm` first verifies that no other CCV channel exists for this consumer chain. Note that this is a sanity check. +Then, it sets the channel mapping in the state. + +### OnChanCloseInit + +`OnChanCloseInit` returns an error. `MsgChannelCloseInit` should be sent to the consumer. + +### OnChanCloseConfirm + +`OnChanCloseConfirm` is a no-op. + +### OnRecvPacket + +`OnRecvPacket` unmarshals the IBC packet data into a `SlashPacketData` struct (see below) and executes the handling logic. + +- Validate the fields in `SlashPacketData`: + - `validator` has a valid address and a non-zero power; + - `infraction` is either downtime or double-singing; + - the provider has in state a mapping from `valset_update_id` to a block height. +- If it is a double-signing infraction, then just log it and return. +- Verify that the consumer chain is launched and the validator is opted in. +- Update the meter used for jail throttling. +- Jail the validator on the provider chain. +- Store in state the ACK that the downtime infraction was handled. + This will be sent to the consumer with the next validator updates to enable it + to send other downtime infractions for this validator. + +```proto +message SlashPacketData { + tendermint.abci.Validator validator = 1 [ + (gogoproto.nullable) = false, + (gogoproto.moretags) = "yaml:\"validator\"" + ]; + // map to the infraction block height on the provider + uint64 valset_update_id = 2; + // tell if the slashing is for a downtime or a double-signing infraction + cosmos.staking.v1beta1.Infraction infraction = 3; +} +``` + +Note that IBC packets with `VSCMaturedPacketData` data are dropped. For more details, check out [ADR 018](../../adrs/adr-018-remove-vscmatured.md). + +### OnAcknowledgementPacket + +`OnAcknowledgementPacket` stops and eventually removes the consumer chain associated with the channel on which the `MsgAcknowledgement` message was received. + +### OnTimeoutPacket + +`OnTimeoutPacket` stops and eventually removes the consumer chain associated with the channel on which the `MsgTimeout` message was received. + +## Messages + +### MsgUpdateParams + +`MsgUpdateParams` updates the [provider module parameters](#parameters). +The params are updated through a governance proposal where the signer is the gov module account address. + +```proto +message MsgUpdateParams { + option (cosmos.msg.v1.signer) = "authority"; + + // authority is the address of the governance account. + string authority = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // params defines the x/provider parameters to update. + Params params = 2 [(gogoproto.nullable) = false]; +} +``` + +### ChangeRewardDenomsProposal + +`MsgChangeRewardDenoms` updates the list of whitelisted denoms accepted by the provider as ICS rewards. +The list of accepted denoms is updated through a governance proposal where the signer is the gov module account address. + +Note that this message replaces `ChangeRewardDenomsProposal`, which is deprecated. + +```proto +message MsgChangeRewardDenoms { + option (cosmos.msg.v1.signer) = "authority"; + + // the list of consumer reward denoms to add + repeated string denoms_to_add = 1; + // the list of consumer reward denoms to remove + repeated string denoms_to_remove = 2; + // signer address + string authority = 3 [(cosmos_proto.scalar) = "cosmos.AddressString"]; +} +``` + +### MsgCreateConsumer + +`MsgCreateConsumer` enables a user to create a consumer chain. + +Both the `chain_id` and `metadata` fields are mandatory. +Both the `initialization_parameters` and `power_shaping_parameters` fields are optional. +The parameters not provided are set to their zero value. + +The owner of the created consumer chain is the submitter of the message. +This message cannot be submitted as part of a governance proposal, i.e., the submitter cannot be the gov module account address. +As a result, if the `power_shaping_parameters` are provided, then `power_shaping_parameters.top_N` must be set to zero (i.e., opt-in consumer chain). + +To create a top-n consumer chain, the following steps are require: + +- Create a opt-in consumer chain (via `MsgCreateConsumer`). +- Change the ownership of the consuemr chain to the gov module account address (via `MsgUpdateConsumer`). +- Change `power_shaping_parameters.top_N` to a value in `[50, 100]` trough a governance proposal with a `MsgUpdateConsumer` message. + +If the `initialization_parameters` field is set and `initialization_parameters.spawn_time > 0`, then the consumer chain will be scheduled to launch at `spawn_time`. + +```proto +message MsgCreateConsumer { + option (cosmos.msg.v1.signer) = "submitter"; + + // Submitter address. If the message is successfully handled, the ownership of + // the consumer chain will given to this address. + string submitter = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // the chain id of the new consumer chain + string chain_id = 2; + + ConsumerMetadata metadata = 3 [ (gogoproto.nullable) = false ]; + + ConsumerInitializationParameters initialization_parameters = 4; + + PowerShapingParameters power_shaping_parameters = 5; +} +``` + +### MsgUpdateConsumer + +`MsgUpdateConsumer` enables the owner of a consumer chain to update its parameters (e.g., set a new owner). + +Note that only the `owner` (i.e., signer) and `consumer_id` fields are mandatory. +The others field are optional. Not providing one of them will leave the existing values unchanged. +Providing one of `metadata`, `initialization_parameters` or `power_shaping_parameters`, +will update all the containing fields. +If one of the containing fields is missing, it will be set to its zero value. +For example, updating the `initialization_parameters` without specifying the `spawn_time`, will set the `spawn_time` to zero. + +If the `initialization_parameters` field is set and `initialization_parameters.spawn_time > 0`, then the consumer chain will be scheduled to launch at `spawn_time`. +Updating the `spawn_time` from a positive value to zero will remove the consumer chain from the list of scheduled to launch chains. +If the consumer chain is already launched, updating the `initialization_parameters` is no longer possible. + +If the `power_shaping_parameters` field is set and `power_shaping_parameters.top_N` is possitive, then the owner needs to be the gov module account address. + +If the `new_owner_address` field is set to a value different than the gov module account address, then `top_N` needs to be zero. + +```proto +message MsgUpdateConsumer { + option (cosmos.msg.v1.signer) = "owner"; + + // the address of the owner of the consumer chain to be updated + string owner = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // the consumer id of the consumer chain to be updated + string consumer_id = 2; + + // the new owner of the consumer when updated + string new_owner_address = 3 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // the metadata of the consumer when updated + ConsumerMetadata metadata = 4; + + // initialization parameters can only be updated before a chain has launched + ConsumerInitializationParameters initialization_parameters = 5; + + // the power-shaping parameters of the consumer when updated + PowerShapingParameters power_shaping_parameters = 6; +} +``` + +### MsgRemoveConsumer + +`MsgRemoveConsumer` enables the owner of a _launched_ consumer chain to remove it from the provider chain. +The message will first stop the consumer chain, which means the provider will stop sending it validator updates over IBC. +Then, once the unbonding period elapses, the consumer chain is removed from the provider state. + +```proto +message MsgRemoveConsumer { + option (cosmos.msg.v1.signer) = "owner"; + + // the consumer id of the consumer chain to be stopped + string consumer_id = 1; + // the address of the owner of the consumer chain to be stopped + string owner = 2 [(cosmos_proto.scalar) = "cosmos.AddressString"]; +} +``` + +### MsgOptIn + +`MsgOptIn` enables a validator to opt in to validate a consumer chain. +Note that _validators can opt in to validate consumer chains that are not launched yet_. +The signer of the message needs to match the validator address on the provider. + +Note that opting in doesn't guarantee a spot in the consumer chain's validator set. +Use the `has-to-validate` query to check if the validator is part of the consumer chain's validator set. +For more details, check out the [validator guide to Partial Set Security](../../validators/partial-set-security-for-validators.md). + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +The `consumer_key` field is optional. +It enables the validator to set the consensus public key to use on the consumer chain. +The validator can assing (or re-assing) this key also later via [MsgAssignConsumerKey](#msgassignconsumerkey). + +:::warning +Validators are strongly recommended to assign a separate key for each consumer chain +and **not** reuse the provider key across consumer chains for security reasons. + +This is especially important since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +that allows multiple consumer chains to have the same chain ID. +A validator using the same consensus key to validate on two chains with the same chain ID might get slashed for double signing. +::: + +```proto +message MsgOptIn { + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + option (cosmos.msg.v1.signer) = "signer"; + + // [DEPRECATED] use `consumer_id` instead + string chain_id = 1 [deprecated = true]; + // the validator address on the provider + string provider_addr = 2 [ (gogoproto.moretags) = "yaml:\"address\"" ]; + // (optional) The consensus public key to use on the consumer in json string format corresponding to proto-any, + // for example `{"@type":"/cosmos.crypto.ed25519.PubKey","key":"Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is="}`. + // This field is optional and can remain empty (i.e., `consumer_key = ""`). A validator can always change the + // consumer public key at a later stage by issuing a `MsgAssignConsumerKey` message. + string consumer_key = 3; + // submitter address + string signer = 4 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + // the consumer id of the consumer chain to opt in to + string consumer_id = 5; +} +``` + +### MsgAssignConsumerKey + +`MsgAssignConsumerKey` enables a validator to assign the consensus public key to use on a consumer chain. +Without assigning a specific key, the validator will need to use the same key as on the provider chain. + +:::warning +Validators are strongly recommended to assign a separate key for each consumer chain +and **not** reuse the provider key across consumer chains for security reasons. + +This is especially important since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +that allows multiple consumer chains to have the same chain ID. +A validator using the same consensus key to validate on two chains with the same chain ID might get slashed for double signing. +::: + +The signer of the message needs to match the validator address on the provider. + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +For more details, check out the [description of the Key Assignment feature](../../features/key-assignment.md). + +```proto +message MsgAssignConsumerKey { + option (cosmos.msg.v1.signer) = "signer"; + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + + + // [DEPRECATED] use `consumer_id` instead + string chain_id = 1 [deprecated = true]; + // The validator address on the provider + string provider_addr = 2 [ (gogoproto.moretags) = "yaml:\"address\"" ]; + // The consensus public key to use on the consumer. + // in json string format corresponding to proto-any, ex: + // `{"@type":"/cosmos.crypto.ed25519.PubKey","key":"Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is="}` + string consumer_key = 3; + + string signer = 4 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // the consumer id of the consumer chain to assign a consensus public key to + string consumer_id = 5; +} +``` +### MsgOptOut + +`MsgOptOut` enables a validator to opt out from validating a launched consumer chain. +The signer of the message needs to match the validator address on the provider. + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +For more details on optin out, check out the [validator guide to Partial Set Security](../../validators/partial-set-security-for-validators.md). + +```proto +message MsgOptOut { + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + option (cosmos.msg.v1.signer) = "signer"; + + // [DEPRECATED] use `consumer_id` instead + string chain_id = 1 [deprecated = true]; + // the validator address on the provider + string provider_addr = 2 [ (gogoproto.moretags) = "yaml:\"address\"" ]; + // submitter address + string signer = 3 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + // the consumer id of the consumer chain to opt out from + string consumer_id = 4; +} +``` + +### MsgSetConsumerCommissionRate + +`MsgSetConsumerCommissionRate` enables validators to set a per-consumer chain commission rate. +The `rate` is a decimal in `[minRate, 1]`, with `minRate` corresponding to the minimum commission rate set on the +provider chain (see `min_commission_rate` in `interchain-security-pd query staking params`). + +The signer of the message needs to match the validator address on the provider. + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +For more details on setting per-consumer chain commission rates, check out the [validator guide to Partial Set Security](../../validators/partial-set-security-for-validators.md). + +```proto +message MsgSetConsumerCommissionRate { + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + option (cosmos.msg.v1.signer) = "signer"; + + // The validator address on the provider + string provider_addr = 1 [ (gogoproto.moretags) = "yaml:\"address\"" ]; + // [DEPRECATED] use `consumer_id` instead + string chain_id = 2 [deprecated = true]; + // The rate to charge delegators on the consumer chain, as a fraction + string rate = 3 [ + (cosmos_proto.scalar) = "cosmos.Dec", + (gogoproto.customtype) = "cosmossdk.io/math.LegacyDec", + (gogoproto.nullable) = false + ]; + // submitter address + string signer = 4 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + // the consumer id of the consumer chain to set the commission rate + string consumer_id = 5; +} +``` + +### MsgSubmitConsumerMisbehaviour + +`MsgSubmitConsumerMisbehaviour` enables users to submit to the provider evidence of a light client attack that occured on a consumer chain. +This message can be submitted directly by users, e.g., via the CLI command `tx provider submit-consumer-misbehaviour`, +or by a relayer that can be set to automatically detect consumer chain misbehaviors, e.g., [Hermes](https://github.com/informalsystems/hermes). + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +For more details on reporting light client attacks that occured on consumer chains, check out the [guide on equivocation infractions](../../features/slashing.md#equivocation-infractions). + +```proto +message MsgSubmitConsumerMisbehaviour { + option (cosmos.msg.v1.signer) = "submitter"; + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + + string submitter = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + // The Misbehaviour of the consumer chain wrapping + // two conflicting IBC headers + ibc.lightclients.tendermint.v1.Misbehaviour misbehaviour = 2; + // the consumer id of the consumer chain where the misbehaviour occurred + string consumer_id = 3; +} +``` + +### MsgSubmitConsumerDoubleVoting + +`MsgSubmitConsumerDoubleVoting` enables users to submit to the provider evidence of a double signing infraction that occured on a consumer chain. +This message can be submitted directly by users, e.g., via the CLI command `tx provider submit-consumer-double-voting`, +or by a relayer that can be set to automatically detect consumer chain misbehaviors, e.g., [Hermes](https://github.com/informalsystems/hermes). + +Note that since the introduction of the +[Permissionless ICS feature](https://cosmos.github.io/interchain-security/adrs/adr-019-permissionless-ics) +the `chain_id` field is deprecated. +Users should use `consumer_id` instead. +You can use the `list-consumer-chains` query to get the list of all consumer chains and their consumer IDs. + +For more details on reporting double signing infractions that occured on consumer chains, check out the [guide on equivocation infractions](../../features/slashing.md#equivocation-infractions). + +```proto +message MsgSubmitConsumerDoubleVoting { + option (cosmos.msg.v1.signer) = "submitter"; + option (gogoproto.equal) = false; + option (gogoproto.goproto_getters) = false; + + string submitter = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + // The equivocation of the consumer chain wrapping + // an evidence of a validator that signed two conflicting votes + tendermint.types.DuplicateVoteEvidence duplicate_vote_evidence = 2; + // The light client header of the infraction block + ibc.lightclients.tendermint.v1.Header infraction_block_header = 3; + // the consumer id of the consumer chain where the double-voting took place + string consumer_id = 4; +} +``` + +## BeginBlock + +In the `BeginBlock` of the provider module the following actions are performed: + +- Launch every consumer chain that has a spawn time that already passed. + - Compute the initial validator set. + - Create the genesis state for the consumer module. + Note that the genesis state contains the [consumer module parameters](./03-consumer.md#parameters) and + both the client state and consensus state needed for creating a provider client on the consumer chain. + - Create a consumer client. +- Remove every stopped consumer chain for which the removal time has passed. +- Replenish the throttling meter if necessary. +- Distribute ICS rewards to the opted in validators. + +Note that for every consumer chain, the computation of its initial validator set is based on the consumer's [power shaping parameters](../../features/power-shaping.md) +and the [validators that opted in on that consumer](../../features/partial-set-security.md). + +## EndBlock + +In the `EndBlock` of the provider module the following actions are performed: + +- Store in state the VSC id to block height mapping needed for determining the height of infractions on consumer chains. +- Prune the no-longer needed public keys assigned by validators to use when validating on consumer chains. +- Send validator updates to the consensus engine. + The maximum number of validators is set through the [MaxProviderConsensusValidators](#maxproviderconsensusvalidators) param. +- At the begining of every epoch, + - for every launched consumer chain, compute the next consumer validator set and send it to the consumer chain via an IBC packet; + - increment the VSC id. + +Note that for every consumer chain, the computation of its validator set is based on the consumer's [power shaping parameters](../../features/power-shaping.md) +and the [validators that opted in on that consumer](../../features/partial-set-security.md). + +## Hooks + +> TBA + +## Events + +> TBA + +## Parameters + +The provider module contains the following parameters. + +### `TemplateClient` + +`TemplateClient` is a template of an IBC `ClientState` used for launching consumer chains. + +### TrustingPeriodFraction + +| Type | Default value | +| ------ | ------------- | +| string | "0.66" | + +`TrustingPeriodFraction` is used to used to compute the trusting period of IBC clients +(for both provider and consumer chains) as `UnbondingPeriod / TrustingPeriodFraction`. +Note that a light clients must be updated within the trusting period in order to avoid being frozen. + +The param is set as a string, and converted to a `sdk.Dec` when used. + +### CcvTimeoutPeriod + +| Type | Default value | +| ------------- | ------------------ | +| time.Duration | 2419200s (4 weeks) | + +`CcvTimeoutPeriod` is the period used to compute the timeout timestamp when sending IBC packets. +For more details, see the [IBC specification of Channel & Packet Semantics](https://github.com/cosmos/ibc/blob/main/spec/core/ics-004-channel-and-packet-semantics/README.md#sending-packets). + +:::warning +If a sent packet is not relayed within this period, then the packet times out. The CCV channel used by the interchain security protocol is closed, and the corresponding consumer is removed. +::: + +`CcvTimeoutPeriod` may have different values on the provider and consumer chains. +`CcvTimeoutPeriod` on the provider **must** be larger than consumer unbonding period. + +### SlashMeterReplenishPeriod + +| Type | Default value | +| ------------- | -------------- | +| time.Duration | 3600s (1 hour) | + +`SlashMeterReplenishPeriod` is the time interval at which the meter for [jail throttling](https://cosmos.github.io/interchain-security/adrs/adr-008-throttle-retries) is replenished. +The meter is replenished to an amount equal to the allowance for that block, or `SlashMeterReplenishFraction * CurrentTotalVotingPower`. + +### SlashMeterReplenishFraction + +| Type | Default value | +| ------ | ------------- | +| string | "0.05" | + +`SlashMeterReplenishFraction` is the fraction (in range `[0, 1]`) of total voting power that is replenished to the slash meter when a replenishment occurs. +This param also serves as a maximum fraction of total voting power that the slash meter can hold. + +The param is set as a string, and converted to a `sdk.Dec` when used. + +### ConsumerRewardDenomRegistrationFee + +`ConsumerRewardDenomRegistrationFee` is deprecated. + +### BlocksPerEpoch + +| Type | Default value | +| ----- | ------------- | +| int64 | 600 | + +`BlocksPerEpoch` is the number of blocks in an ICS epoch. +The provider sends validator updates to the consumer chains only once per epoch. + +:::warning +It is recommended for the length of an ICS epoch to not exceed a day. +Large epochs would lead to delays in validator updates sent to the consumer chains, +which might impact the security of the consumer chains. +::: + +### NumberOfEpochsToStartReceivingRewards + +| Type | Default value | +| ----- | ------------- | +| int64 | 24 | + +`NumberOfEpochsToStartReceivingRewards` is the number of ICS epochs that a validator +needs to wait after opting in on a consumer chain before being eligible to ICS reawards +from that consumer. + +### MaxProviderConsensusValidators + +| Type | Default value | +| ----- | ------------- | +| int64 | 180 | + +`MaxProviderConsensusValidators` is the maximum number of validators sent to +the provider consensus enginer. +This was introduced with the [Inactive Provider Validators feature](https://cosmos.github.io/interchain-security/adrs/adr-017-allowing-inactive-validators) +and it replaces the `MaxValidators` staking module parameter. +As a result, the provider chain can differentiate between +_bonded validators_, i.e., validators that have stake locked on the provider chain, +and _active validator_, i.e., validators that participate actively in the provider chain's consensus. + +## Client + +### CLI + +A user can interact with the `provider` module using the CLI. + +#### Query + +The `query` commands allow users to query `provider` state. + +```bash +interchain-security-pd query provider --help +``` + +##### Consumer Genesis + +The `consumer-genesis` command allows to query for consumer chain genesis state by consumer id. + +```bash +interchain-security-pd query provider consumer-genesis [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider consumer-genesis 0 +``` + +Output: + +```bash +new_chain: true +params: + blocks_per_distribution_transmission: "1000" + ccv_timeout_period: 2419200s + consumer_id: "0" + consumer_redistribution_fraction: "0.75" + distribution_transmission_channel: "" + enabled: true + historical_entries: "10000" + provider_fee_pool_addr_str: "" + provider_reward_denoms: [] + retry_delay_period: 3600s + reward_denoms: [] + soft_opt_out_threshold: "0" + transfer_timeout_period: 3600s + unbonding_period: 1209600s +provider: + client_state: + allow_update_after_expiry: false + allow_update_after_misbehaviour: false + chain_id: provider + frozen_height: + revision_height: "0" + revision_number: "0" + latest_height: + revision_height: "25" + revision_number: "0" + max_clock_drift: 10s + proof_specs: + - inner_spec: + child_order: + - 0 + - 1 + child_size: 33 + empty_child: null + hash: SHA256 + max_prefix_length: 12 + min_prefix_length: 4 + leaf_spec: + hash: SHA256 + length: VAR_PROTO + prefix: AA== + prehash_key: NO_HASH + prehash_value: SHA256 + max_depth: 0 + min_depth: 0 + prehash_key_before_comparison: false + - inner_spec: + child_order: + - 0 + - 1 + child_size: 32 + empty_child: null + hash: SHA256 + max_prefix_length: 1 + min_prefix_length: 1 + leaf_spec: + hash: SHA256 + length: VAR_PROTO + prefix: AA== + prehash_key: NO_HASH + prehash_value: SHA256 + max_depth: 0 + min_depth: 0 + prehash_key_before_comparison: false + trust_level: + denominator: "3" + numerator: "1" + trusting_period: 1197504s + unbonding_period: 1814400s + upgrade_path: + - upgrade + - upgradedIBCState + consensus_state: + next_validators_hash: 632730A03DEF630F77B61DF4092629007AE020B789713158FABCB104962FA54F + root: + hash: Jcck4b/HHJOcjcVjTdMi8qHB4SeCpWpfLiN9DtB99oA= + timestamp: "2024-09-25T09:18:40.262655625Z" + initial_val_set: + - power: "500" + pub_key: + ed25519: RrclQz9bIhkIy/gfL485g3PYMeiIku4qeo495787X10= + - power: "500" + pub_key: + ed25519: Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is= + - power: "500" + pub_key: + ed25519: mAN6RXYxSM4MNGSIriYiS7pHuwAcOHDQAy9/wnlSzOI= +``` + +
+ +##### List Consumer Chains + +The `list-consumer-chains` command allows to query consumer chains supported by the provider chain. +An optional integer parameter can be passed for phase filtering of consumer chains, (Registered=1|Initialized=2|Launched=3|Stopped=4|Deleted=5).` + +```bash +interchain-security-pd query provider list-consumer-chains [phase] [limit] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider list-consumer-chains 3 +``` + +Output: + +```bash +chains: +- allow_inactive_vals: true + allowlist: [] + chain_id: pion-1 + client_id: 07-tendermint-0 + consumer_id: "0" + denylist: ["cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6"] + metadata: + description: description of your chain and all other relevant information + metadata: some metadata about your chain + name: pion-1 + min_power_in_top_N: "500" + min_stake: "0" + phase: CONSUMER_PHASE_LAUNCHED + top_N: 60 + validator_set_cap: 0 + validators_power_cap: 0 +pagination: + next_key: null + total: "1" +``` + +
+ +##### Validator Consumer Key Assignment + +The `validator-consumer-key` command allows to query assigned validator consensus public key for a consumer chain. + +```bash +interchain-security-pd query provider validator-consumer-key [consumer-id] [provider-validator-address] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider validator-consumer-key 0 cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6 +``` + +Output: + + ```bash +consumer_address: cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk + ``` + +
+ +##### Validator Provider Key + +The `validator-provider-key` command allows to query validator consensus public key for the provider chain. + + +```bash +interchain-security-pd query provider validator-provider-key [consumer-id] [consumer-validator-address] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider validator-provider-key 0 cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk +``` + +Output: + + ```bash +provider_address: cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj +``` + +
+ +##### Throttle State + +The `throttle-state` command allows to query on-chain state relevant to slash packet throttling. + + +```bash +interchain-security-pd query provider throttle-state [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider throttle-state +``` + +Output: + +```bash +next_replenish_candidate: "2024-09-26T07:59:51.336971970Z" +slash_meter: "1500" +slash_meter_allowance: "1511" +``` + +
+ +##### Registered Consumer Reward Denoms + +The `registered-consumer-reward-denoms` command allows to query registered consumer reward denoms. + +```bash +interchain-security-pd query provider registered-consumer-reward-denoms [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider registered-consumer-reward-denoms +``` + +Output: + +```bash +denoms: +- ibc/3C3D7B3BE4ECC85A0E5B52A3AEC3B7DFC2AA9CA47C37821E57020D6807043BE9 +- ibc/D549749C93524DA1831A4B3C850DFC1BA9060261BEDFB224B3B0B4744CD77A70 +``` + +
+ +##### All Pairs Valconsensus Address + +The `all-pairs-valconsensus-address` command allows to query all pairs of valconsensus address by consumer id. + +```bash +interchain-security-pd query provider all-pairs-valconsensus-address [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider all-pairs-valconsensus-address 0 +``` + +Output: + +```bash +pair_val_con_addr: +- consumer_address: cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk + consumer_key: + ed25519: Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is= + provider_address: cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6 +``` + +
+ +##### Provider Parameters + +The `params` command allows to query provider parameters information. + +```bash +interchain-security-pd query provider params [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider params +``` + +Output: + +```bash +blocks_per_epoch: "3" +ccv_timeout_period: 2419200s +consumer_reward_denom_registration_fee: + amount: "10000000" + denom: stake +max_provider_consensus_validators: "180" +number_of_epochs_to_start_receiving_rewards: "24" +slash_meter_replenish_fraction: "1.0" +slash_meter_replenish_period: 3600s +template_client: + allow_update_after_expiry: false + allow_update_after_misbehaviour: false + chain_id: "" + frozen_height: + revision_height: "0" + revision_number: "0" + latest_height: + revision_height: "0" + revision_number: "0" + max_clock_drift: 10s + proof_specs: + - inner_spec: + child_order: + - 0 + - 1 + child_size: 33 + empty_child: null + hash: SHA256 + max_prefix_length: 12 + min_prefix_length: 4 + leaf_spec: + hash: SHA256 + length: VAR_PROTO + prefix: AA== + prehash_key: NO_HASH + prehash_value: SHA256 + max_depth: 0 + min_depth: 0 + prehash_key_before_comparison: false + - inner_spec: + child_order: + - 0 + - 1 + child_size: 32 + empty_child: null + hash: SHA256 + max_prefix_length: 1 + min_prefix_length: 1 + leaf_spec: + hash: SHA256 + length: VAR_PROTO + prefix: AA== + prehash_key: NO_HASH + prehash_value: SHA256 + max_depth: 0 + min_depth: 0 + prehash_key_before_comparison: false + trust_level: + denominator: "3" + numerator: "1" + trusting_period: 0s + unbonding_period: 0s + upgrade_path: + - upgrade + - upgradedIBCState +trusting_period_fraction: "0.66" +``` + +
+ +##### Consumer Opted In Validators + +The `consumer-opted-in-validators` command allows to query opted-in validators for a given consumer chain. + +```bash +interchain-security-pd query provider consumer-opted-in-validators [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider consumer-opted-in-validators 0 +``` + +Output: + +```bash +validators_provider_addresses: +- cosmosvalcons1qmq08eruchr5sf5s3rwz7djpr5a25f7xw4mceq +- cosmosvalcons1nx7n5uh0ztxsynn4sje6eyq2ud6rc6klc96w39 +- cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6 +``` + +
+ +##### Consumer Validators + +The `consumer-validators` command allows to query the last set consumer-validator set for a given consumer chain. + +```bash +interchain-security-pd query provider consumer-validators [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider consumer-validators 0 +``` + +Output: + +```bash +validators: +- consumer_commission_rate: "0.100000000000000000" + consumer_key: + ed25519: RrclQz9bIhkIy/gfL485g3PYMeiIku4qeo495787X10= + consumer_power: "511" + description: + details: "" + identity: "" + moniker: validatoralice + security_contact: "" + website: "" + jailed: false + power: "0" + provider_address: cosmosvalcons1qmq08eruchr5sf5s3rwz7djpr5a25f7xw4mceq + provider_commission_rate: "0.100000000000000000" + provider_operator_address: cosmosvaloper19pe9pg5dv9k5fzgzmsrgnw9rl9asf7ddtrgtng + provider_power: "511" + provider_tokens: "511000000" + rate: "0.000000000000000000" + status: BOND_STATUS_BONDED + validates_current_epoch: true +- consumer_commission_rate: "0.100000000000000000" + consumer_key: + ed25519: mAN6RXYxSM4MNGSIriYiS7pHuwAcOHDQAy9/wnlSzOI= + consumer_power: "500" + description: + details: "" + identity: "" + moniker: validatorbob + security_contact: "" + website: "" + jailed: false + power: "0" + provider_address: cosmosvalcons1nx7n5uh0ztxsynn4sje6eyq2ud6rc6klc96w39 + provider_commission_rate: "0.100000000000000000" + provider_operator_address: cosmosvaloper1dkas8mu4kyhl5jrh4nzvm65qz588hy9qakmjnw + provider_power: "500" + provider_tokens: "500000000" + rate: "0.000000000000000000" + status: BOND_STATUS_BONDED + validates_current_epoch: true +``` + +
+ +##### Has to Validate + +The `has-to-validate` command allows to query the consumer chains list a given validator has to validate. + +```bash +interchain-security-pd query provider has-to-validate [provider-validator-address] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider has-to-validate cosmoscons1gghjut3ccd8ay0zduzj64hwre2fxs9ldmqhffj +``` + +Output: + +```bash +consumer_ids: +- "0" +- "2" +``` + +
+ +##### Validator Consumer Commission Rate + +The `validator-consumer-commission-rate` command allows to query the consumer commission rate a validator charges on a consumer chain. + +```bash +interchain-security-pd query provider validator-consumer-commission-rate [consumer-id] [provider-validator-address] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider validator-consumer-commission-rate 0 cosmoscons1gghjut3ccd8ay0zduzj64hwre2fxs9ldmqhffj +``` + +Output: + +```bash +rate: "0.750000000000000000" +``` + +
+ +##### Blocks Until Next Epoch + +The `blocks-until-next-epoch` command allows to query the number of blocks until the next epoch begins and validator updates are sent to consumer chains + +```bash +interchain-security-pd query provider blocks-until-next-epoch [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider blocks-until-next-epoch +``` + +Output: + +```bash +blocks_until_next_epoch: "286" +``` + +
+ +##### Consumer Id From Client Id + +The `consumer-id-from-client-id` command allows to query the consumer id of the chain associated with the provided client id. + +```bash +interchain-security-pd query provider consumer-id-from-client-id [client-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider consumer-id-from-client-id 07-tendermint-0 +``` + +Output: + +```bash +consumer_id: "0" +``` + +
+ +##### Consumer Chain + +The `consumer-chain` command allows to query the consumer chain associated with the consumer id. + +```bash +interchain-security-pd query provider consumer-chain [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd query provider consumer-chain 0 +``` + +Output: + +```bash +chain_id: pion-1 +consumer_id: "0" +init_params: + binary_hash: YmluX2hhc2g= + blocks_per_distribution_transmission: "1000" + ccv_timeout_period: 2419200s + consumer_redistribution_fraction: "0.75" + distribution_transmission_channel: "" + genesis_hash: Z2VuX2hhc2g= + historical_entries: "10000" + initial_height: + revision_height: "1" + revision_number: "0" + spawn_time: "2024-09-26T06:55:14.616054Z" + transfer_timeout_period: 3600s + unbonding_period: 1209600s +metadata: + description: description of your chain and all other relevant information + metadata: some metadata about your chain + name: pion-1 +owner_address: cosmos10d07y265gmmuvt4z0w9aw880jnsr700j6zn9kn +phase: CONSUMER_PHASE_LAUNCHED +power_shaping_params: + allow_inactive_vals: false + allowlist: [] + denylist: [] + min_stake: "0" + top_N: 100 + validator_set_cap: 0 + validators_power_cap: 0 +``` + +
+ +#### Transactions + +The `tx` commands allows users to interact with the `provider` module. + +```bash +interchain-security-pd tx provider --help +``` + +##### Assign Consumer Key + +The `assign-consensus-key` command allows to assign a consensus public key to use for a consumer chain. + +```bash +interchain-security-pd tx provider assign-consensus-key [consumer-id] [consumer-pubkey] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider assign-consensus-key 0 \ + '{"@type":"/cosmos.crypto.ed25519.PubKey","key":"Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is="}' \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +
+ +Note that the consumer pubkey can be obtained by using `interchain-security-cd tendermint show-validator` command. + +##### Create Consumer + +The `create-consumer` command allows to create a consumer chain. + +```bash +interchain-security-pd tx provider create-consumer [consumer-parameters] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider create-consumer path/to/create-consumer-msg.json \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +where `create-consumer-msg.json` contains: + +```json +{ + "chain_id" : "pion-1", + "metadata": { + "name": "pion-1", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your chain" + } +} +``` + +
+ +##### Update Consumer + +The `update-consumer` command allows to update a consumer chain. + +```bash +interchain-security-pd tx provider update-consumer [consumer-parameters] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider update-consumer path/to/update-consumer.json \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +where `update-consumer-msg.json` contains: + +```json +{ + "consumer_id" : "0", + "owner_address": "cosmos1p3ucd3ptpw902fluyjzhq3ffgq4ntddac9sa3s", + "metadata": { + "name": "pion-1", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your" + }, + "initialization_parameters":{ + "initial_height":{ + "revision_number": 0, + "revision_height": 1 + }, + "genesis_hash":"2D5C2110941DA54BE07CBB9FACD7E4A2E3253E79BE7BE3E5A1A7BDA518BAA4BE", + "binary_hash": "6EF05C2F38BE62A833E5AB51EBF3BA72D1BC1664D7E4A2E3253DA54BE07CF38A", + "spawn_time": "2024-09-29T12:57:43Z", + "unbonding_period":"2419200s", + "ccv_timeout_period": "2419200s", + "transfer_timeout_period": "3600s", + "consumer_redistribution_fraction": "0.75", + "blocks_per_distribution_transmission": "1500", + "historical_entries":"1000", + "distribution_transmission_channel": "" + }, + "power_shaping_parameters":{ + "top_N":50, + "validators_power_cap":50, + "validator_set_cap":50, + "allowlist":["cosmosvalcons1l9qq4m300z8c5ez86ak2mp8znftewkwgjlxh88"], + "denylist":[], + "min_stake": "1000", + "allow_inactive_vals":true + } +} +``` + +
+ +##### Remove Consumer + +The `remove-consumer` command allows to remove a consumer chain. + +```bash +interchain-security-pd tx provider remove-consumer [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider remove-consumer 0 +``` + +
+ +##### Opt In + +The `opt-in` command allows a validator to opt in to a consumer chain and optionally set a consensus public key. + +```bash +interchain-security-pd tx provider opt-in [consumer-id] [consumer-pubkey] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider opt-in 0 \ + '{"@type":"/cosmos.crypto.ed25519.PubKey","key":"Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is="}' \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +
+ +##### Opt Out + +The `opt-out` command allows validators to opt out from consumer chains. + +```bash +interchain-security-pd tx provider opt-out [consumer-id] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider opt-out 0 \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +
+ +##### Set Consumer Commission Rate + +The `set-consumer-commission-rate` command allows to set a per-consumer chain commission rate. + +```bash +interchain-security-pd tx provider set-consumer-commission-rate [consumer-id] [commission-rate] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider set-consumer-commission-rate 0 0.5 \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +
+ +##### Submit Consumer Double Voting + +The `submit-consumer-double-voting` command allows to submit a double voting evidence for a consumer chain. + +```bash +interchain-security-pd tx provider submit-consumer-double-voting [consumer-id] [evidence] [infraction_header] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider submit-consumer-double-voting 0 path/to/evidence.json path/to/infraction_header.json \ + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +where `evidence.json` contains: + +```json +{ + "vote_a": { + "type": "SIGNED_MSG_TYPE_PREVOTE", + "height": "59", + "round": 0, + "block_id": { + "hash": "paTPgLrLCZmw5ctQWlaMLJhXLckafakKN9skJbTiCHA=", + "part_set_header": { + "total": 1, + "hash": "pVOTT8MO00rk0HAeVQgzdP3wjIOzN5X5tfPLTtXIn2g=" + } + }, + "timestamp": "2024-09-26T09:34:47.146234009Z", + "validator_address": "mb06cu8SzQJOdYSzrJAK43Q8at8=", + "validator_index": 1, + "signature": "Z9C1oU5AEyFqXVmQ0LKNlaVa+tGh++95EB5HYe0i61PlREOmo/OTLlWedr8kuAThBu/1CpaLz446hYjISAKqBQ==", + "extension": null, + "extension_signature": null + }, + "vote_b": { + "type": "SIGNED_MSG_TYPE_PREVOTE", + "height": "59", + "round": 0, + "block_id": { + "hash": "07tksQsQ0gVBphgP4eeyGII9tEaLUuCauQcmwar9ktk=", + "part_set_header": { + "total": 1, + "hash": "nND/ClxCtoSJ9fC7Jyy884ab+nDh+PnHwI28T2fELCE=" + } + }, + "timestamp": "2024-09-26T09:34:47.051976301Z", + "validator_address": "mb06cu8SzQJOdYSzrJAK43Q8at8=", + "validator_index": 1, + "signature": "QscqC9ilH4gL7+3GPqLMWly+UkO+p0JgcinDZtfHOM4fYosZhx+TzhLrrXNExYpwX3D8qQHmJlLCcXLqbo7aCA==", + "extension": null, + "extension_signature": null + }, + "total_voting_power": "1500", + "validator_power": "500", + "timestamp": "2024-09-26T09:34:45.945436342Z" +} +``` + +and `infraction_header.json` contains: + +```json +{ + "signed_header": { + "header": { + "version": { + "block": "11", + "app": "0" + }, + "chain_id": "pion-1", + "height": "59", + "time": "2024-09-26T09:34:45.945436342Z", + "last_block_id": { + "hash": "t8HcmkQbchpGE1CxqdhcogoT+yD5VIm+cRGLcosTtxE=", + "part_set_header": { + "total": 1, + "hash": "fTediSh8XttUUoxWJLPIxO6iWecqdMMsegD2svBtR5E=" + } + }, + "last_commit_hash": "2U4mFcB6+FffQeFPUaHkd+eBtEV5/5d3Zy0Lk58dwIs=", + "data_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "validators_hash": "D26N3CL1zQt7yn+JUQ8Dcb2vCYG7QmHMiMfY+nGxhts=", + "next_validators_hash": "D26N3CL1zQt7yn+JUQ8Dcb2vCYG7QmHMiMfY+nGxhts=", + "consensus_hash": "BICRvH3cKD93v7+R1zxE2ljD34qcvIZ0Bdi389qtoi8=", + "app_hash": "k/RW/WMOYCS89VBhKMHIRYb30a30JkZ+puyp9ESTBiA=", + "last_results_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "evidence_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "proposer_address": "mb06cu8SzQJOdYSzrJAK43Q8at8=" + }, + "commit": { + "height": "59", + "round": 0, + "block_id": { + "hash": "07tksQsQ0gVBphgP4eeyGII9tEaLUuCauQcmwar9ktk=", + "part_set_header": { + "total": 1, + "hash": "nND/ClxCtoSJ9fC7Jyy884ab+nDh+PnHwI28T2fELCE=" + } + }, + "signatures": [ + { + "block_id_flag": "BLOCK_ID_FLAG_COMMIT", + "validator_address": "BsDz5HzFx0gmkIjcLzZBHTqqJ8Y=", + "timestamp": "2024-09-26T09:34:47.271500717Z", + "signature": "bXA2WgQVVlHAkn9mGIfoUvgn3C+EJCzNGTAjnhoQJwLkh1Okg3oYmwZRz+UGbc95kXyVO7kQSXhavt0ZPcJ4AA==" + }, + { + "block_id_flag": "BLOCK_ID_FLAG_COMMIT", + "validator_address": "mb06cu8SzQJOdYSzrJAK43Q8at8=", + "timestamp": "2024-09-26T09:34:47.305955426Z", + "signature": "YG1OcUhpTKFz+Uo8halNmkw0s6n333+m53laZvyQSHM5gqOG4h8jzij2u9sU4H404OJMgdj+1GTxuHmQ8jWFBg==" + }, + { + "block_id_flag": "BLOCK_ID_FLAG_COMMIT", + "validator_address": "tBw6QBQpY6pbEt3RxOWJDAs5JrE=", + "timestamp": "2024-09-26T09:34:47.255694467Z", + "signature": "EYOC/yo+RaosEVhwBy0bZFjVwHCR7rRZo/FmTRWpAIXZHBVrIiX3iVzRUwn78lsfbaoT97TsqRX61bAiJDM6BA==" + } + ] + } + }, + "validator_set": { + "validators": [ + { + "address": "BsDz5HzFx0gmkIjcLzZBHTqqJ8Y=", + "pub_key": { + "ed25519": "RrclQz9bIhkIy/gfL485g3PYMeiIku4qeo495787X10=" + }, + "voting_power": "500", + "proposer_priority": "-1000" + }, + { + "address": "mb06cu8SzQJOdYSzrJAK43Q8at8=", + "pub_key": { + "ed25519": "mAN6RXYxSM4MNGSIriYiS7pHuwAcOHDQAy9/wnlSzOI=" + }, + "voting_power": "500", + "proposer_priority": "500" + }, + { + "address": "tBw6QBQpY6pbEt3RxOWJDAs5JrE=", + "pub_key": { + "ed25519": "Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is=" + }, + "voting_power": "500", + "proposer_priority": "500" + } + ], + "proposer": { + "address": "BsDz5HzFx0gmkIjcLzZBHTqqJ8Y=", + "pub_key": { + "ed25519": "RrclQz9bIhkIy/gfL485g3PYMeiIku4qeo495787X10=" + }, + "voting_power": "500", + "proposer_priority": "-1000" + }, + "total_voting_power": "0" + }, + "trusted_height": { + "revision_number": "0", + "revision_height": "0" + }, + "trusted_validators": null +} +``` + +
+ +##### Submit Consumer Misbehaviour + +The `submit-consumer-misbehaviour` command allows to submit an IBC misbehaviour for a consumer chain. + +```bash +interchain-security-pd tx provider submit-consumer-misbehaviour [consumer-id] [misbehaviour] [flags] +``` + +
+ Example + +```bash +interchain-security-pd tx provider submit-consumer-misbehaviour 0 path/to/consumer-misbehaviour.json + --chain-id provider \ + --from mykey \ + --gas="auto" \ + --gas-adjustment="1.2" \ + --gas-prices="0.025stake" \ +``` + +where `consumer-misbehaviour.json` contains: + +```json +{ + "client_id": "07-tendermint-0", + "header_1": { + "signed_header": { + "header": { + "version": { + "block": "11", + "app": "0" + }, + "chain_id": "pion-1", + "height": "95", + "time": "2024-09-26T09:15:52.845591095Z", + "last_block_id": { + "hash": "PUph0B9N9X+LdrstqOoGf+W+OS6oHetQUa+0fpcRnF8=", + "part_set_header": { + "total": 1, + "hash": "SlVkAlM1uq3DjgTk0NbZftLlFwOEJrau1Wnhg3jEH3A=" + } + }, + "last_commit_hash": "Hxe4aLTULJ7qxJ10XsQfluKyU1Rn+d+cgDeTm2AATqU=", + "data_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "validators_hash": "+003y0s55+pbijWbJyVVgfNnquSaGGnQmC1hGRUIIjk=", + "next_validators_hash": "+003y0s55+pbijWbJyVVgfNnquSaGGnQmC1hGRUIIjk=", + "consensus_hash": "BICRvH3cKD93v7+R1zxE2ljD34qcvIZ0Bdi389qtoi8=", + "app_hash": "uGHlqLiNp+ZCjE889JDFKnrNkRpZ5xZ5OOamXrCNcOc=", + "last_results_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "evidence_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "proposer_address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=" + }, + "commit": { + "height": "95", + "round": 0, + "block_id": { + "hash": "hkUUob+4UVRE4uJW53fY9UYViGTs2v6P5Sb/hUFYyak=", + "part_set_header": { + "total": 1, + "hash": "0tx9pRIzYJ3vwrYyOgMC8zxf/sSJUtNVm9DBKM8Yxo0=" + } + }, + "signatures": [ + { + "block_id_flag": "BLOCK_ID_FLAG_COMMIT", + "validator_address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "timestamp": "2024-09-26T09:15:53.852414554Z", + "signature": "iiQCCxsCOoNVb2smAVmDO62o9HLf+I4rWk8o86uA1ZoFun/lk1bwrocaMp1It1SjVo/szYsX6Hp5rP1IwcAjDg==" + } + ] + } + }, + "validator_set": { + "validators": [ + { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "511", + "proposer_priority": "0" + } + ], + "proposer": { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "511", + "proposer_priority": "0" + }, + "total_voting_power": "0" + }, + "trusted_height": { + "revision_number": "0", + "revision_height": "20" + }, + "trusted_validators": { + "validators": [ + { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "500", + "proposer_priority": "0" + } + ], + "proposer": { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "500", + "proposer_priority": "0" + }, + "total_voting_power": "0" + } + }, + "header_2": { + "signed_header": { + "header": { + "version": { + "block": "11", + "app": "0" + }, + "chain_id": "consu", + "height": "95", + "time": "2024-09-26T09:15:54.044450012Z", + "last_block_id": { + "hash": "MG9B1h4R9Xb4GRjvaNydD5NSqT37OOjGDcatCZpBlco=", + "part_set_header": { + "total": 1, + "hash": "3jQh26/9EuNAAEL6v2tRuGhKtkotoyTqGtduOOn++vk=" + } + }, + "last_commit_hash": "s1hUy5e7i+GrH5IGW1ck4YHK2CDTY4fjnSiNMInJBWc=", + "data_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "validators_hash": "+003y0s55+pbijWbJyVVgfNnquSaGGnQmC1hGRUIIjk=", + "next_validators_hash": "+003y0s55+pbijWbJyVVgfNnquSaGGnQmC1hGRUIIjk=", + "consensus_hash": "BICRvH3cKD93v7+R1zxE2ljD34qcvIZ0Bdi389qtoi8=", + "app_hash": "bWRmShMthwEAB3lIVMgB673gH5vTdoqfn223M3Xrk6Q=", + "last_results_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "evidence_hash": "47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", + "proposer_address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=" + }, + "commit": { + "height": "95", + "round": 0, + "block_id": { + "hash": "z3MJTCXppRYoIEPOrneYzw/U0CSiYF3zsUv67ynxM6Q=", + "part_set_header": { + "total": 1, + "hash": "BFSlw7bqXxBHl9O5O9sCUB01nbe0T0KGOmv7yyr8KYU=" + } + }, + "signatures": [ + { + "block_id_flag": "BLOCK_ID_FLAG_COMMIT", + "validator_address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "timestamp": "2024-09-26T09:15:55.054809888Z", + "signature": "oi+TQ0yoDEeXyBchFIql9AGxbufnx3FzDKsCp4B8tx42ropD8tyotKOjk0OMuZQC5aMMRndRfKiYYsWiOrcpAg==" + } + ] + } + }, + "validator_set": { + "validators": [ + { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "511", + "proposer_priority": "0" + } + ], + "proposer": { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "511", + "proposer_priority": "0" + }, + "total_voting_power": "0" + }, + "trusted_height": { + "revision_number": "0", + "revision_height": "20" + }, + "trusted_validators": { + "validators": [ + { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "500", + "proposer_priority": "0" + } + ], + "proposer": { + "address": "3wkKSIC1TNV7KnnmTZ6Wm9dRSwk=", + "pub_key": { + "ed25519": "ujY14AgopV907IYgPAk/5x8c9267S4fQf89nyeCPTes=" + }, + "voting_power": "500", + "proposer_priority": "0" + }, + "total_voting_power": "0" + } + } +} +``` + +
+ +### gRPC + +A user can query the `provider` module using gRPC endpoints. + +#### Consumer Genesis + +The `QueryConsumerGenesis` endpoint queries a consumer chain genesis state by consumer id. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerGenesis +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0"}' localhost:9090 \ + interchain_security.ccv.provider.v1.Query/QueryConsumerGenesis +``` + +Output: + + +```json +{ + "genesisState": { + "params": { + "enabled": true, + "blocksPerDistributionTransmission": "1500", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "historicalEntries": "1000", + "unbondingPeriod": "2419200s", + "softOptOutThreshold": "0", + "retryDelayPeriod": "3600s" + }, + "provider": { + "clientState": { + "chainId": "provider", + "trustLevel": { + "numerator": "1", + "denominator": "3" + }, + "trustingPeriod": "57024s", + "unbondingPeriod": "86400s", + "maxClockDrift": "10s", + "frozenHeight": {}, + "latestHeight": { + "revisionHeight": "10" + }, + "proofSpecs": [ + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 33, + "minPrefixLength": 4, + "maxPrefixLength": 12, + "hash": "SHA256" + } + }, + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 32, + "minPrefixLength": 1, + "maxPrefixLength": 1, + "hash": "SHA256" + } + } + ], + "upgradePath": [ + "upgrade", + "upgradedIBCState" + ] + }, + "consensusState": { + "timestamp": "2024-09-26T08:19:42.708111Z", + "root": { + "hash": "xbZV/9QyM3PYzY/HyJAsNogaaJVJtyAGROTcXuqxHas=" + }, + "nextValidatorsHash": "/zLB6RSu9omrO5L0tnDK03hCOUibwl/7eeVC3hTP7so=" + }, + "initialValSet": [ + { + "pubKey": { + "ed25519": "E9bJ6bi7X9MG9s3djQ4MmBxshis9W15y7UzXCxp2Yuk=" + }, + "power": "100" + }, + { + "pubKey": { + "ed25519": "+9BFckSNCI1o/+S85HLjG3pYp1FIzmfYWVKmUH2njxs=" + }, + "power": "100" + }, + { + "pubKey": { + "ed25519": "62d2CCgWXYZHmlsCon2lzVgnu9gfubep2XRPlZKuLAQ=" + }, + "power": "100" + } + ] + }, + "newChain": true + } +} +``` + +
+ +#### List Consumer Chains + +The `QueryConsumerChains` endpoint queries consumer chains supported by the provider chain. +An optional integer parameter can be passed for phase filtering of consumer chains, (Registered=1|Initialized=2|Launched=3|Stopped=4|Deleted=5).` + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerChains +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"phase": "1"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerChains +``` + +Output: + +```json +{ + "chains": [ + { + "chainId": "pion-1", + "minPowerInTopN": "-1", + "phase": "CONSUMER_PHASE_REGISTERED", + "metadata": { + "name": "pion-1", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your chain" + }, + "consumerId": "2" + }, + { + "chainId": "dash-2", + "minPowerInTopN": "-1", + "phase": "CONSUMER_PHASE_REGISTERED", + "metadata": { + "name": "dash-2", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your chain" + }, + "consumerId": "4" + }, + ], + "pagination": { + "total": "6" + } +} +``` + +
+ +#### Validator Consumer Key Assignment + +The `QueryValidatorConsumerAddr` endpoint queries the address assigned by a validator for a consumer chain. + + +```bash +interchain_security.ccv.provider.v1.Query/QueryValidatorConsumerAddr +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0", "provider_address": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryValidatorConsumerAddr +``` + +Output: + +```json +{ + "consumerAddress": "cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk" +} +``` + +
+ +#### Validator Provider Key + +The `QueryValidatorProviderAddr` endpoint queries the provider chain address given a consumer chain validator address. + +```bash +interchain_security.ccv.provider.v1.Query/QueryValidatorProviderAddr +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0", "provider_address": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryValidatorProviderAddr +``` + +Output: + +```json +{ + "consumerAddress": "cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk" + } +``` +
+ +#### Throttle State + +The `QueryThrottleState` queries the main on-chain state relevant to slash packet throttling. + +```bash +interchain_security.ccv.provider.v1.Query/QueryThrottleState +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.provider.v1.Query/QueryThrottleState +``` + +Output: + +```json +{ + "slashMeter": "15", + "slashMeterAllowance": "15", + "nextReplenishCandidate": "2024-09-26T14:27:38.066958Z" +} +``` + +
+ +##### Registered Consumer Reward Denoms + +The `QueryRegisteredConsumerRewardDenoms` command allows to query registered consumer reward denoms + +```bash +interchain_security.ccv.provider.v1.Query/QueryRegisteredConsumerRewardDenoms +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.provider.v1.Query/QueryRegisteredConsumerRewardDenoms +``` + +Output: + +```json +{ + "denoms": [ + "ibc/0025F8A87464A471E66B234C4F93AEC5B4DA3D42D7986451A059273426290DD5", + "ibc/054892D6BB43AF8B93AAC28AA5FD7019D2C59A15DAFD6F45C1FA2BF9BDA22454", + "uatom" + ] +} +``` + +
+ +#### All Pairs Valconsensus Address + +The `QueryAllPairsValConsAddrByConsumer` endpoint queries the list of pair valconsensus address between provider and consumer chain. + +```bash +interchain_security.ccv.provider.v1.Query/QueryAllPairsValConsAddrByConsumer +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryAllPairsValConsAddrByConsumer +``` + +Output: + +```json +{ + "pairValConAddr": [ + { + "providerAddress": "cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6", + "consumerAddress": "cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk", + "consumerKey": { + "ed25519": "Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is=" + } + } + ] +} +``` + +
+ +#### Provider Parameters + +The `QueryParams` endpoint queries all current values of provider parameters. + +```bash +interchain_security.ccv.provider.v1.Query/QueryParams +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.provider.v1.Query/QueryParams +``` + +Output: + +```json +{ + "params": { + "templateClient": { + "trustLevel": { + "numerator": "1", + "denominator": "3" + }, + "trustingPeriod": "0s", + "unbondingPeriod": "0s", + "maxClockDrift": "10s", + "frozenHeight": {}, + "latestHeight": {}, + "proofSpecs": [ + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 33, + "minPrefixLength": 4, + "maxPrefixLength": 12, + "hash": "SHA256" + } + }, + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 32, + "minPrefixLength": 1, + "maxPrefixLength": 1, + "hash": "SHA256" + } + } + ], + "upgradePath": [ + "upgrade", + "upgradedIBCState" + ] + }, + "trustingPeriodFraction": "0.66", + "ccvTimeoutPeriod": "2419200s", + "slashMeterReplenishPeriod": "3600s", + "slashMeterReplenishFraction": "0.05", + "consumerRewardDenomRegistrationFee": { + "denom": "stake", + "amount": "10000000" + }, + "blocksPerEpoch": "5", + "numberOfEpochsToStartReceivingRewards": "24", + "maxProviderConsensusValidators": "180" + } +} +``` + +
+ +#### Consumer Opted In Validators + +The `QueryConsumerChainOptedInValidators` endpoint queries opted-in validators for a given consumer chain. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerChainOptedInValidators +``` + +
+ Example + +```bash + grpcurl -plaintext -d '{"consumer_id": "0"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerChainOptedInValidators +``` + +Output: + +```json +{ + "validatorsProviderAddresses": [ + "cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph", + "cosmosvalcons1jnq3j55qe4f946qj8499w0tntxwz90atx26p4q", + "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj" + ] +} +``` + +
+ +#### Consumer Validators + +The `QueryConsumerValidators` endpoint queries the latest set consumer-validator set for a given consumer ID. +Note that this does not necessarily mean that the consumer chain is using this validator set at this exact moment because a VSCPacket could be delayed to be delivered on the consumer chain. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerValidators +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerValidators +``` + +Output: + +```json +{ + "validators": [ + { + "providerAddress": "cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph", + "consumerKey": { + "ed25519": "62d2CCgWXYZHmlsCon2lzVgnu9gfubep2XRPlZKuLAQ=" + }, + "rate": "0", + "consumerPower": "101", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "bob" + }, + "providerOperatorAddress": "cosmosvaloper1a7u5k6f54ua3tptl9yn6u82yrvayet6sxn9ywn", + "status": "BOND_STATUS_BONDED", + "providerTokens": "101000000", + "providerPower": "101", + "validatesCurrentEpoch": true + }, + { + "providerAddress": "cosmosvalcons1jnq3j55qe4f946qj8499w0tntxwz90atx26p4q", + "consumerKey": { + "ed25519": "+9BFckSNCI1o/+S85HLjG3pYp1FIzmfYWVKmUH2njxs=" + }, + "rate": "0", + "consumerPower": "100", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "coordinator" + }, + "providerOperatorAddress": "cosmosvaloper1jk2pp655zxy2gazhxj50s8jk3750y8np6wz4lm", + "status": "BOND_STATUS_BONDED", + "providerTokens": "100000000", + "providerPower": "100", + "validatesCurrentEpoch": true + }, + { + "providerAddress": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj", + "consumerKey": { + "ed25519": "Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is=" + }, + "rate": "0", + "consumerPower": "100", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "alice" + }, + "providerOperatorAddress": "cosmosvaloper19vfen9jn3uk3e6rrkt3pxansunujnlm40wpdvg", + "status": "BOND_STATUS_BONDED", + "providerTokens": "100000000", + "providerPower": "100", + "validatesCurrentEpoch": true + } + ] +} +``` + +
+ +#### Has to Validate + +The `QueryConsumerChainsValidatorHasToValidate` endpoint queries a list of consumer chains that a given validator must validate. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerChainsValidatorHasToValidate +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"provider_address": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerChainsValidatorHasToValidate +``` + +Output: + +```json +{ + "consumerIds": [ + "0", + "2" + ] +} +``` + +
+ +#### Validator Consumer Commission Rate + +The `QueryValidatorConsumerCommissionRate` endpoint queries the consumer commission rate a validator charges on a consumer chain. + +```bash +interchain_security.ccv.provider.v1.Query/QueryValidatorConsumerCommissionRate +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0", "provider_address": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryValidatorConsumerCommissionRate +``` + +Output: + +```json +{ + "rate": "750000000000000000" +} +``` + +
+ +#### Blocks Until Next Epoch + +The `QueryBlocksUntilNextEpoch` endpoint allows to query the number of blocks until the next epoch begins and validator updates are sent to consumer chains. + +```bash +interchain_security.ccv.provider.v1.Query/QueryBlocksUntilNextEpoch +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.provider.v1.Query/QueryBlocksUntilNextEpoch +``` + +Output: + +```json +{ + "blocks_until_next_epoch":"4" +} +``` + +
+ +#### Consumer Id From Client Id + +The `QueryConsumerIdFromClientId` endpoint allows to query the consumer id of the chain associated with the provided client id. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerIdFromClientId +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"client_id":"07-tendermint-0"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerIdFromClientId +``` + +Output: + +```bash +{ + "consumerId": "0" +} +``` + +
+ +#### Consumer Chain + +The `QueryConsumerChain` command allows to query the consumer chain associated with the consumer id. + +```bash +interchain_security.ccv.provider.v1.Query/QueryConsumerChain +``` + +
+ Example + +```bash +grpcurl -plaintext -d '{"consumer_id": "0"}' localhost:9090 interchain_security.ccv.provider.v1.Query/QueryConsumerChain +``` + +```json +{ + "consumerId": "0", + "chainId": "pion-1", + "ownerAddress": "cosmos10d07y265gmmuvt4z0w9aw880jnsr700j6zn9kn", + "phase": "CONSUMER_PHASE_LAUNCHED", + "metadata": { + "name": "description of your chain and all other relevant information", + "description": "some metadata about your chain", + "metadata": "pion-1" + }, + "initParams": { + "initialHeight": { + "revisionHeight": "1" + }, + "genesisHash": "2D5C2110941DA54BE07CBB9FACD7E4A2E3253E79BE7BE3E5A1A7BDA518BAA4BE", + "binaryHash": "2D5C2110941DA54BE07CBB9FACD7E4A2E3253E79BE7BE3E5A1A7BDA518BAA4BE", + "spawnTime": "2023-03-11T17:02:14.718477Z", + "unbondingPeriod": "2419200s", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "blocksPerDistributionTransmission": "1500", + "historicalEntries": "1000" + }, + "powerShapingParams": { + "topN": 100, + "validatorSetCap": 50, + "minStake": "1000", + "allowInactiveVals": true + } +} +``` + +
+ +### REST + +A user can query the `provider` module using REST endpoints. + + +#### Consumer Genesis + +The `consumer_genesis` endpoint queries a consumer chain genesis state by consumer id. + +```bash +interchain_security/ccv/provider/consumer_genesis/{consumer_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_genesis/0 +``` + +Output: + +```json +{ + "genesisState": { + "params": { + "enabled": true, + "blocksPerDistributionTransmission": "1500", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "historicalEntries": "1000", + "unbondingPeriod": "2419200s", + "softOptOutThreshold": "0", + "retryDelayPeriod": "3600s" + }, + "provider": { + "clientState": { + "chainId": "provider", + "trustLevel": { + "numerator": "1", + "denominator": "3" + }, + "trustingPeriod": "57024s", + "unbondingPeriod": "86400s", + "maxClockDrift": "10s", + "frozenHeight": {}, + "latestHeight": { + "revisionHeight": "10" + }, + "proofSpecs": [ + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 33, + "minPrefixLength": 4, + "maxPrefixLength": 12, + "hash": "SHA256" + } + }, + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 32, + "minPrefixLength": 1, + "maxPrefixLength": 1, + "hash": "SHA256" + } + } + ], + "upgradePath": [ + "upgrade", + "upgradedIBCState" + ] + }, + "consensusState": { + "timestamp": "2024-09-26T08:19:42.708111Z", + "root": { + "hash": "xbZV/9QyM3PYzY/HyJAsNogaaJVJtyAGROTcXuqxHas=" + }, + "nextValidatorsHash": "/zLB6RSu9omrO5L0tnDK03hCOUibwl/7eeVC3hTP7so=" + }, + "initialValSet": [ + { + "pubKey": { + "ed25519": "E9bJ6bi7X9MG9s3djQ4MmBxshis9W15y7UzXCxp2Yuk=" + }, + "power": "100" + }, + { + "pubKey": { + "ed25519": "+9BFckSNCI1o/+S85HLjG3pYp1FIzmfYWVKmUH2njxs=" + }, + "power": "100" + }, + { + "pubKey": { + "ed25519": "62d2CCgWXYZHmlsCon2lzVgnu9gfubep2XRPlZKuLAQ=" + }, + "power": "100" + } + ] + }, + "newChain": true + } +} +``` + +
+ +#### List Consumer Chains + +The `consumer_chains` endpoint queries consumer chains supported by the provider chain. +An optional integer parameter can be passed for phase filtering of consumer chains, (Registered=1|Initialized=2|Launched=3|Stopped=4|Deleted=5).` + +```bash +interchain_security/ccv/provider/consumer_chains/{phase} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_chains/3 +``` + +Output: + +```json +{ + "chains": [ + { + "chainId": "pion-1", + "minPowerInTopN": "-1", + "phase": "CONSUMER_PHASE_REGISTERED", + "metadata": { + "name": "pion-1", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your chain" + }, + "consumerId": "2" + }, + { + "chainId": "dash-2", + "minPowerInTopN": "-1", + "phase": "CONSUMER_PHASE_REGISTERED", + "metadata": { + "name": "dash-2", + "description":"description of your chain and all other relevant information", + "metadata": "some metadata about your chain" + }, + "consumerId": "4" + }, + ], + "pagination": { + "total": "6" + } +} +``` + +
+ +#### Validator Consumer Key Assignment + +The `validator_consumer_addr` endpoint queries the address assigned by a validator for a consumer chain. + +```bash +/interchain_security/ccv/provider/validator_consumer_addr/{consumer_id}/{provider_address} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/validator_consumer_addr/0/cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj +``` + +Output: + +```json +{ + "consumerAddress": "cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk" +} +``` + +
+ +#### Validator Provider Key + +The `validator_provider_addr` endpoint queries the provider chain address given a consumer chain validator address. + +```bash +/interchain_security/ccv/provider/validator_provider_addr/{consumer_id}/{consumer_address} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/validator_provider_addr/0/cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk +``` + +Output: + +```json +{ + "providerAddress": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj" +} +``` + +
+ +#### Throttle State + +The `throttle_state` queries the main on-chain state relevant to slash packet throttling. + +```bash +"/interchain_security/ccv/provider/throttle_state" +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/throttle_state +``` + +Output: + +```json +{ + "slashMeter": "15", + "slashMeterAllowance": "15", + "nextReplenishCandidate": "2024-09-26T14:27:38.066958Z" +} +``` + +
+ +#### Registered Consumer Reward Denoms + +The `registered_consumer_reward_denoms` endpoint allows to query registered consumer reward denoms + +```bash +interchain_security/ccv/provider/registered_consumer_reward_denoms +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/registered_consumer_reward_denoms +``` + +Output: + +```json +{ + "denoms": [ + "ibc/0025F8A87464A471E66B234C4F93AEC5B4DA3D42D7986451A059273426290DD5", + "ibc/054892D6BB43AF8B93AAC28AA5FD7019D2C59A15DAFD6F45C1FA2BF9BDA22454", + "uatom" + ] +} +``` + +
+ +#### All Pairs Valconsensus Address + +The `address_pairs` endpoint queries the list of pair valconsensus address between provider and consumer chain. + +```bash +interchain_security/ccv/provider/address_pairs/{consumer_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/address_pairs/0 +``` + +Output: + +```json +{ + "pairValConAddr": [ + { + "providerAddress": "cosmosvalcons1ezyrq65s3gshhx5585w6mpusq3xsj3ayzf4uv6", + "consumerAddress": "cosmosvalcons1kswr5sq599365kcjmhgufevfps9njf43e4lwdk", + "consumerKey": { + "ed25519": "Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is=" + } + } + ] +} +``` + +
+ +#### Provider Parameters + +The `params` endpoint queries all current values of provider parameters + +```bash +interchain_security/ccv/provider/params +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/params +``` + +Output: + +```json +{ + "params": { + "templateClient": { + "trustLevel": { + "numerator": "1", + "denominator": "3" + }, + "trustingPeriod": "0s", + "unbondingPeriod": "0s", + "maxClockDrift": "10s", + "frozenHeight": {}, + "latestHeight": {}, + "proofSpecs": [ + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 33, + "minPrefixLength": 4, + "maxPrefixLength": 12, + "hash": "SHA256" + } + }, + { + "leafSpec": { + "hash": "SHA256", + "prehashValue": "SHA256", + "length": "VAR_PROTO", + "prefix": "AA==" + }, + "innerSpec": { + "childOrder": [ + 0, + 1 + ], + "childSize": 32, + "minPrefixLength": 1, + "maxPrefixLength": 1, + "hash": "SHA256" + } + } + ], + "upgradePath": [ + "upgrade", + "upgradedIBCState" + ] + }, + "trustingPeriodFraction": "0.66", + "ccvTimeoutPeriod": "2419200s", + "slashMeterReplenishPeriod": "3600s", + "slashMeterReplenishFraction": "0.05", + "consumerRewardDenomRegistrationFee": { + "denom": "stake", + "amount": "10000000" + }, + "blocksPerEpoch": "5", + "numberOfEpochsToStartReceivingRewards": "24", + "maxProviderConsensusValidators": "180" + } +} +``` + +
+ +#### Consumer Opted In Validators + +The `opted_in_validators` endpoint queries opted-in validators for a given consumer chain + +```bash +/interchain_security/ccv/provider/opted_in_validators/{consumer_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/opted_in_validators/0 +``` + +Output: + +```json +{ + "validatorsProviderAddresses": [ + "cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph", + "cosmosvalcons1jnq3j55qe4f946qj8499w0tntxwz90atx26p4q", + "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj" + ] +} +``` + +
+ +#### Consumer Validators + +The `consumer_validators` endpoint queries the latest set consumer-validator set for a given consumer ID +Note that this does not necessarily mean that the consumer chain is using this validator set at this exact moment because a VSCPacket could be delayed to be delivered on the consumer chain. + +```bash +/interchain_security/ccv/provider/consumer_validators/{consumer_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_validators/0 +``` + +Output: + +```json +{ + "validators": [ + { + "providerAddress": "cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph", + "consumerKey": { + "ed25519": "62d2CCgWXYZHmlsCon2lzVgnu9gfubep2XRPlZKuLAQ=" + }, + "rate": "0", + "consumerPower": "101", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "bob" + }, + "providerOperatorAddress": "cosmosvaloper1a7u5k6f54ua3tptl9yn6u82yrvayet6sxn9ywn", + "status": "BOND_STATUS_BONDED", + "providerTokens": "101000000", + "providerPower": "101", + "validatesCurrentEpoch": true + }, + { + "providerAddress": "cosmosvalcons1jnq3j55qe4f946qj8499w0tntxwz90atx26p4q", + "consumerKey": { + "ed25519": "+9BFckSNCI1o/+S85HLjG3pYp1FIzmfYWVKmUH2njxs=" + }, + "rate": "0", + "consumerPower": "100", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "coordinator" + }, + "providerOperatorAddress": "cosmosvaloper1jk2pp655zxy2gazhxj50s8jk3750y8np6wz4lm", + "status": "BOND_STATUS_BONDED", + "providerTokens": "100000000", + "providerPower": "100", + "validatesCurrentEpoch": true + }, + { + "providerAddress": "cosmosvalcons1h7zs5nwruzvhyzkktvhwypfuxlch6nrrw4jjmj", + "consumerKey": { + "ed25519": "Ui5Gf1+mtWUdH8u3xlmzdKID+F3PK0sfXZ73GZ6q6is=" + }, + "rate": "0", + "consumerPower": "100", + "consumerCommissionRate": "100000000000000000", + "providerCommissionRate": "100000000000000000", + "description": { + "moniker": "alice" + }, + "providerOperatorAddress": "cosmosvaloper19vfen9jn3uk3e6rrkt3pxansunujnlm40wpdvg", + "status": "BOND_STATUS_BONDED", + "providerTokens": "100000000", + "providerPower": "100", + "validatesCurrentEpoch": true + } + ] +} +``` + +
+ +#### Has to Validate + +The `consumer_chains_per_validator` endpoint queries a list of consumer chains that a given validator must validate. + +```bash +interchain_security/ccv/provider/consumer_chains_per_validator/{provider_address} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_chains_per_validator/cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph +``` + +Output: + +```json +{ + "consumerIds": [ + "0", + "2" + ] +} +``` + +
+ +#### Validator Consumer Commission Rate + +The `consumer_commission_rate` endpoint queries the consumer commission rate a validator charges on a consumer chain. + +```bash +/interchain_security/ccv/provider/consumer_commission_rate/{consumer_id}/{provider_address} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_commission_rate/0/cosmosvalcons1znhu88l6dsvexunfem4u0392kwqyvdkrj66wph +``` + +Output: + +```json +{ + "rate": "0.100000000000000000" +} +``` + +
+ +#### Blocks Until Next Epoch + +The `blocks_until_next_epoch` endpoint allows to query the number of blocks until the next epoch begins and validator updates are sent to consumer chains + +```bash +interchain_security/ccv/provider/blocks_until_next_epoch +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/blocks_until_next_epoch +``` + +Output: + +```json +{ + "blocks_until_next_epoch":"3" +} +``` + +
+ +#### Consumer Id From Client Id + +The `consumer_id` endpoint allows to query the consumer id of the chain associated with the provided client id + +```bash +/interchain_security/ccv/provider/consumer_id/{client_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_id/07-tendermint-0 +``` + +Output: + +```json +{ + "consumer_id":"0" +} +``` + +
+ +#### Consumer Chain + +The `consumer_chain` endpoint allows to query the consumer chain associated with the consumer id. + +```bash +interchain_security/ccv/provider/consumer_chain/{consumer_id} +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/provider/consumer_chain/0 +``` + +Output: + +```json +{ + "consumerId": "0", + "chainId": "pion-1", + "ownerAddress": "cosmos10d07y265gmmuvt4z0w9aw880jnsr700j6zn9kn", + "phase": "CONSUMER_PHASE_LAUNCHED", + "metadata": { + "name": "description of your chain and all other relevant information", + "description": "some metadata about your chain", + "metadata": "pion-1" + }, + "initParams": { + "initialHeight": { + "revisionHeight": "1" + }, + "genesisHash": "2D5C2110941DA54BE07CBB9FACD7E4A2E3253E79BE7BE3E5A1A7BDA518BAA4BE", + "binaryHash": "2D5C2110941DA54BE07CBB9FACD7E4A2E3253E79BE7BE3E5A1A7BDA518BAA4BE", + "spawnTime": "2023-03-11T17:02:14.718477Z", + "unbondingPeriod": "2419200s", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "blocksPerDistributionTransmission": "1500", + "historicalEntries": "1000" + }, + "powerShapingParams": { + "topN": 100, + "validatorSetCap": 50, + "minStake": "1000", + "allowInactiveVals": true + } +} +``` + +
diff --git a/docs/docs/build/modules/03-consumer.md b/docs/docs/build/modules/03-consumer.md new file mode 100644 index 0000000000..289ffb2bcb --- /dev/null +++ b/docs/docs/build/modules/03-consumer.md @@ -0,0 +1,882 @@ +--- +sidebar_position: 3 +toc_min_heading_level: 2 +toc_max_heading_level: 5 +--- + +# x/ccv/consumer + +## Overview + +The ICS consumer module enables consumer chains to use stake locked on a provider chain +as collateral for their own proof-of-stake based block production. + +The consumer module established a IBC ordered channel to the provider chain. +This channel is used by the provider chain to regularly sent validator updates the the consumer chain. +The consumer sends these updates to its own consensus engine. +This means that the consumer module acts as a staking module of the consumer chain. + +Regularly, the consumer module sends a part of the consumer chain's block rewards +to the provider chain as ICS rewards. + +If one of the validators in the consumer chain's validator set is missing enough blocks (i.e., downtime infraction), +the consumer module notifies the provider chain by sending an IBC packet to the provider module. +As a result, the misbehaving validator is punished on the provider chain. + +## State + +For clarity, the description of the the consumer module state is split into features. +For a more accurate description, check out the `x/ccv/consumer/types/keys.go` file, which contains the definitions of all the keys. + +### Provider Connection + +#### ProviderClientID + +`ProviderClientID` is the ID of the provider client on which the CCV channel is built. + +Format: `byte(3) -> string` + +#### ProviderChannelID + +`ProviderChannelID` is the ID of the CCV channel. + +Format: `byte(4) -> string` + +### Changeover + +#### PreCCV + +`PreCCV` is the flag set when the consumer chain is in the process of a standalone to consumer chain changeover. + +Format: `byte(7) -> uint64` + +#### InitialValSet + +`InitialValSet` is the initial validator set on the consumer chain. + +Format: `byte(8) -> GenesisState` + +Note that only the `InitialValSet` field of the `ProviderInfo` field of `GenesisState` is set, i.e., + +```proto +message GenesisState { + ... + ProviderInfo provider = 14 + [ (gogoproto.nullable) = false ]; +} + +message ProviderInfo { + // InitialValset filled in on new chain and on restart. + repeated .tendermint.abci.ValidatorUpdate initial_val_set = 3 + [ (gogoproto.nullable) = false ]; +} +``` + +#### InitGenesisHeight + +`InitGenesisHeight` is the height when the consumer module was initialized (i.e., the `InitGenesis` method was called). + +Format: `byte(17) -> uint64` + +#### PrevStandaloneChain + +`PrevStandaloneChain` is the flag set when the consumer chain was previously a standalone chain. + +Format: `byte(19) -> []byte{}` + +### Validator Updates + +#### PendingChanges + +`PendingChanges` are the validator updates received from the provider that were not yet sent to the consensus engine. + +Format: `byte(5) -> ValidatorSetChangePacketData` + +Note that only the `ValidatorUpdates` field of `ValidatorSetChangePacketData` is set. + +#### CrossChainValidator + +`CrossChainValidator` is the internal state of a consumer validator with consensus address `addr`. + +Format: `byte(16) | addr -> CrossChainValidator`, where `CrossChainValidator` is defined as + +```proto +message CrossChainValidator { + bytes address = 1; + int64 power = 2; + // pubkey is the consensus public key of the validator, as a Protobuf Any. + google.protobuf.Any pubkey = 3 [ + (cosmos_proto.accepts_interface) = "cosmos.crypto.PubKey", + (gogoproto.moretags) = "yaml:\"consensus_pubkey\"" + ]; + // deprecated + bool opted_out = 4 [deprecated = true]; +} +``` + +#### HistoricalInfo + +`HistoricalInfo` is the header and validator information for a given block. +For more details, see the [Cosmos SDK docs](https://docs.cosmos.network/v0.50/build/modules/staking#historicalinfo). + +Format: `byte(11) | height -> HistoricalInfo`, where `HistoricalInfo` is define in the staking module as + +```proto +message HistoricalInfo { + tendermint.types.Header header = 1 [(gogoproto.nullable) = false, (amino.dont_omitempty) = true]; + repeated Validator valset = 2 [(gogoproto.nullable) = false, (amino.dont_omitempty) = true]; +} +``` + +### Reward Distribution + +#### LastDistributionTransmission + +`LastDistributionTransmission` is the block height of the last attempt to send ICS rewards to the provider module. + +Format: `byte(1) -> LastTransmissionBlockHeight`, where `LastTransmissionBlockHeight` is defined as + +```proto +message LastTransmissionBlockHeight { + int64 height = 1; +} +``` + +### Downtime Infractions + +#### OutstandingDowntime + +`OutstandingDowntime` is the flag set when a `SlashPacket` is queued to be sent to the provider for a downtime infraction of a validator with consensus address `addr`. +The flag is unset when receiving from the provider a `VSCPacket` with a slash acknowledgement (see `SlashAcks` in `ValidatorSetChangePacketData`). + +Format: `byte(14) | addr -> []byte{}` + +#### HeightValsetUpdateID + +`HeightValsetUpdateID` is the validator set update ID associated with a block height. + +Format: `byte(13) | height -> uint64` + +#### PendingPacketsIndex + +`PendingPacketsIndex` is the next index available to store packet data to be sent to the provider chain (see below). + +Format: `byte(20) -> uint64` + +#### PendingDataPacketsV1 + +`PendingDataPacketsV1` is the queue of packet data to be sent to the provider chain. +In general, packets in this queue will be sent to the provider in the end blocker, unless + +- the CCV channel is not yet established; +- the provider client is expired; +- the last slash packet sent was not yet acknowledged by the provider chain. + +Format: `byte(15) | index -> ConsumerPacketData`, where `index` is the index of the packet in the queue and `ConsumerPacketData` is defined as + +```proto +message ConsumerPacketData { + ConsumerPacketDataType type = 1; + + oneof data { + SlashPacketData slashPacketData = 2; + VSCMaturedPacketData vscMaturedPacketData = 3; + } +} +``` + +#### SlashRecord + +`SlashRecord` is the record storing the state of a SlashPacket sent to the provider chain that was not yet acknowledged. +See [ADR 008](../../adrs/adr-008-throttle-retries.md) for more details. + +Format: `byte(21) -> SlashRecord`, where `SlashRecord` is defined as + +```proto +message SlashRecord { + bool waiting_on_reply = 1; + google.protobuf.Timestamp send_time = 2 + [ (gogoproto.stdtime) = true, (gogoproto.nullable) = false ]; +} +``` + +## State Transitions + +> TBA + +## IBC Callbacks + +The consumer module is an IBC application that implements the [IBC module callback](https://ibc.cosmos.network/v8/ibc/apps/apps/#create-a-custom-ibc-application-module). + +### OnChanOpenInit + +`OnChanOpenInit` first verifies that the CCV channel was not already created. +Then, it validates the channel parameters -- an ordered IBC channel connected on the `consumer` port +and with the counterparty port set to `provider` -- and asserts that the version matches the expected version +(only verions `1` is supported). + +Finally, it verifies that the underlying client is the expected client of the provider chain +(i.e., provided in the consumer module genesis state). + +### OnChanOpenTry + +`OnChanOpenTry` returns an error. `MsgChannelOpenTry` should be sent to the provider. + +### OnChanOpenAck + +`OnChanOpenAck` first verifies that the CCV channel was not already created. +Then it verifies that the counterparty version matches the expected version +(only verions `1` is supported). + +If the verification passes, it stores the [ProviderFeePoolAddr](#providerfeepooladdrstr) in the state. + +Finally, if the [DistributionTransmissionChannel](#distributiontransmissionchannel) parameter is not set, +it initiate the opening handshake for a token transfer channel over the same connection as the CCV channel +by calling the `ChannelOpenInit` method of the IBC module. + +### OnChanOpenConfirm + +`OnChanOpenConfirm` returns an error. `MsgChanOpenConfirm` should be sent to the provider. + +### OnChanCloseInit + +`OnChanCloseInit` allow relayers to close duplicate OPEN channels, if the channel handshake is completed. + +### OnChanCloseConfirm + +`OnChanCloseConfirm` is a no-op. + +### OnRecvPacket + +`OnRecvPacket` unmarshals the IBC packet data into a `ValidatorSetChangePacketData` struct (see below) and executes the handling logic. + +- If it is the first packet received, sets the underlying IBC channel as the canonical CCV channel. +- Collects validator updates to be sent to the consensus engine at the end of the block. +- Store in state the block height to VSC id (i.e., `valset_update_id`) mapping. +- Removed the outstanding downtime flags from the validator for which the jailing + for downtime infractions was acknowledged by the provider chain (see the `slash_acks` field in `ValidatorSetChangePacketData`). + +```proto +message ValidatorSetChangePacketData { + repeated .tendermint.abci.ValidatorUpdate validator_updates = 1 [ + (gogoproto.nullable) = false, + (gogoproto.moretags) = "yaml:\"validator_updates\"" + ]; + uint64 valset_update_id = 2; + // consensus address of consumer chain validators + // successfully jailed on the provider chain + repeated string slash_acks = 3; +} +``` + +### OnAcknowledgementPacket + +`OnAcknowledgementPacket` enables the consumer module to confirm that the provider module received +the previously sent `SlashPacket` and it unblocks the sending of the next `SlashPacket`. +This functionality is needed for throttling jailing on the provider chain. For more details, see [ADR-008](../../adrs/adr-008-throttle-retries.md). + +### OnTimeoutPacket + +`OnTimeoutPacket` is a no-op. + +## Messages + +### MsgUpdateParams + +`MsgUpdateParams` updates the [consumer module parameters](#parameters). +The params are updated through a governance proposal where the signer is the gov module account address. + +```proto +message MsgUpdateParams { + option (cosmos.msg.v1.signer) = "authority"; + + // signer is the address of the governance account. + string authority = 1 [(cosmos_proto.scalar) = "cosmos.AddressString"]; + + // params defines the x/consumer parameters to update. + interchain_security.ccv.v1.ConsumerParams params = 2 [(gogoproto.nullable) = false]; +} +``` + +## BeginBlock + +In the `BeginBlock` of the consumer module the following actions are performed: + +- Store in state the block height to VSC id mapping needed for sending to the provider the height of infractions committed on the consumer chain. +- Track historical entries. This is the same lofic as in the `x/staking` module. + +## EndBlock + +In the `EndBlock` of the consumer module the following actions are performed: + +- If `PreCCV` state is active, i.e., the consumer chain is a previously standalone chain + that was just upgraded to include the consumer module, then execute the [changeover logic](../../consumer-development/changeover-procedure.md). +- Otherwise, distribute block rewards internally and once every [BlocksPerDistributionTransmission](#blocksperdistributiontransmission) send + ICS rewards to the provider chain. +- Send slash packets to the provider chain reporting infractions validators commited on the consumer chain. +- Send to the consensus engine validator updates reveived from the provider chain. + +## Hooks + +> TBA + +## Events + +> TBA + +## Parameters + +:::warning +The consumer module parameters are set by the provider when creating the consumer genesis (i.e., when launching the consumer chain). +As a result, changes of these parameters might results in incompatibilities between different versions of consumers and providers. +::: + +The consumer module contains the following parameters. + +### Enabled + +`Enabled` is deprecated. + +### BlocksPerDistributionTransmission + + +| Type | Default value | +| ----- | ------------- | +| int64 | 1000 | + +`BlocksPerDistributionTransmission` is the number of blocks between rewards transfers from the consumer to the provider. + +### DistributionTransmissionChannel + +| Type | Default value | +| ------ | ------------- | +| string | "" | + +`DistributionTransmissionChannel` is the provider chain IBC channel used for receiving consumer chain reward distribution token transfers. This is automatically set during the consumer-provider handshake procedure. + +Providing an IBC transfer channel enables a consumer chain to re-use one of the existing channels to the provider for consumer chain rewards distribution. +This will preserve the `ibc denom` that may already be in use. +This is especially important for standalone chains transitioning to become consumer chains. +For more details, see the [changeover procedure](../consumer-development/changeover-procedure.md). + +### ProviderFeePoolAddrStr + +| Type | Default value | +| ------ | ------------- | +| string | "" | + +`ProviderFeePoolAddrStr` is the provider chain fee pool address used for receiving consumer chain reward distribution token transfers. This is automatically set during the consumer-provider handshake procedure. + +### CcvTimeoutPeriod + +| Type | Default value | +| ------------- | ------------------ | +| time.Duration | 2419200s (4 weeks) | + +`CcvTimeoutPeriod` is the period used to compute the timeout timestamp when sending IBC packets. +`CcvTimeoutPeriod` may have different values on the provider and consumer chains. + +### TransferTimeoutPeriod + +| Type | Default value | +| ------------- | -------------- | +| time.Duration | 3600s (1 hour) | + +`TransferTimeoutPeriod` is the timeout period for consumer chain reward distribution IBC packets. + +### ConsumerRedistributionFraction + +| Type | Default value | +| ------ | ------------- | +| string | "0.75" | + +`ConsumerRedistributionFraction` is the fraction of tokens allocated to the consumer redistribution address during distribution events. +The fraction is a string representing a decimal number. For example `"0.75"` would represent `75%`. +For example, a consumer with `ConsumerRedistributionFraction` set to `"0.75"` would send `75%` of its block rewards and accumulated fees to the consumer redistribution address, and the remaining `25%` to the provider chain every `BlocksPerDistributionTransmission` blocks. + +### HistoricalEntries + +| Type | Default value | +| ----- | ------------- | +| int64 | 10000 | + +`HistoricalEntries` is the number of historical info entries to persist in store (see the staking module parameter with the same name for details). +`HistoricalEntries` is needed since the consumer module acts as a staking module on the consumer chain. + +### UnbondingPeriod + +| Type | Default value | +| ------------- | ------------------ | +| time.Duration | 1209600s (2 weeks) | + +`UnbondingPeriod` is the unbonding period on the consumer chain. +It is recommended that every consumer chain set and unbonding period shorter than provider unbonding period, e.g., one week shorter. + +### SoftOptOutThreshold + +`SoftOptOutThreshold` is deprecated. + +### RewardDenoms + +| Type | Default value | +| -------- | ------------- | +| []string | []string{} | + +`RewardDenoms` are the denominations which are allowed to be sent to the provider as ICS rewards. + +### ProviderRewardDenoms + +| Type | Default value | +| -------- | ------------- | +| []string | []string{} | + +`ProviderRewardDenoms` are the denominations coming from the provider which are allowed to be used as ICS rewards. e.g. "uatom". + +### RetryDelayPeriod + +| Type | Default value | +| ------------- | -------------- | +| time.Duration | 3600s (1 hour) | + +`RetryDelayPeriod` is the period at which the consumer retries to send a `SlashPacket` that was rejected by the provider. +For more details, see [ADR-008](../adrs/adr-008-throttle-retries.md). + +## Client + +### CLI + +A user can interact with the `consumer` module using the CLI. + +#### Query + +The `query` commands allow users to query `consumer` state. + +```bash +interchain-security-cd query ccvconsumer --help +``` + +##### Next Fee Distribution + +The `next-fee-distribution` command allows to query next fee distribution data. + +```bash +interchain-security-cd query ccvconsumer next-fee-distribution [flags] +``` + +
+ Example + +```bash +interchain-security-cd query ccvconsumer next-fee-distribution +``` + +Output: + +```bash +data: + currentHeight: "967" + distribution_fraction: "0.75" + lastHeight: "960" + nextHeight: "980" + toConsumer: "" + toProvider: "" + total: "" +``` + +
+ +##### Provider Info + +The `provider-info` command allows to query provider info. + +```bash +interchain-security-cd query ccvconsumer provider-info [flags] +``` + +
+ Example + +```bash +interchain-security-cd query ccvconsumer provider-info +``` + +Output: + +```bash +consumer: + chainID: pion-1 + channelID: channel-0 + clientID: 07-tendermint-0 + connectionID: connection-0 +provider: + chainID: provider + channelID: channel-0 + clientID: 07-tendermint-0 + connectionID: connection-0 +``` + +
+ +##### Throttle State + +The `throttle-state` command allows to query on-chain state relevant with slash packet throttling. + +```bash +interchain-security-cd query ccvconsumer throttle-state [flags] +``` + +
+ Example + +```bash +interchain-security-cd query ccvconsumer throttle-state +``` + +Output: + +```bash +packet_data_queue: +- slashPacketData: + infraction: INFRACTION_DOWNTIME + validator: + address: mb06cu8SzQJOdYSzrJAK43Q8at8= + power: "500" + valset_update_id: "48" + type: CONSUMER_PACKET_TYPE_SLASH +slash_record: + send_time: "2024-10-02T07:58:24.405645924Z" + waiting_on_reply: true +``` + +
+ +##### Params + +The `params` command allows to query consumer module parameters. + +```bash +interchain-security-cd query ccvconsumer params [flags] +``` + +
+ Example + +```bash +interchain-security-cd query ccvconsumer params +``` + +Output: + +```bash +params: + blocks_per_distribution_transmission: "1000" + ccv_timeout_period: 2419200s + consumer_id: "0" + consumer_redistribution_fraction: "0.75" + distribution_transmission_channel: channel-1 + enabled: true + historical_entries: "10000" + provider_fee_pool_addr_str: cosmos1ap0mh6xzfn8943urr84q6ae7zfnar48am2erhd + provider_reward_denoms: [] + retry_delay_period: 3600s + reward_denoms: [] + soft_opt_out_threshold: "0" + transfer_timeout_period: 3600s + unbonding_period: 1209600s +``` + +
+ +### gRPC + +A user can query the `consumer` module using gRPC endpoints. + +#### Next Fee Distribution + +The `QueryNextFeeDistribution` endpoint queries next fee distribution data. + +```bash +interchain_security.ccv.consumer.v1.Query/QueryNextFeeDistribution +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.consumer.v1.Query/QueryNextFeeDistribution +``` + +Output: + +```json +{ + "data": { + "currentHeight": "402", + "lastHeight": "400", + "nextHeight": "420", + "distributionFraction": "0.75" + } +} +``` + +
+ +#### Provider Info + +The `QueryProviderInfo` endpoint queries provider info. + +```bash +interchain_security.ccv.consumer.v1.Query/QueryProviderInfo +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.consumer.v1.Query/QueryProviderInfo +``` + +Output: + +```json +{ + "consumer": { + "chainID": "pion-1", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + }, + "provider": { + "chainID": "provider", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + } +} +``` + +
+ +#### Throttle State + +The `QueryThrottleState` endpoint queries on-chain state relevant with slash packet throttling. + +```bash +interchain_security.ccv.consumer.v1.Query/QueryThrottleState +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.consumer.v1.Query/QueryThrottleState +``` + +Output: + +```json +{ + "slashRecord": { + "waitingOnReply": true, + "sendTime": "2024-10-02T07:58:24.405645924Z" + }, + "packetDataQueue": [ + { + "type": "CONSUMER_PACKET_TYPE_SLASH", + "slashPacketData": { + "validator": { + "address": "mb06cu8SzQJOdYSzrJAK43Q8at8=", + "power": "500" + }, + "valsetUpdateId": "48", + "infraction": "INFRACTION_DOWNTIME" + } + } + ] +} +``` + +
+ +#### Params + +The `QueryParams` endpoint queries consumer module parameters. + +```bash +interchain_security.ccv.consumer.v1.Query/QueryParams +``` + +
+ Example + +```bash +grpcurl -plaintext localhost:9090 interchain_security.ccv.consumer.v1.Query/QueryParams +``` + +Output: + +```json +{ + "params": { + "enabled": true, + "blocksPerDistributionTransmission": "1000", + "distributionTransmissionChannel": "channel-1", + "providerFeePoolAddrStr": "cosmos1ap0mh6xzfn8943urr84q6ae7zfnar48am2erhd", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "historicalEntries": "10000", + "unbondingPeriod": "1209600s", + "softOptOutThreshold": "0", + "retryDelayPeriod": "3600s", + "consumerId": "0" + } +} +``` + +
+ +### REST + +A user can query the `consumer` module using REST endpoints. + +#### Next Fee Distribution + +The `next-fee-distribution` endpoint queries next fee distribution data. + +```bash +/interchain_security/ccv/consumer/next-fee-distribution +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/consumer/next-fee-distribution +``` + +Output: + +```json +{ + "data": { + "currentHeight": "402", + "lastHeight": "400", + "nextHeight": "420", + "distributionFraction": "0.75" + } +} +``` + +
+ +#### Provider Info + +The `QueryProviderInfo` endpoint queries provider info. + +```bash +/interchain_security/ccv/consumer/provider-info +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/consumer/provider-info +``` + +Output: + +```json +{ + "consumer": { + "chainID": "pion-1", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + }, + "provider": { + "chainID": "provider", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + } +} +``` + +
+ +#### Throttle State + +The `throttle_state` endpoint queries on-chain state relevant with slash packet throttling. + + +```bash +/interchain_security/ccv/consumer/throttle_state +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/consumer/throttle_state +``` + +Output: + +```json +{ + "consumer": { + "chainID": "pion-1", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + }, + "provider": { + "chainID": "provider", + "clientID": "07-tendermint-0", + "connectionID": "connection-0", + "channelID": "channel-0" + } +} +``` + +
+ +#### Params + +The `params` endpoint queries consumer module parameters. + +```bash +/interchain_security/ccv/consumer/params +``` + +
+ Example + +```bash +curl http://localhost:1317/interchain_security/ccv/consumer/params +``` + +Output: + +```json +{ + "params": { + "enabled": true, + "blocksPerDistributionTransmission": "1000", + "distributionTransmissionChannel": "channel-1", + "providerFeePoolAddrStr": "cosmos1ap0mh6xzfn8943urr84q6ae7zfnar48am2erhd", + "ccvTimeoutPeriod": "2419200s", + "transferTimeoutPeriod": "3600s", + "consumerRedistributionFraction": "0.75", + "historicalEntries": "10000", + "unbondingPeriod": "1209600s", + "softOptOutThreshold": "0", + "retryDelayPeriod": "3600s", + "consumerId": "0" + } +} +``` + +
diff --git a/docs/docs/features/democracy-modules.md b/docs/docs/build/modules/04-democracy.md similarity index 54% rename from docs/docs/features/democracy-modules.md rename to docs/docs/build/modules/04-democracy.md index 1566f76c9c..01f6fdef26 100644 --- a/docs/docs/features/democracy-modules.md +++ b/docs/docs/build/modules/04-democracy.md @@ -2,11 +2,9 @@ sidebar_position: 4 --- -# Democracy modules +# x/ccv/democracy -This section is relevant for chains transitioning from a standalone chain and new consumer chains that require some functionality from the `x/staking` module. - -The democracy modules comprise `x/staking`, `x/distribution` and `x/governance` with overrides and extensions required for normal operation when participating in ICS. +The democracy modules comprise `x/ccv/democracy/staking`, `x/ccv/democracy/distribution` and `x/ccv/democracy/governance` with overrides and extensions required for normal operation when participating in ICS. The modules are plug-and-play and only require small wiring changes to be enabled. @@ -14,7 +12,7 @@ For a full integration check the `consumer-democracy` [example app](https://gith ## Staking -The democracy staking module allows the cosmos-sdk `x/staking` module to be used alongside the interchain security `consumer` module. +The `x/ccv/democracy/staking` module allows the cosmos-sdk `x/staking` module to be used alongside the interchain security `consumer` module. The module uses overrides that allow the full `x/staking` functionality with one notable difference - the staking module will no longer be used to provide the validator set to the consensus engine. @@ -25,14 +23,14 @@ The validator set coming from the provider chain does not need to participate in #### Governators (aka. Governors) -Validators registered with the `x/staking` module become __Governators__. +Validators registered with the `x/ccv/democracy/staking` module become __Governators__. Unlike validators, governators are not required to run any chain infrastructure since they are not signing any blocks. However, governators retain a subset of the validator properties: - new governators can be created (via `MsgCreateValidator`) - governators can accept delegations - governators can vote on governance proposals (with their self stake and delegations) -- governators earn block rewards -- the block rewards kept on the consumer (see the [ConsumerRedistributionFraction param](../introduction/params.md#consumerredistributionfraction)) are distributed to all governators and their delegators. +- governators earn block rewards -- the block rewards kept on the consumer (see the [ConsumerRedistributionFraction param](../build/modules/03-consumer.md#consumerredistributionfraction)) are distributed to all governators and their delegators. With these changes, governators can become community advocates that can specialize in chain governance and they get rewarded for their participation the same way the validators do. Additionally, governators can choose to provide additional infrastructure such as RPC/API access points, archive nodes, indexers and similar software. @@ -51,20 +49,20 @@ The `x/ccv/democracy/staking` module provides these `x/staking` overrides: // however, it returns no validator updates as validators are tracked via the // consumer chain's x/cvv/consumer module and so this module is not responsible for returning the initial validator set. func (am AppModule) InitGenesis(ctx sdk.Context, cdc codec.JSONCodec, data json.RawMessage) []abci.ValidatorUpdate { - var genesisState types.GenesisState + var genesisState types.GenesisState - cdc.MustUnmarshalJSON(data, &genesisState) - _ = am.keeper.InitGenesis(ctx, &genesisState) // run staking InitGenesis + cdc.MustUnmarshalJSON(data, &genesisState) + _ = am.keeper.InitGenesis(ctx, &genesisState) // run staking InitGenesis - return []abci.ValidatorUpdate{} // do not return validator updates + return []abci.ValidatorUpdate{} // do not return validator updates } // EndBlock delegates the EndBlock call to the underlying x/staking module. // However, no validator updates are returned as validators are tracked via the // consumer chain's x/cvv/consumer module. func (am AppModule) EndBlock(ctx sdk.Context, _ abci.RequestEndBlock) []abci.ValidatorUpdate { - _ = am.keeper.BlockValidatorUpdates(ctx) // perform staking BlockValidatorUpdates - return []abci.ValidatorUpdate{} // do not return validator updates + _ = am.keeper.BlockValidatorUpdates(ctx) // perform staking BlockValidatorUpdates + return []abci.ValidatorUpdate{} // do not return validator updates } ``` @@ -80,9 +78,9 @@ If some of your modules are returning validator updates please disable them whil ```diff func (am AppModule) InitGenesis(ctx sdk.Context, cdc codec.JSONCodec, data json.RawMessage) []abci.ValidatorUpdate { - var genesisState types.GenesisState + var genesisState types.GenesisState - cdc.MustUnmarshalJSON(data, &genesisState) + cdc.MustUnmarshalJSON(data, &genesisState) - return am.keeper.InitGenesis(ctx, &genesisState) + _ = am.keeper.InitGenesis(ctx, &genesisState) // run InitGenesis but drop the result + return []abci.ValidatorUpdate{} // return empty validator updates @@ -108,11 +106,11 @@ import ( var ( // replace the staking.AppModuleBasic - ModuleBasics = module.NewBasicManager( - auth.AppModuleBasic{}, - genutil.NewAppModuleBasic(genutiltypes.DefaultMessageValidator), - bank.AppModuleBasic{}, - capability.AppModuleBasic{}, + ModuleBasics = module.NewBasicManager( + auth.AppModuleBasic{}, + genutil.NewAppModuleBasic(genutiltypes.DefaultMessageValidator), + bank.AppModuleBasic{}, + capability.AppModuleBasic{}, - sdkstaking.AppModuleBasic{}, + ccvstaking.AppModuleBasic{}, // replace sdkstaking ... @@ -123,35 +121,35 @@ var ( func NewApp(...) { ... - // use sdk StakingKeepeer - app.StakingKeeper = stakingkeeper.NewKeeper( - appCodec, - keys[stakingtypes.StoreKey], - app.AccountKeeper, - app.BankKeeper, - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) - - app.MintKeeper = mintkeeper.NewKeeper( - appCodec, - keys[minttypes.StoreKey], - app.StakingKeeper, - app.AccountKeeper, - app.BankKeeper, - authtypes.FeeCollectorName, - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) - - // no changes required for the distribution keeper + // use sdk StakingKeepeer + app.StakingKeeper = stakingkeeper.NewKeeper( + appCodec, + keys[stakingtypes.StoreKey], + app.AccountKeeper, + app.BankKeeper, + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) + + app.MintKeeper = mintkeeper.NewKeeper( + appCodec, + keys[minttypes.StoreKey], + app.StakingKeeper, + app.AccountKeeper, + app.BankKeeper, + authtypes.FeeCollectorName, + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) + + // no changes required for the distribution keeper app.DistrKeeper = distrkeeper.NewKeeper( - appCodec, - keys[distrtypes.StoreKey], - app.AccountKeeper, - app.BankKeeper, - app.StakingKeeper, // keep StakingKeeper! - authtypes.FeeCollectorName, - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) + appCodec, + keys[distrtypes.StoreKey], + app.AccountKeeper, + app.BankKeeper, + app.StakingKeeper, // keep StakingKeeper! + authtypes.FeeCollectorName, + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) + // pre-initialize ConsumerKeeper to satsfy ibckeeper.NewKeeper + app.ConsumerKeeper = consumerkeeper.NewNonZeroKeeper( @@ -193,28 +191,28 @@ func NewApp(...) { - // change the slashing keeper dependency - app.SlashingKeeper = slashingkeeper.NewKeeper( - appCodec, - legacyAmino, - keys[slashingtypes.StoreKey], + // change the slashing keeper dependency + app.SlashingKeeper = slashingkeeper.NewKeeper( + appCodec, + legacyAmino, + keys[slashingtypes.StoreKey], - app.StakingKeeper, + &app.ConsumerKeeper, // ConsumerKeeper implements StakingKeeper interface - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) - // register slashing module StakingHooks to the consumer keeper + // register slashing module StakingHooks to the consumer keeper + app.ConsumerKeeper = *app.ConsumerKeeper.SetHooks(app.SlashingKeeper.Hooks()) + consumerModule := consumer.NewAppModule(app.ConsumerKeeper, app.GetSubspace(consumertypes.ModuleName)) - // register the module with module manager + // register the module with module manager // replace the x/staking module - app.MM = module.NewManager( - ... + app.MM = module.NewManager( + ... - sdkstaking.NewAppModule(appCodec, &app.StakingKeeper, app.AccountKeeper, app.BankKeeper, app.GetSubspace(stakingtypes.ModuleName)), + ccvstaking.NewAppModule(appCodec, *app.StakingKeeper, app.AccountKeeper, app.BankKeeper, app.GetSubspace(stakingtypes.ModuleName)), - ... - ) + ... + ) } ``` @@ -234,107 +232,107 @@ Add new `AnteHandler` to your `app`. package ante import ( - "fmt" + "fmt" - sdk "github.com/cosmos/cosmos-sdk/types" - govv1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1" - govv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" - ibctransfertypes "github.com/cosmos/ibc-go/v7/modules/apps/transfer/types" + sdk "github.com/cosmos/cosmos-sdk/types" + govv1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1" + govv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" + ibctransfertypes "github.com/cosmos/ibc-go/v7/modules/apps/transfer/types" - "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" - "github.com/cosmos/cosmos-sdk/x/params/types/proposal" + "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" + "github.com/cosmos/cosmos-sdk/x/params/types/proposal" ) type ForbiddenProposalsDecorator struct { - isLegacyProposalWhitelisted func(govv1beta1.Content) bool - isModuleWhiteList func(string) bool + isLegacyProposalWhitelisted func(govv1beta1.Content) bool + isModuleWhiteList func(string) bool } func NewForbiddenProposalsDecorator( - whiteListFn func(govv1beta1.Content) bool, - isModuleWhiteList func(string) bool, + whiteListFn func(govv1beta1.Content) bool, + isModuleWhiteList func(string) bool, ) ForbiddenProposalsDecorator { - return ForbiddenProposalsDecorator{ - isLegacyProposalWhitelisted: whiteListFn, - isModuleWhiteList: isModuleWhiteList, - } + return ForbiddenProposalsDecorator{ + isLegacyProposalWhitelisted: whiteListFn, + isModuleWhiteList: isModuleWhiteList, + } } func (decorator ForbiddenProposalsDecorator) AnteHandle(ctx sdk.Context, tx sdk.Tx, simulate bool, next sdk.AnteHandler) (newCtx sdk.Context, err error) { - currHeight := ctx.BlockHeight() - - for _, msg := range tx.GetMsgs() { - // if the message is MsgSubmitProposal, check if proposal is whitelisted - submitProposalMgs, ok := msg.(*govv1.MsgSubmitProposal) - if !ok { - continue - } - - messages := submitProposalMgs.GetMessages() - for _, message := range messages { - if sdkMsg, isLegacyProposal := message.GetCachedValue().(*govv1.MsgExecLegacyContent); isLegacyProposal { - // legacy gov proposal content - content, err := govv1.LegacyContentFromMessage(sdkMsg) - if err != nil { - return ctx, fmt.Errorf("tx contains invalid LegacyContent") - } - if !decorator.isLegacyProposalWhitelisted(content) { - return ctx, fmt.Errorf("tx contains unsupported proposal message types at height %d", currHeight) - } - continue - } - // not legacy gov proposal content and not whitelisted - if !decorator.isModuleWhiteList(message.TypeUrl) { - return ctx, fmt.Errorf("tx contains unsupported proposal message types at height %d", currHeight) - } - } - } - - return next(ctx, tx, simulate) + currHeight := ctx.BlockHeight() + + for _, msg := range tx.GetMsgs() { + // if the message is MsgSubmitProposal, check if proposal is whitelisted + submitProposalMgs, ok := msg.(*govv1.MsgSubmitProposal) + if !ok { + continue + } + + messages := submitProposalMgs.GetMessages() + for _, message := range messages { + if sdkMsg, isLegacyProposal := message.GetCachedValue().(*govv1.MsgExecLegacyContent); isLegacyProposal { + // legacy gov proposal content + content, err := govv1.LegacyContentFromMessage(sdkMsg) + if err != nil { + return ctx, fmt.Errorf("tx contains invalid LegacyContent") + } + if !decorator.isLegacyProposalWhitelisted(content) { + return ctx, fmt.Errorf("tx contains unsupported proposal message types at height %d", currHeight) + } + continue + } + // not legacy gov proposal content and not whitelisted + if !decorator.isModuleWhiteList(message.TypeUrl) { + return ctx, fmt.Errorf("tx contains unsupported proposal message types at height %d", currHeight) + } + } + } + + return next(ctx, tx, simulate) } func IsProposalWhitelisted(content v1beta1.Content) bool { - switch c := content.(type) { - case *proposal.ParameterChangeProposal: - return isLegacyParamChangeWhitelisted(c.Changes) + switch c := content.(type) { + case *proposal.ParameterChangeProposal: + return isLegacyParamChangeWhitelisted(c.Changes) - default: - return false - } + default: + return false + } } func isLegacyParamChangeWhitelisted(paramChanges []proposal.ParamChange) bool { - for _, paramChange := range paramChanges { - _, found := LegacyWhitelistedParams[legacyParamChangeKey{Subspace: paramChange.Subspace, Key: paramChange.Key}] - if !found { - return false - } - } - return true + for _, paramChange := range paramChanges { + _, found := LegacyWhitelistedParams[legacyParamChangeKey{Subspace: paramChange.Subspace, Key: paramChange.Key}] + if !found { + return false + } + } + return true } type legacyParamChangeKey struct { - Subspace, Key string + Subspace, Key string } // Legacy params can be whitelisted var LegacyWhitelistedParams = map[legacyParamChangeKey]struct{}{ - {Subspace: ibctransfertypes.ModuleName, Key: "SendEnabled"}: {}, - {Subspace: ibctransfertypes.ModuleName, Key: "ReceiveEnabled"}: {}, + {Subspace: ibctransfertypes.ModuleName, Key: "SendEnabled"}: {}, + {Subspace: ibctransfertypes.ModuleName, Key: "ReceiveEnabled"}: {}, } // New proposal types can be whitelisted var WhiteListModule = map[string]struct{}{ - "/cosmos.gov.v1.MsgUpdateParams": {}, - "/cosmos.bank.v1beta1.MsgUpdateParams": {}, - "/cosmos.staking.v1beta1.MsgUpdateParams": {}, - "/cosmos.distribution.v1beta1.MsgUpdateParams": {}, - "/cosmos.mint.v1beta1.MsgUpdateParams": {}, + "/cosmos.gov.v1.MsgUpdateParams": {}, + "/cosmos.bank.v1beta1.MsgUpdateParams": {}, + "/cosmos.staking.v1beta1.MsgUpdateParams": {}, + "/cosmos.distribution.v1beta1.MsgUpdateParams": {}, + "/cosmos.mint.v1beta1.MsgUpdateParams": {}, } func IsModuleWhiteList(typeUrl string) bool { - _, found := WhiteListModule[typeUrl] - return found + _, found := WhiteListModule[typeUrl] + return found } ``` @@ -345,7 +343,7 @@ Add the `AnteHandler` to the list of supported antehandlers: package app import ( - ... + ... + democracyante "github.com/cosmos/interchain-security/v4/app/consumer-democracy/ante" + consumerante "github.com/cosmos/interchain-security/v4/app/consumer/ante" @@ -353,24 +351,24 @@ import ( ) type HandlerOptions struct { - ante.HandlerOptions + ante.HandlerOptions - IBCKeeper *ibckeeper.Keeper + IBCKeeper *ibckeeper.Keeper + ConsumerKeeper ibcconsumerkeeper.Keeper } func NewAnteHandler(options HandlerOptions) (sdk.AnteHandler, error) { - .... + .... - anteDecorators := []sdk.AnteDecorator{ + anteDecorators := []sdk.AnteDecorator{ ... + consumerante.NewMsgFilterDecorator(options.ConsumerKeeper), + consumerante.NewDisabledModulesDecorator("/cosmos.evidence", "/cosmos.slashing"), + democracyante.NewForbiddenProposalsDecorator(IsProposalWhitelisted, IsModuleWhiteList), - ... - } + ... + } - return sdk.ChainAnteDecorators(anteDecorators...), nil + return sdk.ChainAnteDecorators(anteDecorators...), nil } ``` @@ -382,9 +380,9 @@ package app import ( ... sdkgov "github.com/cosmos/cosmos-sdk/x/gov" - govkeeper "github.com/cosmos/cosmos-sdk/x/gov/keeper" - govtypes "github.com/cosmos/cosmos-sdk/x/gov/types" - govv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" + govkeeper "github.com/cosmos/cosmos-sdk/x/gov/keeper" + govtypes "github.com/cosmos/cosmos-sdk/x/gov/types" + govv1beta1 "github.com/cosmos/cosmos-sdk/x/gov/types/v1beta1" + ccvgov "github.com/cosmos/interchain-security/v4/x/ccv/democracy/governance" ) @@ -392,54 +390,54 @@ import ( var ( // use sdk governance module - ModuleBasics = module.NewBasicManager( - ... - sdkgov.NewAppModuleBasic( - []govclient.ProposalHandler{ - paramsclient.ProposalHandler, - upgradeclient.LegacyProposalHandler, - upgradeclient.LegacyCancelProposalHandler, - }, - ), + ModuleBasics = module.NewBasicManager( + ... + sdkgov.NewAppModuleBasic( + []govclient.ProposalHandler{ + paramsclient.ProposalHandler, + upgradeclient.LegacyProposalHandler, + upgradeclient.LegacyCancelProposalHandler, + }, + ), ) ) func NewApp(...) { // retain sdk gov router and keeper registrations - sdkgovRouter := govv1beta1.NewRouter() - sdkgovRouter. - AddRoute(govtypes.RouterKey, govv1beta1.ProposalHandler). - AddRoute(paramproposal.RouterKey, params.NewParamChangeProposalHandler(app.ParamsKeeper)). - AddRoute(upgradetypes.RouterKey, upgrade.NewSoftwareUpgradeProposalHandler(&app.UpgradeKeeper)) - govConfig := govtypes.DefaultConfig() - - app.GovKeeper = *govkeeper.NewKeeper( - appCodec, - keys[govtypes.StoreKey], - app.AccountKeeper, - app.BankKeeper, - app.StakingKeeper, - app.MsgServiceRouter(), - govConfig, - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) - - app.GovKeeper.SetLegacyRouter(sdkgovRouter) + sdkgovRouter := govv1beta1.NewRouter() + sdkgovRouter. + AddRoute(govtypes.RouterKey, govv1beta1.ProposalHandler). + AddRoute(paramproposal.RouterKey, params.NewParamChangeProposalHandler(app.ParamsKeeper)). + AddRoute(upgradetypes.RouterKey, upgrade.NewSoftwareUpgradeProposalHandler(&app.UpgradeKeeper)) + govConfig := govtypes.DefaultConfig() + + app.GovKeeper = *govkeeper.NewKeeper( + appCodec, + keys[govtypes.StoreKey], + app.AccountKeeper, + app.BankKeeper, + app.StakingKeeper, + app.MsgServiceRouter(), + govConfig, + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) + + app.GovKeeper.SetLegacyRouter(sdkgovRouter) // register the module with module manager // replace the x/gov module - app.MM = module.NewManager( + app.MM = module.NewManager( - sdkgov.NewAppModule(appCodec, app.GovKeeper, app.AccountKeeper, app.BankKeeper, IsProposalWhitelisted, app.GetSubspace(govtypes.ModuleName), IsModuleWhiteList), + ccvgov.NewAppModule(appCodec, app.GovKeeper, app.AccountKeeper, app.BankKeeper, IsProposalWhitelisted, app.GetSubspace(govtypes.ModuleName), IsModuleWhiteList), - ... + ... ) } ``` ## Distribution -The `democracy/distribution` module allows the consumer chain to send rewards to the provider chain while retaining the logic of the `x/distribution` module for internal reward distribution to governators and their delegators. +The `x/ccv/democracy/distribution` module allows the consumer chain to send rewards to the provider chain while retaining the logic of the `x/distribution` module for internal reward distribution to governators and their delegators. ### How it works @@ -456,7 +454,7 @@ Change the wiring in `app.go` import ( ... distrkeeper "github.com/cosmos/cosmos-sdk/x/distribution/keeper" - distrtypes "github.com/cosmos/cosmos-sdk/x/distribution/types" + distrtypes "github.com/cosmos/cosmos-sdk/x/distribution/types" sdkdistr "github.com/cosmos/cosmos-sdk/x/distribution" + ccvdistr "github.com/cosmos/interchain-security/v4/x/ccv/democracy/distribution" @@ -464,13 +462,13 @@ import ( var ( // replace sdk distribution AppModuleBasic - ModuleBasics = module.NewBasicManager( - auth.AppModuleBasic{}, - genutil.NewAppModuleBasic(genutiltypes.DefaultMessageValidator), - bank.AppModuleBasic{}, - capability.AppModuleBasic{}, - ccvstaking.AppModuleBasic{}, // make sure you first swap the staking keeper - mint.AppModuleBasic{}, + ModuleBasics = module.NewBasicManager( + auth.AppModuleBasic{}, + genutil.NewAppModuleBasic(genutiltypes.DefaultMessageValidator), + bank.AppModuleBasic{}, + capability.AppModuleBasic{}, + ccvstaking.AppModuleBasic{}, // make sure you first swap the staking keeper + mint.AppModuleBasic{}, - sdkdistr.AppModuleBasic{}, + ccvdistr.AppModuleBasic{}, ) @@ -479,24 +477,24 @@ var ( func NewApp(...) { .... - app.DistrKeeper = distrkeeper.NewKeeper( - appCodec, - keys[distrtypes.StoreKey], - app.AccountKeeper, - app.BankKeeper, - app.StakingKeeper, // connect to sdk StakingKeeper - consumertypes.ConsumerRedistributeName, - authtypes.NewModuleAddress(govtypes.ModuleName).String(), - ) + app.DistrKeeper = distrkeeper.NewKeeper( + appCodec, + keys[distrtypes.StoreKey], + app.AccountKeeper, + app.BankKeeper, + app.StakingKeeper, // connect to sdk StakingKeeper + consumertypes.ConsumerRedistributeName, + authtypes.NewModuleAddress(govtypes.ModuleName).String(), + ) // register with the module manager - app.MM = module.NewManager( - ... + app.MM = module.NewManager( + ... - sdkdistr.NewAppModule(appCodec, app.DistrKeeper, app.AccountKeeper, app.BankKeeper, *app.StakingKeeper, authtypes.FeeCollectorName, app.GetSubspace(distrtypes.ModuleName)), + ccvdistr.NewAppModule(appCodec, app.DistrKeeper, app.AccountKeeper, app.BankKeeper, *app.StakingKeeper, authtypes.FeeCollectorName, app.GetSubspace(distrtypes.ModuleName)), - ccvstaking.NewAppModule(appCodec, *app.StakingKeeper, app.AccountKeeper, app.BankKeeper, app.GetSubspace(stakingtypes.ModuleName)), - ... + ccvstaking.NewAppModule(appCodec, *app.StakingKeeper, app.AccountKeeper, app.BankKeeper, app.GetSubspace(stakingtypes.ModuleName)), + ... ) } ``` diff --git a/docs/docs/build/modules/_category_.json b/docs/docs/build/modules/_category_.json new file mode 100644 index 0000000000..2536f423ff --- /dev/null +++ b/docs/docs/build/modules/_category_.json @@ -0,0 +1,4 @@ +{ + "label": "Modules", + "position": 1 +} diff --git a/docs/docs/consumer-development/consumer-genesis-transformation.md b/docs/docs/consumer-development/consumer-genesis-transformation.md index 3b253fb99d..d77e8ec559 100644 --- a/docs/docs/consumer-development/consumer-genesis-transformation.md +++ b/docs/docs/consumer-development/consumer-genesis-transformation.md @@ -7,12 +7,12 @@ sidebar_position: 6 Preparing a consumer chain for onboarding requires some information explaining how to run your chain. This includes a genesis file with CCV data where the CCV data is exported from the provider chain and added to the consumers genesis file (for more details check the documentation on [Onboarding](./onboarding.md) and [Changeover](./changeover-procedure.md)). In case that the provider chain is running an older version of the InterChainSecurity (ICS) module than the consumer chain - or vice versa - the exported CCV data might need to be transformed to the format supported by the ICS implementation run on the consumer chain. This is the case if the consumer chain runs version 4 of ICS or later and the provider is running version 3 or older of the ICS module. -Check the [compatibility notes](../../../RELEASES.md#backwards-compatibility) for known incompatibilities between provider and consumer versions and indications if a consumer genesis transformation is required. +Check the [compatibility notes](https://github.com/cosmos/interchain-security/blob/main/RELEASES.md#backwards-compatibility) for known incompatibilities between provider and consumer versions and indications if a consumer genesis transformation is required. To transform such CCV data follow the instructions below ## 1. Prerequisite -- used provider and consumer versions require transformation step as indicated in in the [compatibility notes](../../../RELEASES.md#backwards-compatibility) +- used provider and consumer versions require transformation step as indicated in in the [compatibility notes](https://github.com/cosmos/interchain-security/blob/main/RELEASES.md#backwards-compatibility) - interchain-security-cd application supports the versions used by the consumer and provider ## 2. Export the CCV data diff --git a/docs/docs/features/_category_.json b/docs/docs/features/_category_.json index 4aaeb07417..e456a6a85c 100644 --- a/docs/docs/features/_category_.json +++ b/docs/docs/features/_category_.json @@ -1,4 +1,4 @@ { "label": "Features", - "position": 3 + "position": 2 } diff --git a/docs/docs/features/reward-distribution.md b/docs/docs/features/reward-distribution.md index 593330939b..e91bbe5818 100644 --- a/docs/docs/features/reward-distribution.md +++ b/docs/docs/features/reward-distribution.md @@ -8,8 +8,8 @@ sidebar_position: 2 Sending and distributing rewards from consumer chains to the provider chain is handled by the [Reward Distribution sub-protocol](https://github.com/cosmos/ibc/blob/main/spec/app/ics-028-cross-chain-validation/overview_and_basic_concepts.md#reward-distribution). Consumer chains have the option of sharing _a portion of_ their block rewards (inflation tokens and fees) with the provider chain as ICS rewards. -These rewards are periodically sent from the consumer to the provider according to [consumer chain parameters](../introduction/params.md#reward-distribution-parameters) using an IBC transfer channel. -This channel is created during consumer chain initialization, unless it is provided when creating a new consumer chain (see the [DistributionTransmissionChannel param](../introduction/params.md#distributiontransmissionchannel)). +These rewards are periodically sent from the consumer to the provider according to [consumer chain parameters](../build/modules/03-consumer.md#parameters) using an IBC transfer channel. +This channel is created during consumer chain initialization, unless it is provided when creating a new consumer chain (see the [DistributionTransmissionChannel param](../build/modules/03-consumer.md#distributiontransmissionchannel)). Providing an IBC transfer channel enables a consumer chain to re-use one of the existing channels to the provider for consumer chain rewards distribution. This will preserve the `ibc denom` that may already be in use. diff --git a/docs/docs/frequently-asked-questions.md b/docs/docs/frequently-asked-questions.md index 9619772445..417353bc03 100644 --- a/docs/docs/frequently-asked-questions.md +++ b/docs/docs/frequently-asked-questions.md @@ -1,5 +1,5 @@ --- -sidebar_position: 6 +sidebar_position: 8 title: "Frequently Asked Questions" slug: /faq --- @@ -49,7 +49,7 @@ In case the provider chain halts or experiences difficulties, the consumer chain As the validators run separate infrastructure on these networks, **_the provider chain liveness does not impact the liveness of consumer chains_**. Every consumer chain communicates with the provider chain via a CCV channel -- an IBC ordered channel. -If any of the packets sent over the CCV channel timeout (see the [CCVTimeoutPeriod param](./introduction/params.md#ccvtimeoutperiod)), then the channel is closed and, consequently, the consumer chain transitions to a Proof of Authority (PoA) chain. +If any of the packets sent over the CCV channel timeout (see the [CCVTimeoutPeriod param](./build/modules/03-consumer.md#ccvtimeoutperiod)), then the channel is closed and, consequently, the consumer chain transitions to a Proof of Authority (PoA) chain. This means that the validator set on the consumer will no longer be updated with information from the provider. ### What happens to provider if any of the consumers are down? @@ -62,13 +62,13 @@ The ICS protocol is concerned only with validator set management, and the only c As any other Cosmos SDK chains, **_consumer chains can issue their own token_** and manage inflation parameters. Note that the ICS protocol does not impact the transaction fee system on the consumer chains. This means consumer chains can use any token (including their own token) to pay gas fees. -For more details, see the [democracy modules](./features/democracy-modules.md#tokenomics). +For more details, see the [democracy modules](./build/modules/04-democracy.md#tokenomics). ### Can consumer chains have their own governance? Yes. ICS allows consumer chains to **_separate governance from block production_**. Validator operators (with their stake locked on the provider) are responsible for block production, while _representatives_ (aka governators, governors) are responsible for on-chain governance. -For more details, see the [democracy modules](./features/democracy-modules.md). +For more details, see the [democracy modules](./build/modules/04-democracy.md). ### Can a consumer chain modify its power shaping parameters? @@ -109,7 +109,7 @@ Also note that validators can assign consensus keys before a consumer chain is l ### What are the benefits for validators running consumer chains? -The consumer chains sends a portion of its block rewards (e.g., transaction fees and inflation) to the provider chain as defined by the [ConsumerRedistributionFraction param](./introduction/params.md#consumerredistributionfraction). +The consumer chains sends a portion of its block rewards (e.g., transaction fees and inflation) to the provider chain as defined by the [ConsumerRedistributionFraction param](./build/modules/03-consumer.md#consumerredistributionfraction). These rewards are sent periodically to the provider (via IBC transfers), where they are distributed **ONLY** to the _opted in_ validators and their delegators. For more details, see the [Reward Distribution feature](./features/reward-distribution.md). ### Can validators set per consumer chain commission rates? diff --git a/docs/docs/index.mdx b/docs/docs/index.mdx index 576da1feee..f0cc59b104 100644 --- a/docs/docs/index.mdx +++ b/docs/docs/index.mdx @@ -8,11 +8,11 @@ import homeCards from '/src/components/Cards/card-data/home-cards'; # Cosmos Hub - Interchain Security -Welcome to the official documentation for the Cosmos Hub's Interchain Security - a platform for launching Cosmos-SDK chains. +Welcome to the official documentation for the Cosmos Hub's Interchain Security (ICS) - a platform for launching Cosmos-SDK chains. *The Cosmos Hub is the best place to launch a chain.* -With the recent launch of [Partial Set Security](/features/partial-set-security) (PSS), it's easier than ever to leverage the Cosmos Hub validators, stake, and community for your chain. PSS provides powerful new capabilities for permissionlessly launching your chain and [shaping your validator set](/features/power-shaping). +With the recent launch of the [Partial Set Security](/features/partial-set-security) and [Permissionless](/features/permissionless) features, it's easier than ever to leverage the Cosmos Hub validators, stake, and community for your chain. ICS provides powerful capabilities for permissionlessly launching your chain and [shaping your validator set](/features/power-shaping). Here you can find information about Interchain Security, consumer chain development and instructions for validator onboarding. diff --git a/docs/docs/integrators/_category_.json b/docs/docs/integrators/_category_.json index 88e78bd99d..38c3cf5b47 100644 --- a/docs/docs/integrators/_category_.json +++ b/docs/docs/integrators/_category_.json @@ -1,4 +1,4 @@ { "label": "Integrators Guide", - "position": 5 + "position": 6 } diff --git a/docs/docs/introduction/_category_.json b/docs/docs/introduction/_category_.json index 34d319ff6a..fa1c06ac80 100644 --- a/docs/docs/introduction/_category_.json +++ b/docs/docs/introduction/_category_.json @@ -1,4 +1,4 @@ { "label": "Introduction", - "position": 2 + "position": 1 } diff --git a/docs/docs/introduction/overview.md b/docs/docs/introduction/overview.md index 8837a7cbe2..ad7e16c608 100644 --- a/docs/docs/introduction/overview.md +++ b/docs/docs/introduction/overview.md @@ -17,6 +17,10 @@ This way, provider chain validators start validating the consumer chain as well Additionally, consumer chains have the power to shape the validator set to their specific requirements by setting allow & deny lists, capping its size, etc. This allows for a wide range of security tradeoffs. For example, it enables emerging projects to deploy on consumer chains that don’t need high level of security. +- **Permissionless deployment.** + Opt-in consumer chains -- consumers that do not compel any validator to participate -- can be launched permissionlessly. + User can create and update opt-in consumer chains by submitting transactions to the provider chain. + Provider validators that are eligible can opt in on any consumer they wish to validate. - **Separation of governance from block production.** Consumer chains can separate their governance mechanism from block production. Block production is handled by provider validators, which means it is an extension of the proof-of-state (PoS) mechanism on the provider chain. @@ -46,7 +50,7 @@ If the opted-in validators misbehave on the consumer chains, then they will be p ICS currently differentiates between two types of infractions -- double signing and downtime. Double signing on consumer chains results in the misbehaving validators having their provider stake slashed and being permanently jailed on the provider, while downtime on consumer chains results in the misbehaving validators being temporarily jailed. -Note that jailing entails removing the validator from the provider active validator set and, consequently, from any of the consumer validato sets. +Note that jailing entails removing the validator from the provider active validator set and, consequently, from any of the consumer validator sets. This entails the validator will miss out on both staking and ICS rewards. **Tokenomics and rewards.** diff --git a/docs/docs/introduction/terminology.md b/docs/docs/introduction/terminology.md index 1838973e7c..85f09234b8 100644 --- a/docs/docs/introduction/terminology.md +++ b/docs/docs/introduction/terminology.md @@ -16,6 +16,11 @@ Basically, any protocol or technology that can allow one blockchain to lend/shar Interchain Security is the Cosmos-specific category of Shared Security that uses IBC (Inter-Blockchain Communication). +## Consumer Chain + +Chain that is secured by the validator set of the provider, instead of its own. +Interchain Security allows a subset of the provider chain's validator set to validate blocks on the consumer chain. + ## Replicated Security A particular protocol/implementation of Interchain Security that fully replicates the security and decentralization of a validator set across multiple blockchains. @@ -48,3 +53,7 @@ Chain that is secured by its own validator set. This chain does not participate Chains that were not initially launched as consumers of Interchain Security can still participate in the protocol and leverage the economic security of the provider chain. The process where a standalone chain transitions to being a replicated consumer chain is called the **changeover procedure** and is part of the ICS protocol. After the changeover, the new consumer chain will retain all existing state, including the IBC clients, connections and channels already established by the chain. + +## Mesh Security + +A protocol built on IBC that allows delegators on a Cosmos chain to re-delegate their stake to validators in another chain's own validator set, using the original chain's token (which remains bonded on the original chain). For a deeper exploration of Mesh Security, see [Replicated vs. Mesh Security on the Informal Blog](https://informal.systems/blog/replicated-vs-mesh-security). diff --git a/docs/docs/validators/_category_.json b/docs/docs/validators/_category_.json index 4fb7fbe69d..e4a84fad30 100644 --- a/docs/docs/validators/_category_.json +++ b/docs/docs/validators/_category_.json @@ -1,4 +1,4 @@ { "label": "Validators Guide", - "position": 4 + "position": 5 } diff --git a/docs/docusaurus.config.js b/docs/docusaurus.config.js index 1f3b46069e..0ef23ac466 100644 --- a/docs/docusaurus.config.js +++ b/docs/docusaurus.config.js @@ -88,13 +88,6 @@ const config = { type: "docsVersionDropdown", position: "left", dropdownActiveClassDisabled: false, - dropdownItemsAfter: [ - { - href: "https://cosmos.github.io/interchain-security/legacy", - label: "<= v3.x", - target: "_blank", - }, - ], }, ], }, diff --git a/docs/supported_versions.json b/docs/supported_versions.json index 5d08202e48..b81ffadf87 100644 --- a/docs/supported_versions.json +++ b/docs/supported_versions.json @@ -1,4 +1,5 @@ [ + "v6.1.0", "v5.2.0", - "v4.4.1" + "v4.5.0" ] \ No newline at end of file diff --git a/tests/e2e/test_runner.go b/tests/e2e/test_runner.go index e07360954e..18ebf340fc 100644 --- a/tests/e2e/test_runner.go +++ b/tests/e2e/test_runner.go @@ -94,7 +94,7 @@ func (tr *TestRunner) Run() error { } tr.result.Passed() - err = tr.teardownEnvironment() + // err = tr.teardownEnvironment() fmt.Printf("==========================================\n") return err } diff --git a/x/ccv/consumer/keeper/keeper.go b/x/ccv/consumer/keeper/keeper.go index 67146f7b8c..4f473badfa 100644 --- a/x/ccv/consumer/keeper/keeper.go +++ b/x/ccv/consumer/keeper/keeper.go @@ -331,6 +331,7 @@ func (k Keeper) DeletePreCCV(ctx sdk.Context) { func (k Keeper) SetInitialValSet(ctx sdk.Context, initialValSet []tmtypes.ValidatorUpdate) { store := ctx.KVStore(k.storeKey) + // TODO it's not necessary to store the entire genesis state initialValSetState := types.GenesisState{ Provider: ccv.ProviderInfo{InitialValSet: initialValSet}, } diff --git a/x/ccv/consumer/types/keys.go b/x/ccv/consumer/types/keys.go index 302b45883e..53824cbcbb 100644 --- a/x/ccv/consumer/types/keys.go +++ b/x/ccv/consumer/types/keys.go @@ -133,6 +133,7 @@ func LastDistributionTransmissionKey() []byte { } // UnbondingTimeKey returns the key for storing the unbonding period +// TODO remove as it's never used func UnbondingTimeKey() []byte { return []byte{UnbondingTimeByteKey} } @@ -212,6 +213,7 @@ func InitGenesisHeightKey() []byte { // StandaloneTransferChannelIDKey returns the key to the transfer channelID that existed from a standalone chain // changing over to a consumer +// TODO remove as it's never used func StandaloneTransferChannelIDKey() []byte { return []byte{StandaloneTransferChannelIDByteKey} } diff --git a/x/ccv/provider/keeper/permissionless.go b/x/ccv/provider/keeper/permissionless.go index f096234d2a..43504a483e 100644 --- a/x/ccv/provider/keeper/permissionless.go +++ b/x/ccv/provider/keeper/permissionless.go @@ -70,10 +70,10 @@ func (k Keeper) GetConsumerOwnerAddress(ctx sdk.Context, consumerId string) (str return string(bz), nil } -// SetConsumerOwnerAddress sets the chain id associated with this consumer id -func (k Keeper) SetConsumerOwnerAddress(ctx sdk.Context, consumerId, chainId string) { +// SetConsumerOwnerAddress sets the owner address associated with this consumer id +func (k Keeper) SetConsumerOwnerAddress(ctx sdk.Context, consumerId, owner string) { store := ctx.KVStore(k.storeKey) - store.Set(types.ConsumerIdToOwnerAddressKey(consumerId), []byte(chainId)) + store.Set(types.ConsumerIdToOwnerAddressKey(consumerId), []byte(owner)) } // DeleteConsumerOwnerAddress deletes the owner address associated with this consumer id