sidebar_position |
---|
1 |
This paper specifies the Governance module for AtomOne, a fork of the module of the Cosmos SDK, which was first described in the Cosmos Whitepaper in June 2016.
The module enables Cosmos SDK based blockchain to support an on-chain governance system. In this system, holders of the native staking token of the chain can vote on proposals on a 1 token 1 vote basis. Next is a list of features the module currently supports:
- Proposal submission: Users can submit proposals with a deposit. Once the minimum deposit is reached, the proposal enters the voting period. The deposit system is dynamic and can adjust automatically to discourage excessive spam or an excessive number of simultaneous active proposals.
- Vote: Participants can vote on proposals that reached the dynamic minimum deposit and entered the voting period.
- Claiming deposit: Users that deposited on proposals can recover their deposits if the proposal was accepted or rejected. If the proposal never entered the voting period (the dynamic minimum deposit was never reached within the deposit period), the deposit is burned.
This module is in use in AtomOne). Features that may be added in the future are described in Future Improvements.
The following specification uses ATONE as the native staking token. The module can be adapted to any Proof-Of-Stake blockchain by replacing ATONE with the native staking token of the chain.
Disclaimer: This is work in progress. Mechanisms are susceptible to change.
The governance process is divided in a few steps that are outlined below:
- Proposal submission: Proposal is submitted to the blockchain with a deposit.
- Vote: Once the deposit reaches the (dynamically set)
MinDeposit
, the proposal is confirmed and voting opens. Bonded Atone holders can then sendMsgVote
transactions to vote on the proposal. - Execution: After a period of time, the votes are tallied and depending on the result, the messages in the proposal will be executed.
Every account can submit proposals by sending a MsgSubmitProposal
transaction.
Once a proposal is submitted, it is identified by its unique proposalID
.
A proposal includes an array of sdk.Msgs
which are executed automatically if the
proposal passes. The messages are executed by the governance ModuleAccount
itself.
Modules such as x/upgrade
, that want to allow certain messages to be executed by
governance only should add a whitelist within the respective msg server, granting
the governance module the right to execute the message once a quorum has been
reached. The governance module uses the MsgServiceRouter
to check that these
messages are correctly constructed and have a respective path to execute on but
do not perform a full validity check.
To prevent spam, proposals must be submitted with an initial deposit in the
coins defined by the dynamic MinInitialDeposit
. After the proposal is submitted,
the deposit from any token holder can increase until it meets or exceeds the current
dynamic MinDeposit
. Once that threshold is reached (within the deposit period),
the proposal moves into the voting period.
In previous versions, MinDeposit
was a fixed parameter and a fraction of it (called
MinInitialDepositRatio
) was required at proposal submission.
Now, these parameters are determined by a dynamic system that can raise or
lower each threshold depending on the number of concurrent proposals in that state:
-
MinInitialDeposit
: The minimum deposit required to create (submit) a proposal. This threshold scales dynamically based on the number of proposals in the deposit period. A floor value is set so that it cannot go below a certain amount, but it can increase indefintely beyond that floor if many proposals enter deposit status at once. -
MinDeposit
: The total deposit required for a proposal to enter the voting period. This threshold also adapts to the current number of active (voting) proposals. If the system detects too many simultaneous active proposals, the minimum deposit can increase significantly, discouraging spam and helping the chain governance remain more focused.
Both dynamic deposit mechanisms have their own sets of parameters (see Parameters), can be queried via dedicated endpoints, and are individually updated as proposals enter or exit the deposit or voting stages. They also continue adjusting as time passes, ensuring that the system remains responsive to the current state of the chain.
Threshold updates are triggered both by activation or deactivation of proposals ( meaning when they enter/exit either the deposit or the voting periods) and by the passage of time. More details on the mechanism and the update formulae available in ADR-003.
When a proposal is submitted, it must be accompanied by a deposit that is at least
the current MinInitialDeposit
value. If the submitted deposit is valid, the
newly created proposal is placed in an inactive proposal queue (a.k.a. deposit
period queue). If the total deposit on the proposal is raised (through
MsgDeposit
) to meet or exceed the current (dynamic) MinDeposit
within
the deposit period, the proposal is immediately moved to the active proposal
queue and enters the voting period.
If, by the end of the deposit period, the total deposit is still below the required
MinDeposit value, the proposal is removed from state and the entire deposit is
burned. However, at the end of the deposit period if for any reason the MinDeposit
was lowered and the proposal now meets the new threshold, the proposal is promoted
to voting period instead of being removed.
The deposit is kept in escrow and held by the governance ModuleAccount
until the
proposal is finalized (passed or rejected).
When a proposal is finalized, the coins from the deposit are refunded regardless of wether the proposal is approved or rejected. If the proposal never moved to the voting period, the deposit is instead burned. All refunded or burned deposits are removed from the state. Events are issued when burning or refunding a deposit.
Participants are users that have the right to vote on proposals. On AtomOne, participants are bonded Atone holders. Unbonded Atone holders and other users do not get the right to participate in governance. However, they can still submit and deposit on proposals.
Note that when participants have bonded and unbonded Atones, their voting power is calculated from their bonded Atone holdings only.
Once a proposal reaches the dynamic MinDeposit
, it immediately enters
Voting period
. We define Voting period
as the interval between the moment
the vote opens and the moment the vote closes. The initial value of
Voting period
is 3 weeks, which is also set as a hard lower bound.
The option set of a proposal refers to the set of choices a participant can choose from when casting its vote.
The initial option set includes the following options:
Yes
No
Abstain
Abstain
option allows voters to signal that they do not intend to vote in
favor or against the proposal but accept the result of the vote.
At the end of the voting period, if the percentage of No
votes (excluding
Abstain
votes) is greater than a specific threshold (see Burnable
Params section), then the proposal is considered as SPAM and
its deposit is burned.
ADR-037 introduces the weighted vote feature which allows a staker to split their votes into several voting options. For example, it could use 70% of its voting power to vote Yes and 30% of its voting power to vote No.
Often times the entity owning that address might not be a single individual. For example, a company might have different stakeholders who want to vote differently, and so it makes sense to allow them to split their voting power. Currently, it is not possible for them to do "passthrough voting" and giving their users voting rights over their tokens. However, with this system, exchanges can poll their users for voting preferences, and then vote on-chain proportionally to the results of the poll.
To represent weighted vote on chain, we use the following Protobuf message.
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L27-L35
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L134-L150
For a weighted vote to be valid, the options
field must not contain duplicate
vote options, and the sum of weights of all options must be equal to 1.
Quorum is defined as the minimum percentage of voting power that needs to be cast on a proposal for the result to be valid.
Threshold is defined as the minimum proportion of Yes
votes (excluding
Abstain
votes) for the proposal to be accepted.
Initially, the threshold is set at 66.7% of Yes
votes, excluding Abstain
votes. Note, the value is derived from the TallyParams
on-chain parameter,
which is modifiable by governance. This means that proposals are accepted if:
- There exist bonded tokens.
- Quorum has been achieved.
- The proportion of
Abstain
votes is inferior to 1/1. - The proportion of
Yes
votes, excludingAbstain
votes, at the end of the voting period is superior to 2/3.
If a delegator does not vote, it won't inherit its validator vote.
Similarly, a validator's voting power is only equal to its own stake.
At present, validators are not punished for failing to vote.
Later, we may add permissioned keys that could only sign txs from certain modules.
For the MVP, the Governance address
will be the main validator address generated
at account creation. This address corresponds to a different PrivKey than the CometBFT
PrivKey which is responsible for signing consensus messages. Validators thus do not
have to sign governance transactions with the sensitive CometBFT PrivKey.
There are three parameters that define if the deposit of a proposal should be burned or returned to the depositors.
BurnDepositNoThreshold
burns the proposal deposit at the end of the voting period if the percentage ofNo
votes (excludingAbstain
votes) exceeds the threshold.BurnVoteQuorum
burns the proposal deposit if the proposal deposit if the vote does not reach quorum.BurnProposalDepositPrevote
burns the proposal deposit if it does not enter the voting phase.
Note: These parameters are modifiable via governance.
Proposal
objects are used to tally votes and generally track the proposal's state.
They contain an array of arbitrary sdk.Msg
's which the governance module will attempt
to resolve and then execute if the proposal passes. Proposal
's are identified by a
unique id and contains a series of timestamps: submit_time
, deposit_end_time
,
voting_start_time
, voting_end_time
which track the lifecycle of a proposal
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L51-L101
A proposal will generally require more than just a set of messages to explain its
purpose but need some greater justification and allow a means for interested participants
to discuss and debate the proposal.
In most cases, it is encouraged to have an off-chain system that supports the on-chain governance process.
To accommodate for this, a proposal contains a special metadata
field, a string,
which can be used to add context to the proposal. The metadata
field allows custom use for networks,
however, it is expected that the field contains a URL or some form of CID using a system such as
IPFS. To support the case of
interoperability across networks, the SDK recommends that the metadata
represents
the following JSON
template:
{
"title": "...",
"description": "...",
"forum": "...", // a link to the discussion platform (i.e. Discord)
"other": "..." // any extra data that doesn't correspond to the other fields
}
This makes it far easier for clients to support multiple networks.
The metadata has a maximum length that is chosen by the app developer, and passed into the gov keeper as a config. The default maximum length in the SDK is 255 characters.
There are many aspects of a chain, or of the individual modules that you may want to
use governance to perform such as changing various parameters. This is very simple
to do. First, write out your message types and MsgServer
implementation. Add an
authority
field to the keeper which will be populated in the constructor with the
governance module account: govKeeper.GetGovernanceAccount().GetAddress()
. Then for
the methods in the msg_server.go
, perform a check on the message that the signer
matches authority
. This will prevent any user from executing that message.
Parameters
define the rules according to which votes are run. There can only
be one active parameter set at any given time. If governance wants to change a
parameter set, either to modify a value or add/remove a parameter field, a new
parameter set has to be created and the previous one rendered inactive.
Due to the new dynamic depositfeature, the prior MinDeposit
parameter is
deprecated and replaced by the dynamic mechanism defined via the
MinDepositThrottler
struct. The same applies to MinInitialDepositRatio
,
which is deprecated and replaced by a dynamic MinInitialDeposit
controlled via the MinInitialDepositThrottler
struct.
(MinDeposit
inside the DepositParams
is no longer used in code. Instead, see
Params.min_deposit_throttler.)
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L167-L181
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L183-L187
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L189-L209
Parameters are stored in a global GlobalParams
KVStore.
Additionally, we introduce some basic types:
type Vote byte
const (
VoteYes = 0x1
VoteAbstain = 0x2
VoteNo = 0x3
)
type ProposalType string
const (
ProposalTypePlainText = "Text"
ProposalTypeSoftwareUpgrade = "SoftwareUpgrade"
)
type ProposalStatus byte
const (
StatusNil ProposalStatus = 0x00
StatusDepositPeriod ProposalStatus = 0x01 // Proposal is submitted. Participants can deposit on it but not vote
StatusVotingPeriod ProposalStatus = 0x02 // MinDeposit is reached, participants can vote
StatusPassed ProposalStatus = 0x03 // Proposal passed and successfully executed
StatusRejected ProposalStatus = 0x04 // Proposal has been rejected
StatusFailed ProposalStatus = 0x05 // Proposal passed but failed execution
)
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/gov.proto#L37-L49
:::note Stores are KVStores in the multi-store. The key to find the store is the first parameter in the list :::
We will use one KVStore Governance
to store four mappings:
- A mapping from
proposalID|'proposal'
toProposal
. - A mapping from
proposalID|'addresses'|address
toVote
. This mapping allows us to query all addresses that voted on the proposal along with their vote by doing a range query onproposalID:addresses
. - A mapping from
ParamsKey|'Params'
toParams
. This map allows to query all x/gov params. - A mapping from
VotingPeriodProposalKeyPrefix|proposalID
to a single byte. This allows us to know if a proposal is in the voting period or not with very low gas cost.
For pseudocode purposes, here are the two function we will use to read or write in stores:
load(StoreKey, Key)
: Retrieve item stored at keyKey
in store found at keyStoreKey
in the multistorestore(StoreKey, Key, value)
: Write valueValue
at keyKey
in store found at keyStoreKey
in the multistore
Store:
ProposalProcessingQueue
: A queuequeue[proposalID]
containing all theProposalIDs
of proposals that reachedMinDeposit
. During eachEndBlock
, all the proposals that have reached the end of their voting period are processed. To process a finished proposal, the application tallies the votes, computes the votes of each validator and checks if every validator in the validator set has voted. If the proposal is accepted, deposits are refunded. Finally, the proposal contentHandler
is executed.
And the pseudocode for the ProposalProcessingQueue
:
in EndBlock do
for finishedProposalID in GetAllFinishedProposalIDs(block.Time)
proposal = load(Governance, <proposalID|'proposal'>) // proposal is a const key
validators = Keeper.getAllValidators()
tmpValMap := map(sdk.AccAddress)stakingtypes.ValidatorI
// Tally
voterIterator = rangeQuery(Governance, <proposalID|'addresses'>) //return all the addresses that voted on the proposal
for each (voterAddress, vote) in voterIterator
delegations = stakingKeeper.getDelegations(voterAddress) // get all delegations for current voter
for each delegation in delegations
proposal.updateTally(vote, delegation.Shares)
_, isVal = stakingKeeper.getValidator(voterAddress)
if (isVal)
tmpValMap(voterAddress).Vote = vote
tallyingParam = load(GlobalParams, 'TallyingParam')
// Check if proposal is accepted or rejected
totalNonAbstain := proposal.YesVotes + proposal.NoVotes
if (proposal.Votes.YesVotes/totalNonAbstain > tallyingParam.Threshold)
// proposal was accepted at the end of the voting period
// refund deposits (non-voters already punished)
for each (amount, depositor) in proposal.Deposits
depositor.AtoneBalance += amount
stateWriter, err := proposal.Handler()
if err != nil
// proposal passed but failed during state execution
proposal.CurrentStatus = ProposalStatusFailed
else
// proposal pass and state is persisted
proposal.CurrentStatus = ProposalStatusAccepted
stateWriter.save()
else
// proposal was rejected
proposal.CurrentStatus = ProposalStatusRejected
store(Governance, <proposalID|'proposal'>, proposal)
A legacy proposal is the old implementation of governance proposal. Contrary to proposal that can contain any messages, a legacy proposal allows to submit a set of pre-defined proposals. These proposal are defined by their types.
While proposals should use the new implementation of the governance proposal, we need
still to use legacy proposal in order to submit a software-upgrade
and a
cancel-software-upgrade
proposal.
More information on how to submit proposals in the client section.
The module provides an extension mechanism for the voting period. By enforcing a delay when quorum is reached too close to the end of the voting period, we ensure that the community has enough time to understand all the proposal's implications and potentially react accordingly without the worry of an imminent end to the voting period.
QuorumTimeout
: This parameter defines the time window after which, if the quorum is reached, the voting end time is extended. This value must be strictly less thanparams.VotingPeriod
.MaxVotingPeriodExtension
: This parameter defines the maximum amount of time by which a proposal's voting end time can be extended. This value must be greater or equal thanVotingPeriod - QuorumTimeout
.QuorumCheckCount
: This parameter specifies the number of times a proposal should be checked for achieving quorum after the expiration ofQuorumTimeout
. It is used to determine the intervals at which these checks will take place. The intervals are calculated as(VotingPeriod - QuorumTimeout) / QuorumCheckCount
. This avoids the need to check for quorum at the end of each block, which would have a significant impact on performance. Furthermore, if this value is set to 0, the quorum check and voting period extension system is considered disabled.
Store:
We also introduce a new keeper.QuorumCheckQueue
similar to keeper.ActiveProposalsQueue
and keeper.InactiveProposalsQueue
. This queue stores proposals that are due to be
checked for quorum. The key for each proposal in the queue is a pair containing the time
at which the proposal should be checked for quorum as the first part, and the proposal.Id
as the second. The value will instead be a QuorumCheckQueueEntry
struct that will store:
QuorumTimeoutTime
, indicating the time at which this proposal will pass theQuorumTimeout
and computed asproposal.VotingStartTime + QuorumTimeout
QuorumCheckCount
, a copy of the value of the module parameter with the same name at the time of first insertion of this proposal in theQuorumCheckQueue
QuorumChecksDone
, indicating the number of quorum checks that have been already performed, initially 0
When a proposal is added to the keeper.ActiveProposalsQueue
, it is also added to the
keeper.QuorumCheckQueue
. The time part of the key for the proposal in the
QuorumCheckQueue
is initially calculated as proposal.VotingStartTime + QuorumTimeout
(i.e. the QuorumTimeoutTime
), therefore scheduling the first quorum check to happen
right after QuorumTimeout
has expired.
In the EndBlocker()
function of the x/gov
module, we add a new call to
keeper.IterateQuorumCheckQueue()
between the calls to keeper.IterateInactiveProposalsQueue()
and keeper.IterateActiveProposalsQueue(
, where we iterate over proposals
that are due to be checked for quorum, meaning that their time part of the key is
before the current block time.
If a proposal has reached quorum (approximately) before or right at the
QuorumTimeout
- i.e. the QuorumChecksDone
is 0, meaning more precisely
that no previous quorum checks were performed - remove it from the QuorumCheckQueue
and do nothing, the proposal should end as expected.
If a proposal has reached quorum after the QuorumTimeout
- i.e.
QuorumChecksDone
> 0 - update the proposal.VotingEndTime
as
ctx.BlockTime() + MaxVotingPeriodExtension
and remove it from the
keeper.QuorumCheckQueue
.
If a proposal is still active and has not yet reached quorum, remove the corresponding
item from keeper.QuorumCheckQueue
, modify the last QuorumCheckQueueEntry
value by
incrementing QuorumChecksDone
to record this latest unsuccessful quorum check, and add
the proposal back to keeper.QuorumCheckQueue
with updated keys and value.
To compute the time part of the new key, add a quorum check interval - which is computed as
(VotingPeriod - QuorumTimeout) / QuorumCheckCount
- to the time part of the last key used in
keeper.QuorumCheckQueue
for this proposal. Specifically, use the formula
lastKey.K1.Add((VotingPeriod - QuorumTimeout) / QuorumCheckCount)
. As previously stated,
the value will remain the same struct as before, with QuorumChecksDone
incremented by 1 to reflect
the additional unsuccessful quorum check that was performed.
If a proposal has passed its VoteEndTime
and has not reached quorum, it should be removed from
keeper.QuorumCheckQueue
without any additional actions. The proposal's failure will be handled
in the subsequent keeper.IterateActiveProposalsQueue
.
A constitution
string can be set at genesis with arbitrary content and is intended to be used
to store the chain established constitution upon launch.
The constitution
can be updated through Constitution Amendment Proposals which must include
a valid patch of the constitution
string expressed in unified diff format.
Example (from gnu.org):
--- lao 2002-02-21 23:30:39.942229878 -0800
+++ tzu 2002-02-21 23:30:50.442260588 -0800
@@ -1,7 +1,6 @@
-The Way that can be told of is not the eternal Way;
-The name that can be named is not the eternal name.
The Nameless is the origin of Heaven and Earth;
-The Named is the mother of all things.
+The named is the mother of all things.
+
Therefore let there always be non-being,
so we may see their subtlety,
And let there always be being,
@@ -9,3 +8,6 @@
The two are the same,
But after they are produced,
they have different names.
+They both may be called deep and profound.
+Deeper and more profound,
+The door of all subtleties!
If Law or Constitution Amendment Proposals are submitted - by providing either a
MsgProposeLaw
or a MsgProposeConstitutionAmendment
in the MsgSubmitProposal.messages
field, the related proposal will be tallied using specific quorum and threshold values
instead of the default ones for regular proposals. More specifically, the following parameters
are added to enable this behavior:
constitution_amendment_quorum
which defines the quorum for constitution amendment proposalsconstitution_amendment_threshold
which defines the minimum proportion of Yes votes for a Constitution Amendment proposal to pass.law_quorum
which defines the quorum for law proposalslaw_threshold
which defines the minimum proportion of Yes votes for a Law proposal to pass.
The MsgProposeLaw
just contains for now an authority
field indicating who will execute the
sdk.Msg
(which should be the governance module account), and has no effects for now. The conent
of Laws is entirely defined in the proposal summary
. Example:
{
"authority": "cosmos10d07y265gmmuvt4z0w9aw880jnsr700j6zn9kn"
}
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/tx.proto#L195-L202
The MsgProposeConstitutionAmendment
contains the authority
field and also an amendment
field
that needs to be a string representing a valid patch for the constitution
expressed in
unified diff format. Example:
{
"authority": "cosmos10d07y265gmmuvt4z0w9aw880jnsr700j6zn9kn",
"amendment": "--- src\\n+++ dst\\n@@ -1 +1 @@\\n-Old Constitution\\n+Modified Constitution\\n\"
}
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/tx.proto#L209-L219
Upon execution of the MsgProposeConstitutionAmendment
(which will happen if the proposal passes)
The constitution
string will be updated by applying the patch defined in the amendment
string.
An error will be returned if the amendment
string is malformed, so constitution amendment proposals
need to be crafted with care.
The LastMinDeposit
and LastMinInitialDeposit
are used to store the last values
of the dynamic MinDeposit
and MinInitialDeposit
respectively upon proposals
activation or deactivation. These values are used to determine the current values
for the dynamic MinDeposit
and MinInitialDeposit
accounting also for the
passage of time as detailed in ADR-003
Store:
https://github.com/atomone-hub/atomone/blob/fb05dcaba40c7a1531a6806487fcd47a3e4aaef4/proto/atomone/gov/v1/gov.proto#L51-L60
Proposals can be submitted by any account via a MsgSubmitProposal
transaction.
If the total deposit in the message is below the required dynamic MinInitialDeposit
at the time of submission, the transaction will fail. Otherwise, a new proposal
is created, the deposit is moved under governance escrow, and the proposal enters
the deposit period.
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/tx.proto#L53-L82
All sdk.Msgs
passed into the messages
field of a MsgSubmitProposal
message
must be registered in the app's MsgServiceRouter
. Each of these messages must
have one signer, namely the gov module account. And finally, the metadata length
must not be larger than the maxMetadataLen
config passed into the gov keeper.
State modifications:
- Generate new
proposalID
- Create new
Proposal
- Initialise
Proposal
's attributes - Decrease balance of sender by
InitialDeposit
- If
MinDeposit
is reached:- Push
proposalID
inProposalProcessingQueue
- Push
- Transfer
InitialDeposit
from theProposer
to the governanceModuleAccount
A MsgSubmitProposal
transaction can be handled according to the following
pseudocode.
// PSEUDOCODE //
// Check if MsgSubmitProposal is valid. If it is, create proposal //
upon receiving txGovSubmitProposal from sender do
if !correctlyFormatted(txGovSubmitProposal)
// check if proposal is correctly formatted and the messages have routes to other modules. Includes fee payment.
// check if all messages' unique Signer is the gov acct.
// check if the metadata is not too long.
throw
initialDeposit = txGovSubmitProposal.InitialDeposit
if (initialDeposit.Atones <= 0) OR (sender.AtoneBalance < initialDeposit.Atones)
// InitialDeposit is negative or null OR sender has insufficient funds
throw
if (txGovSubmitProposal.Type != ProposalTypePlainText) OR (txGovSubmitProposal.Type != ProposalTypeSoftwareUpgrade)
sender.AtoneBalance -= initialDeposit.Atones
depositParam = load(GlobalParams, 'DepositParam')
proposalID = generate new proposalID
proposal = NewProposal()
proposal.Messages = txGovSubmitProposal.Messages
proposal.Metadata = txGovSubmitProposal.Metadata
proposal.TotalDeposit = initialDeposit
proposal.SubmitTime = <CurrentTime>
proposal.DepositEndTime = <CurrentTime>.Add(depositParam.MaxDepositPeriod)
proposal.Deposits.append({initialDeposit, sender})
proposal.Submitter = sender
proposal.YesVotes = 0
proposal.NoVotes = 0
proposal.AbstainVotes = 0
proposal.CurrentStatus = ProposalStatusOpen
store(Proposals, <proposalID|'proposal'>, proposal) // Store proposal in Proposals mapping
return proposalID
Once a proposal is submitted, if
Proposal.TotalDeposit < ActiveParam.MinDeposit
, Atone holders can send
MsgDeposit
transactions to increase the proposal's deposit until the
proposal’s total deposit meets the dynamic MinDeposit
.
If it surpasses the dynamic threshold within the deposit period, the
proposal is moved into the voting period immediately. Otherwise, the deposit
period eventually ends, and if the threshold is never met, the proposal is removed
from state and all deposits are burned. If however by the time the deposit
period ends the MinDeposit
has been lowered and the proposal now meets the
new threshold, the proposal is activated instead of being removed.
Any deposit from Atone holders (including the proposer) need to be of at least
ActiveParam.MinDeposit
* ActiveParam.MinDepositRatio
, where
ActiveParam.MinDepositRatio
must be a valid percentage between 0 and 1.
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/tx.proto#L150-L165
State modifications:
- Decrease balance of sender by
deposit
- Add
deposit
of sender inproposal.Deposits
- Increase
proposal.TotalDeposit
by sender'sdeposit
- If
MinDeposit
is reached:- Push
proposalID
inProposalProcessingQueueEnd
- Push
- Transfer
Deposit
from theproposer
to the governanceModuleAccount
A MsgDeposit
transaction has to go through a number of checks to be valid.
These checks are outlined in the following pseudocode.
// PSEUDOCODE //
// Check if MsgDeposit is valid. If it is, increase deposit and check if MinDeposit is reached
upon receiving txGovDeposit from sender do
// check if proposal is correctly formatted. Includes fee payment.
if !correctlyFormatted(txGovDeposit)
throw
proposal = load(Proposals, <txGovDeposit.ProposalID|'proposal'>) // proposal is a const key, proposalID is variable
if (proposal == nil)
// There is no proposal for this proposalID
throw
if (txGovDeposit.Deposit.Atones <= 0) OR (sender.AtoneBalance < txGovDeposit.Deposit.Atones) OR (proposal.CurrentStatus != ProposalStatusOpen)
// deposit is negative or null
// OR sender has insufficient funds
// OR proposal is not open for deposit anymore
throw
depositParam = load(GlobalParams, 'DepositParam')
if (CurrentBlock >= proposal.SubmitBlock + depositParam.MaxDepositPeriod)
proposal.CurrentStatus = ProposalStatusClosed
else
// sender can deposit
sender.AtoneBalance -= txGovDeposit.Deposit.Atones
proposal.Deposits.append({txGovVote.Deposit, sender})
proposal.TotalDeposit.Plus(txGovDeposit.Deposit)
if (proposal.TotalDeposit >= depositParam.MinDeposit)
// MinDeposit is reached, vote opens
proposal.VotingStartBlock = CurrentBlock
proposal.CurrentStatus = ProposalStatusActive
ProposalProcessingQueue.push(txGovDeposit.ProposalID)
store(Proposals, <txGovVote.ProposalID|'proposal'>, proposal)
Once ActiveParam.MinDeposit
is reached, voting period starts. From there,
bonded Atone holders are able to send MsgVote
transactions to cast their
vote on the proposal.
https://github.com/atomone-hub/atomone/blob/b9631ed2e3b781cd82a14316f6086802d8cb4dcf/proto/atomone/gov/v1/tx.proto#L106-L123
State modifications:
- Record
Vote
of sender
:::note Gas cost for this message has to take into account the future tallying of the vote in EndBlocker. :::
Next is a pseudocode outline of the way MsgVote
transactions are handled:
// PSEUDOCODE //
// Check if MsgVote is valid. If it is, count vote//
upon receiving txGovVote from sender do
// check if proposal is correctly formatted. Includes fee payment.
if !correctlyFormatted(txGovDeposit)
throw
proposal = load(Proposals, <txGovDeposit.ProposalID|'proposal'>)
if (proposal == nil)
// There is no proposal for this proposalID
throw
if (proposal.CurrentStatus == ProposalStatusActive)
// Sender can vote if
// Proposal is active
// Sender has some bonds
store(Governance, <txGovVote.ProposalID|'addresses'|sender>, txGovVote.Vote) // Voters can vote multiple times. Re-voting overrides previous vote. This is ok because tallying is done once at the end.
The governance module emits the following events:
Type | Attribute Key | Attribute Value |
---|---|---|
inactive_proposal | proposal_id | {proposalID} |
inactive_proposal | proposal_result | {proposalResult} |
active_proposal | proposal_id | {proposalID} |
active_proposal | proposal_result | {proposalResult} |
quorum_check | proposal_id | {proposalID} |
quorum_check | proposal_result | {proposalResult} |
Type | Attribute Key | Attribute Value |
---|---|---|
submit_proposal | proposal_id | {proposalID} |
submit_proposal [0] | voting_period_start | {proposalID} |
proposal_deposit | amount | {depositAmount} |
proposal_deposit | proposal_id | {proposalID} |
message | module | governance |
message | action | submit_proposal |
message | sender | {senderAddress} |
- [0] Event only emitted if the voting period starts during the submission.
Type | Attribute Key | Attribute Value |
---|---|---|
proposal_vote | option | {voteOption} |
proposal_vote | proposal_id | {proposalID} |
message | module | governance |
message | action | vote |
message | sender | {senderAddress} |
Type | Attribute Key | Attribute Value |
---|---|---|
proposal_vote | option | {weightedVoteOptions} |
proposal_vote | proposal_id | {proposalID} |
message | module | governance |
message | action | vote |
message | sender | {senderAddress} |
Type | Attribute Key | Attribute Value |
---|---|---|
proposal_deposit | amount | {depositAmount} |
proposal_deposit | proposal_id | {proposalID} |
proposal_deposit [0] | voting_period_start | {proposalID} |
message | module | governance |
message | action | deposit |
message | sender | {senderAddress} |
- [0] Event only emitted if the voting period starts during the submission.
Below is an updated parameter set with new fields related to dynamic deposit.
Some older fields have been deprecated but remain in gov.proto
for backward compatibility:
Key | Type | Example |
---|---|---|
max_deposit_period | string (time ns) | "172800000000000" (17280s) |
voting_period | string (time ns) | "172800000000000" (17280s) |
quorum | string (dec) | "0.334000000000000000" |
threshold | string (dec) | "0.500000000000000000" |
burn_proposal_deposit_prevote | bool | false |
burn_vote_quorum | bool | false |
min_deposit_ratio | string (dec) | "0.010000000000000000" |
constitution_amendment_quorum | string (dec) | "0.334000000000000000" |
constitution_amendment_threshold | string (dec) | "0.900000000000000000" |
law_quorum | string (dec) | "0.334000000000000000" |
law_threshold | string (dec) | "0.900000000000000000" |
quorum_timeout | string (time ns) | "172800000000000" (17280s) |
max_voting_period_extension | string (time ns) | "172800000000000" (17280s) |
quorum_check_count | uint64 | 2 |
min_deposit_throttler | object (MinDepositThrottler) | See below |
min_initial_deposit_throttler | object (MinInitialDepositThrottler) | See below |
The min_deposit_throttler
field in Params
controls how MinDeposit
is computed dynamically:
floor_value
: The floor (lowest possible) deposit requirement.update_period
: After how long the system should recalculate (time-based updates).target_active_proposals
: The number of active proposals the dynamic deposit tries to target.increase_ratio
/decrease_ratio
: How fast the min deposit goes up/down relative to exceeding or being under that target.sensitivity_target_distance
: A positive integer indicating how sensitive the multiplier is to how far away we are from the target number of active proposals.
Similarly, the min_initial_deposit_throttler
sub-structure defines the dynamic
MinInitialDeposit
:
floor_value
: The floor (lowest possible) initial deposit requirement.update_period
: After how long the system should recalculate (time-based updates).target_proposals
: The target number of proposals in the deposit period.increase_ratio
/decrease_ratio
: Rate of upward/downward adjustments when the number of deposit-period proposals deviates from the target.sensitivity_target_distance
: Like above, how sharply the required deposit reacts to that deviation.
:::note Both dynamic thresholds are maintained internally and automatically updated whenever proposals enter/exit their respective states (deposit or voting) and with the passage of time. At any time, one can do:
atomoned query gov min-deposit
atomoned query gov min-initial-deposit
to see the current required deposit thresholds. :::
A user can query and interact with the gov
module using the CLI.
The query
commands allow users to query gov
state.
atomoned query gov --help
The deposit
command allows users to query a deposit for a given proposal from a given depositor.
atomoned query gov deposit [proposal-id] [depositer-addr] [flags]
Example:
atomoned query gov deposit 1 atone1..
Example Output:
amount:
- amount: "100"
denom: atone
depositor: atone1..
proposal_id: "1"
The deposits
command allows users to query all deposits for a given proposal.
atomoned query gov deposits [proposal-id] [flags]
Example:
atomoned query gov deposits 1
Example Output:
deposits:
- amount:
- amount: "100"
denom: atone
depositor: atone1..
proposal_id: "1"
pagination:
next_key: null
total: "0"
The min-deposit
command allows users to query the
dynamic minimum deposit required for a proposal
to enter voting period.
atomoned query gov min-deposit [flags]
Example:
atomoned query gov min-deposit
Example Output:
min_deposit:
- amount: "10000000"
denom: atone
The min-initial-deposit
command allows users to query the
dynamic minimum initial deposit required for a proposal to
be submitted.
atomoned query gov min-initial-deposit [flags]
Example:
atomoned query gov min-initial-deposit
Example Output:
min_initial_deposit:
- amount: "10000000"
denom: atone
The param
command allows users to query a given parameter for the gov
module.
atomoned query gov param [param-type] [flags]
Example:
atomoned query gov param voting
Example Output:
voting_period: "172800000000000"
The params
command allows users to query all parameters for the gov
module.
atomoned query gov params [flags]
Example:
atomoned query gov params
Example Output:
deposit_params:
max_deposit_period: "172800000000000"
min_deposit:
- amount: "10000000"
denom: atone
tally_params:
quorum: "0.334000000000000000"
threshold: "0.500000000000000000"
voting_params:
voting_period: "172800000000000"
The proposal
command allows users to query a given proposal.
atomoned query gov proposal [proposal-id] [flags]
Example:
atomoned query gov proposal 1
Example Output:
deposit_end_time: "2022-03-30T11:50:20.819676256Z"
final_tally_result:
abstain_count: "0"
no_count: "0"
yes_count: "0"
id: "1"
messages:
- '@type': /cosmos.bank.v1beta1.MsgSend
amount:
- amount: "10"
denom: atone
from_address: atone1..
to_address: atone1..
metadata: AQ==
status: PROPOSAL_STATUS_DEPOSIT_PERIOD
submit_time: "2022-03-28T11:50:20.819676256Z"
total_deposit:
- amount: "10"
denom: atone
voting_end_time: null
voting_start_time: null
The proposals
command allows users to query all proposals with optional filters.
atomoned query gov proposals [flags]
Example:
atomoned query gov proposals
Example Output:
pagination:
next_key: null
total: "0"
proposals:
- deposit_end_time: "2022-03-30T11:50:20.819676256Z"
final_tally_result:
abstain_count: "0"
no_count: "0"
yes_count: "0"
id: "1"
messages:
- '@type': /cosmos.bank.v1beta1.MsgSend
amount:
- amount: "10"
denom: atone
from_address: atone1..
to_address: atone1..
metadata: AQ==
status: PROPOSAL_STATUS_DEPOSIT_PERIOD
submit_time: "2022-03-28T11:50:20.819676256Z"
total_deposit:
- amount: "10"
denom: atone
voting_end_time: null
voting_start_time: null
- deposit_end_time: "2022-03-30T14:02:41.165025015Z"
final_tally_result:
abstain_count: "0"
no_count: "0"
yes_count: "0"
id: "2"
messages:
- '@type': /cosmos.bank.v1beta1.MsgSend
amount:
- amount: "10"
denom: atone
from_address: atone1..
to_address: atone1..
metadata: AQ==
status: PROPOSAL_STATUS_DEPOSIT_PERIOD
submit_time: "2022-03-28T14:02:41.165025015Z"
total_deposit:
- amount: "10"
denom: atone
voting_end_time: null
voting_start_time: null
The proposer
command allows users to query the proposer for a given proposal.
atomoned query gov proposer [proposal-id] [flags]
Example:
atomoned query gov proposer 1
Example Output:
proposal_id: "1"
proposer: atone1..
The tally
command allows users to query the tally of a given proposal vote.
atomoned query gov tally [proposal-id] [flags]
Example:
atomoned query gov tally 1
Example Output:
abstain: "0"
"no": "0"
"yes": "1"
The vote
command allows users to query a vote for a given proposal.
atomoned query gov vote [proposal-id] [voter-addr] [flags]
Example:
atomoned query gov vote 1 atone1..
Example Output:
option: VOTE_OPTION_YES
options:
- option: VOTE_OPTION_YES
weight: "1.000000000000000000"
proposal_id: "1"
voter: atone1..
The votes
command allows users to query all votes for a given proposal.
atomoned query gov votes [proposal-id] [flags]
Example:
atomoned query gov votes 1
Example Output:
pagination:
next_key: null
total: "0"
votes:
- option: VOTE_OPTION_YES
options:
- option: VOTE_OPTION_YES
weight: "1.000000000000000000"
proposal_id: "1"
voter: atone1..
The tx
commands allow users to interact with the gov
module.
atomoned tx gov --help
The deposit
command allows users to deposit tokens for a given proposal.
atomoned tx gov deposit [proposal-id] [deposit] [flags]
Example:
atomoned tx gov deposit 1 10000000atone --from atone1..
The draft-proposal
command allows users to draft any type of proposal.
The command returns a draft_proposal.json
, to be used by submit-proposal
after being completed.
The draft_metadata.json
is meant to be uploaded to IPFS.
atomoned tx gov draft-proposal
The generate-constitution-amendment
command allows users to generate a constitution amendment
proposal message from the current constitution, either queried or provided as an .md
file through
the flag --current-constitution
and the provided updated constitution .md
file.
atomoned tx gov generate-constitution-amendment path/to/updated/constitution.md
The submit-proposal
command allows users to submit a governance proposal along with some messages and metadata.
Messages, metadata and deposit are defined in a JSON file.
atomoned tx gov submit-proposal [path-to-proposal-json] [flags]
Example:
atomoned tx gov submit-proposal /path/to/proposal.json --from atone1..
where proposal.json
contains:
{
"messages": [
{
"@type": "/cosmos.bank.v1beta1.MsgSend",
"from_address": "atone1...", // The gov module module address
"to_address": "atone1...",
"amount":[{"denom": "atone","amount": "10"}]
}
],
"metadata": "AQ==",
"deposit": "10atone",
"title": "Proposal Title",
"summary": "Proposal Summary"
}
:::note By default the metadata, summary and title are both limited by 255 characters, this can be overridden by the application developer. :::
The submit-legacy-proposal
command allows users to submit a governance legacy proposal along with an initial deposit.
atomoned tx gov submit-legacy-proposal [command] [flags]
Example:
atomoned tx gov submit-legacy-proposal --title="Test Proposal" --description="testing" --type="Text" --deposit="100000000atone" --from atone1..
Example (cancel-software-upgrade
):
atomoned tx gov submit-legacy-proposal cancel-software-upgrade --title="Test Proposal" --description="testing" --deposit="100000000atone" --from atone1..
Example (param-change
):
atomoned tx gov submit-legacy-proposal param-change proposal.json --from atone1..
{
"title": "Test Proposal",
"description": "testing, testing, 1, 2, 3",
"changes": [
{
"subspace": "staking",
"key": "MaxValidators",
"value": 100
}
],
"deposit": "10000000atone"
}
Example (software-upgrade
):
atomoned tx gov submit-legacy-proposal software-upgrade v2 --title="Test Proposal" --description="testing, testing, 1, 2, 3" --upgrade-height 1000000 --from atone1..
The vote
command allows users to submit a vote for a given governance proposal.
atomoned tx gov vote [command] [flags]
Example:
atomoned tx gov vote 1 yes --from atone1..
The weighted-vote
command allows users to submit a weighted vote for a given governance proposal.
atomoned tx gov weighted-vote [proposal-id] [weighted-options] [flags]
Example:
atomoned tx gov weighted-vote 1 yes=0.5,no=0.5 --from atone1..
A user can query the gov
module using gRPC endpoints.
The Proposal
endpoint allows users to query a given proposal.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Proposal
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Proposal
Example Output:
{
"proposal": {
"proposalId": "1",
"content": {"@type":"/cosmos.gov.v1beta1.TextProposal","description":"testing, testing, 1, 2, 3","title":"Test Proposal"},
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"finalTallyResult": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submitTime": "2021-09-16T19:40:08.712440474Z",
"depositEndTime": "2021-09-18T19:40:08.712440474Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10000000"
}
],
"votingStartTime": "2021-09-16T19:40:08.712440474Z",
"votingEndTime": "2021-09-18T19:40:08.712440474Z",
"title": "Test Proposal",
"summary": "testing, testing, 1, 2, 3"
}
}
Using v1:
cosmos.gov.v1.Query/Proposal
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1.Query/Proposal
Example Output:
{
"proposal": {
"id": "1",
"messages": [
{"@type":"/cosmos.bank.v1beta1.MsgSend","amount":[{"denom":"atone","amount":"10"}],"fromAddress":"atone1..","toAddress":"atone1.."}
],
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"finalTallyResult": {
"yesCount": "0",
"abstainCount": "0",
"noCount": "0",
},
"submitTime": "2022-03-28T11:50:20.819676256Z",
"depositEndTime": "2022-03-30T11:50:20.819676256Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10000000"
}
],
"votingStartTime": "2022-03-28T14:25:26.644857113Z",
"votingEndTime": "2022-03-30T14:25:26.644857113Z",
"metadata": "AQ==",
"title": "Test Proposal",
"summary": "testing, testing, 1, 2, 3"
}
}
The Proposals
endpoint allows users to query all proposals with optional filters.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Proposals
Example:
grpcurl -plaintext \
localhost:9090 \
cosmos.gov.v1beta1.Query/Proposals
Example Output:
{
"proposals": [
{
"proposalId": "1",
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"finalTallyResult": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submitTime": "2022-03-28T11:50:20.819676256Z",
"depositEndTime": "2022-03-30T11:50:20.819676256Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10000000010"
}
],
"votingStartTime": "2022-03-28T14:25:26.644857113Z",
"votingEndTime": "2022-03-30T14:25:26.644857113Z"
},
{
"proposalId": "2",
"status": "PROPOSAL_STATUS_DEPOSIT_PERIOD",
"finalTallyResult": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submitTime": "2022-03-28T14:02:41.165025015Z",
"depositEndTime": "2022-03-30T14:02:41.165025015Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10"
}
],
"votingStartTime": "0001-01-01T00:00:00Z",
"votingEndTime": "0001-01-01T00:00:00Z"
}
],
"pagination": {
"total": "2"
}
}
Using v1:
cosmos.gov.v1.Query/Proposals
Example:
grpcurl -plaintext \
localhost:9090 \
cosmos.gov.v1.Query/Proposals
Example Output:
{
"proposals": [
{
"id": "1",
"messages": [
{"@type":"/cosmos.bank.v1beta1.MsgSend","amount":[{"denom":"atone","amount":"10"}],"fromAddress":"atone1..","toAddress":"atone1.."}
],
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"finalTallyResult": {
"yesCount": "0",
"abstainCount": "0",
"noCount": "0",
},
"submitTime": "2022-03-28T11:50:20.819676256Z",
"depositEndTime": "2022-03-30T11:50:20.819676256Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10000000010"
}
],
"votingStartTime": "2022-03-28T14:25:26.644857113Z",
"votingEndTime": "2022-03-30T14:25:26.644857113Z",
"metadata": "AQ==",
"title": "Proposal Title",
"summary": "Proposal Summary"
},
{
"id": "2",
"messages": [
{"@type":"/cosmos.bank.v1beta1.MsgSend","amount":[{"denom":"atone","amount":"10"}],"fromAddress":"atone1..","toAddress":"atone1.."}
],
"status": "PROPOSAL_STATUS_DEPOSIT_PERIOD",
"finalTallyResult": {
"yesCount": "0",
"abstainCount": "0",
"noCount": "0",
},
"submitTime": "2022-03-28T14:02:41.165025015Z",
"depositEndTime": "2022-03-30T14:02:41.165025015Z",
"totalDeposit": [
{
"denom": "atone",
"amount": "10"
}
],
"metadata": "AQ==",
"title": "Proposal Title",
"summary": "Proposal Summary"
}
],
"pagination": {
"total": "2"
}
}
The Vote
endpoint allows users to query a vote for a given proposal.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Vote
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1","voter":"atone1.."}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Vote
Example Output:
{
"vote": {
"proposalId": "1",
"voter": "atone1..",
"option": "VOTE_OPTION_YES",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1000000000000000000"
}
]
}
}
Using v1:
cosmos.gov.v1.Query/Vote
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1","voter":"atone1.."}' \
localhost:9090 \
cosmos.gov.v1.Query/Vote
Example Output:
{
"vote": {
"proposalId": "1",
"voter": "atone1..",
"option": "VOTE_OPTION_YES",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
]
}
}
The Votes
endpoint allows users to query all votes for a given proposal.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Votes
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Votes
Example Output:
{
"votes": [
{
"proposalId": "1",
"voter": "atone1..",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1000000000000000000"
}
]
}
],
"pagination": {
"total": "1"
}
}
Using v1:
cosmos.gov.v1.Query/Votes
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1.Query/Votes
Example Output:
{
"votes": [
{
"proposalId": "1",
"voter": "atone1..",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
]
}
],
"pagination": {
"total": "1"
}
}
The Params
endpoint allows users to query all parameters for the gov
module.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Params
Example:
grpcurl -plaintext \
-d '{"params_type":"voting"}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Params
Example Output:
{
"votingParams": {
"votingPeriod": "172800s"
},
"depositParams": {
"maxDepositPeriod": "0s"
},
"tallyParams": {
"quorum": "MA==",
"threshold": "MA==",
}
}
Using v1:
cosmos.gov.v1.Query/Params
Example:
grpcurl -plaintext \
-d '{"params_type":"voting"}' \
localhost:9090 \
cosmos.gov.v1.Query/Params
Example Output:
{
"votingParams": {
"votingPeriod": "172800s"
}
}
The Deposit
endpoint allows users to query a deposit for a given proposal from a given depositor.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Deposit
Example:
grpcurl -plaintext \
'{"proposal_id":"1","depositor":"atone1.."}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Deposit
Example Output:
{
"deposit": {
"proposalId": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
}
Using v1:
cosmos.gov.v1.Query/Deposit
Example:
grpcurl -plaintext \
'{"proposal_id":"1","depositor":"atone1.."}' \
localhost:9090 \
cosmos.gov.v1.Query/Deposit
Example Output:
{
"deposit": {
"proposalId": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
}
The Deposits
endpoint allows users to query all deposits for a given proposal.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/Deposits
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/Deposits
Example Output:
{
"deposits": [
{
"proposalId": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
],
"pagination": {
"total": "1"
}
}
Using v1:
cosmos.gov.v1.Query/Deposits
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1.Query/Deposits
Example Output:
{
"deposits": [
{
"proposalId": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
],
"pagination": {
"total": "1"
}
}
The TallyResult
endpoint allows users to query the tally of a given proposal.
Using legacy v1beta1:
cosmos.gov.v1beta1.Query/TallyResult
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1beta1.Query/TallyResult
Example Output:
{
"tally": {
"yes": "1000000",
"abstain": "0",
"no": "0",
}
}
Using v1:
cosmos.gov.v1.Query/TallyResult
Example:
grpcurl -plaintext \
-d '{"proposal_id":"1"}' \
localhost:9090 \
cosmos.gov.v1.Query/TallyResult
Example Output:
{
"tally": {
"yes": "1000000",
"abstain": "0",
"no": "0",
}
}
A user can query the gov
module using REST endpoints.
The proposals
endpoint allows users to query a given proposal.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1
Example Output:
{
"proposal": {
"proposal_id": "1",
"content": null,
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"final_tally_result": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submit_time": "2022-03-28T11:50:20.819676256Z",
"deposit_end_time": "2022-03-30T11:50:20.819676256Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10000000010"
}
],
"voting_start_time": "2022-03-28T14:25:26.644857113Z",
"voting_end_time": "2022-03-30T14:25:26.644857113Z"
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}
Example:
curl localhost:1317/atomone/gov/v1/proposals/1
Example Output:
{
"proposal": {
"id": "1",
"messages": [
{
"@type": "/cosmos.bank.v1beta1.MsgSend",
"from_address": "atone1..",
"to_address": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10"
}
]
}
],
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"final_tally_result": {
"yes_count": "0",
"abstain_count": "0",
"no_count": "0",
},
"submit_time": "2022-03-28T11:50:20.819676256Z",
"deposit_end_time": "2022-03-30T11:50:20.819676256Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10000000"
}
],
"voting_start_time": "2022-03-28T14:25:26.644857113Z",
"voting_end_time": "2022-03-30T14:25:26.644857113Z",
"metadata": "AQ==",
"title": "Proposal Title",
"summary": "Proposal Summary"
}
}
The proposals
endpoint also allows users to query all proposals with optional filters.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals
Example Output:
{
"proposals": [
{
"proposal_id": "1",
"content": null,
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"final_tally_result": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submit_time": "2022-03-28T11:50:20.819676256Z",
"deposit_end_time": "2022-03-30T11:50:20.819676256Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10000000"
}
],
"voting_start_time": "2022-03-28T14:25:26.644857113Z",
"voting_end_time": "2022-03-30T14:25:26.644857113Z"
},
{
"proposal_id": "2",
"content": null,
"status": "PROPOSAL_STATUS_DEPOSIT_PERIOD",
"final_tally_result": {
"yes": "0",
"abstain": "0",
"no": "0",
},
"submit_time": "2022-03-28T14:02:41.165025015Z",
"deposit_end_time": "2022-03-30T14:02:41.165025015Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10"
}
],
"voting_start_time": "0001-01-01T00:00:00Z",
"voting_end_time": "0001-01-01T00:00:00Z"
}
],
"pagination": {
"next_key": null,
"total": "2"
}
}
Using v1:
/atomone/gov/v1/proposals
Example:
curl localhost:1317/atomone/gov/v1/proposals
Example Output:
{
"proposals": [
{
"id": "1",
"messages": [
{
"@type": "/cosmos.bank.v1beta1.MsgSend",
"from_address": "atone1..",
"to_address": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10"
}
]
}
],
"status": "PROPOSAL_STATUS_VOTING_PERIOD",
"final_tally_result": {
"yes_count": "0",
"abstain_count": "0",
"no_count": "0",
},
"submit_time": "2022-03-28T11:50:20.819676256Z",
"deposit_end_time": "2022-03-30T11:50:20.819676256Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10000000010"
}
],
"voting_start_time": "2022-03-28T14:25:26.644857113Z",
"voting_end_time": "2022-03-30T14:25:26.644857113Z",
"metadata": "AQ==",
"title": "Proposal Title",
"summary": "Proposal Summary"
},
{
"id": "2",
"messages": [
{
"@type": "/cosmos.bank.v1beta1.MsgSend",
"from_address": "atone1..",
"to_address": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10"
}
]
}
],
"status": "PROPOSAL_STATUS_DEPOSIT_PERIOD",
"final_tally_result": {
"yes_count": "0",
"abstain_count": "0",
"no_count": "0",
},
"submit_time": "2022-03-28T14:02:41.165025015Z",
"deposit_end_time": "2022-03-30T14:02:41.165025015Z",
"total_deposit": [
{
"denom": "atone",
"amount": "10"
}
],
"voting_start_time": null,
"voting_end_time": null,
"metadata": "AQ==",
"title": "Proposal Title",
"summary": "Proposal Summary"
}
],
"pagination": {
"next_key": null,
"total": "2"
}
}
The votes
endpoint allows users to query a vote for a given proposal.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}/votes/{voter}
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1/votes/atone1..
Example Output:
{
"vote": {
"proposal_id": "1",
"voter": "atone1..",
"option": "VOTE_OPTION_YES",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
]
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}/votes/{voter}
Example:
curl localhost:1317/atomone/gov/v1/proposals/1/votes/atone1..
Example Output:
{
"vote": {
"proposal_id": "1",
"voter": "atone1..",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
],
"metadata": ""
}
}
The votes
endpoint allows users to query all votes for a given proposal.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}/votes
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1/votes
Example Output:
{
"votes": [
{
"proposal_id": "1",
"voter": "atone1..",
"option": "VOTE_OPTION_YES",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
]
}
],
"pagination": {
"next_key": null,
"total": "1"
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}/votes
Example:
curl localhost:1317/atomone/gov/v1/proposals/1/votes
Example Output:
{
"votes": [
{
"proposal_id": "1",
"voter": "atone1..",
"options": [
{
"option": "VOTE_OPTION_YES",
"weight": "1.000000000000000000"
}
],
"metadata": ""
}
],
"pagination": {
"next_key": null,
"total": "1"
}
}
The params
endpoint allows users to query all parameters for the gov
module.
Using legacy v1beta1:
/atomone/gov/v1beta1/params/{params_type}
Example:
curl localhost:1317/atomone/gov/v1beta1/params/voting
Example Output:
{
"voting_params": {
"voting_period": "172800s"
},
"deposit_params": {
"min_deposit": [
],
"max_deposit_period": "0s"
},
"tally_params": {
"quorum": "0.000000000000000000",
"threshold": "0.000000000000000000",
}
}
Using v1:
/atomone/gov/v1/params/{params_type}
Example:
curl localhost:1317/atomone/gov/v1/params/voting
Example Output:
{
"voting_params": {
"voting_period": "172800s"
},
"deposit_params": {
"min_deposit": [
],
"max_deposit_period": "0s"
},
"tally_params": {
"quorum": "0.000000000000000000",
"threshold": "0.000000000000000000",
}
}
The min_deposit
endpoint allows users to query the minimum deposit for a given proposal
to enter the voting period.
Using v1:
/atomone/gov/v1/mindeposit
Example:
curl localhost:1317/atomone/gov/v1/mindeposit
Example Output:
{
"min_deposit": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
The min_initial_deposit
endpoint allows users to query the minimum deposit for
a given proposal to enter the voting period.
Using v1:
/atomone/gov/v1/mininitialdeposit
Example:
curl localhost:1317/atomone/gov/v1/mininitialdeposit
Example Output:
{
"min_initial_deposit": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
The deposits
endpoint allows users to query a deposit for a given proposal from a given depositor.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}/deposits/{depositor}
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1/deposits/atone1..
Example Output:
{
"deposit": {
"proposal_id": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}/deposits/{depositor}
Example:
curl localhost:1317/atomone/gov/v1/proposals/1/deposits/atone1..
Example Output:
{
"deposit": {
"proposal_id": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
}
The deposits
endpoint allows users to query all deposits for a given proposal.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}/deposits
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1/deposits
Example Output:
{
"deposits": [
{
"proposal_id": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
],
"pagination": {
"next_key": null,
"total": "1"
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}/deposits
Example:
curl localhost:1317/atomone/gov/v1/proposals/1/deposits
Example Output:
{
"deposits": [
{
"proposal_id": "1",
"depositor": "atone1..",
"amount": [
{
"denom": "atone",
"amount": "10000000"
}
]
}
],
"pagination": {
"next_key": null,
"total": "1"
}
}
The tally
endpoint allows users to query the tally of a given proposal.
Using legacy v1beta1:
/atomone/gov/v1beta1/proposals/{proposal_id}/tally
Example:
curl localhost:1317/atomone/gov/v1beta1/proposals/1/tally
Example Output:
{
"tally": {
"yes": "1000000",
"abstain": "0",
"no": "0",
}
}
Using v1:
/atomone/gov/v1/proposals/{proposal_id}/tally
Example:
curl localhost:1317/atomone/gov/v1/proposals/1/tally
Example Output:
{
"tally": {
"yes": "1000000",
"abstain": "0",
"no": "0",
}
}
The gov module has two locations for metadata where users can provide further context about the on-chain actions they are taking. By default all metadata fields have a 255 character length field where metadata can be stored in json format, either on-chain or off-chain depending on the amount of data required. Here we provide a recommendation for the json structure and where the data should be stored. There are two important factors in making these recommendations. First, that the gov and group modules are consistent with one another, note the number of proposals made by all groups may be quite large. Second, that client applications such as block explorers and governance interfaces have confidence in the consistency of metadata structure accross chains.
Location: off-chain as json object stored on IPFS (mirrors group proposal)
{
"title": "",
"authors": [""],
"summary": "",
"details": "",
"proposal_forum_url": "",
"vote_option_context": "",
}
:::note
The authors
field is an array of strings, this is to allow for multiple authors
to be listed in the metadata.
In v0.46, the authors
field is a comma-separated string. Frontends are encouraged
to support both formats for backwards compatibility.
:::
Location: on-chain as json within 255 character limit (mirrors group vote)
{
"justification": "",
}
The current documentation only describes the minimum viable product for the governance module. Future improvements may include:
BountyProposals
: If accepted, aBountyProposal
creates an open bounty. TheBountyProposal
specifies how many Atones will be given upon completion. These Atones will be taken from thereserve pool
. After aBountyProposal
is accepted by governance, anybody can submit aSoftwareUpgradeProposal
with the code to claim the bounty. Note that once aBountyProposal
is accepted, the corresponding funds in thereserve pool
are locked so that payment can always be honored. In order to link aSoftwareUpgradeProposal
to an open bounty, the submitter of theSoftwareUpgradeProposal
will use theProposal.LinkedProposal
attribute. If aSoftwareUpgradeProposal
linked to an open bounty is accepted by governance, the funds that were reserved are automatically transferred to the submitter.- Complex delegation: Delegators could choose other representatives than their validators. Ultimately, the chain of representatives would always end up to a validator, but delegators could inherit the vote of their chosen representative before they inherit the vote of their validator. In other words, they would only inherit the vote of their validator if their other appointed representative did not vote.
- Better process for proposal review: There would be two parts to
proposal.Deposit
, one for anti-spam (same as in MVP) and an other one to reward third party auditors.