Gov

The Gov Contract contains logic for holding polls and Mirror Token (MIR) staking, and allows the Mirror Protocol to be governed by its users in a decentralized manner. After the initial bootstrapping of Mirror Protocol contracts, the Gov Contract is assigned to be the owner of itself and Mirror Factory.

New proposals for change are submitted as polls, and are voted on by MIR stakers through the voting procedure. Polls can contain messages that can be executed directly without changing the Mirror Protocol code.

The Gov Contract keeps a balance of MIR tokens, which it uses to reward stakers with funds it receives from trading fees sent by the Mirror Collector and user deposits from creating new governance polls. This balance is separate from the Community Pool, which is held by the Community contract (owned by the Gov contract).

Config

Key
Type
Description

mirror_token

HumanAddr

Contract address of Mirror Token (MIR)

quorum

Decimal

Minimum percentage of participation required for a poll to pass

threshold

Decimal

Minimum percentage of yes votes required for a poll to pass

voting_period

u64

Number of seconds during which votes can be cast

proposal_deposit

Uint128

MIR deposit required for a new poll to be submitted

effective_delay

u64

Number of seconds after a poll passes to apply changes

voter_weight

Decimal

Ratio of protocol fee which will be distributed among the governance poll voters

snapshot_period

u64

Minimum number of blocks before the end of voting period which snapshot could be taken to lock the current quorum for a poll

admin_manager

String

Address of the admin manager contract

poll_gas_limit

u64

Maximum amount of gas which a poll can consume during its execution

InitMsg

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
pub struct InstantiateMsg {
    pub mirror_token: String,
    pub effective_delay: u64,
    pub default_poll_config: PollConfig,
    pub migration_poll_config: PollConfig,
    pub auth_admin_poll_config: PollConfig,
    pub voter_weight: Decimal,
    pub snapshot_period: u64,
    pub admin_manager: String,
    pub poll_gas_limit: u64,
}

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub struct PollConfig {
    pub proposal_deposit: Uint128,
    pub voting_period: u64,
    pub quorum: Decimal,
    pub threshold: Decimal,
}
Key
Type
Description

mirror_token

HumanAddr

Contract address of Mirror Token (MIR)

effective_delay

u64

Minimum percentage of participation required for a poll to pass

default_poll_config

PollConfig

PollConfig for default poll types

migration_poll_config

PollConfig

PollConfig for migration poll types

auth_admin_poll_config

PollConfig

PollConfig for governance configuration change and admin key transfer polls

effective_delay

u64

Number of blocks after a poll passes to apply changes

voter_weight

Decimal

Ratio of protocol fee which will be distributed among the governance poll voters

snapshot_period

u64

Minimum number of blocks before the end of voting period which snapshot could be taken to lock the current quorum for a poll

admin_manager

String

Address of the admin manager contract

poll_gas_limit

u64

Maximum amount of gas which a poll can consume during its execution

PollConfig

Key
Type
Description

proposal_deposit

Uint128

MIR deposit required for new polls to be submitted

voting_period

u64

Number of seconds during which votes can be cast

quorum

Decimal

Minimum percentage of participation required for a poll to pass

threshold

Decimal

Minimum percentage of yes votes required for a poll to pass

ExecuteMsg

Receive

Can be called during a CW20 token transfer when the Gov contract is the recipient. Allows the token transfer to execute a Receive Hook as a subsequent action within the same transaction.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    Receive {
        amount: Uint128,
        sender: HumanAddr,
        msg: Option<Binary>,
    }
}
Key
Type
Description

amount

Uint128

Amount of tokens received

sender

HumanAddr

Sender of token transfer

msg*

Binary

* = optional

UpdateConfig

Updates the configuration for the Gov contract.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    UpdateConfig {
        owner: Option<HumanAddr>,
        quorum: Option<Decimal>,
        threshold: Option<Decimal>,
        voting_period: Option<u64>,
        effective_delay: Option<u64>,
        expiration_period: Option<u64>,
        proposal_deposit: Option<Uint128>,
        voter_weight: Option<Decimal>,
        snapshot_period: Option<u64>,
    }
}
Key
Type
Description

owner*

HumanAddr

Address of owner of governance contract

quorum*

Decimal

Minimum percentage of participation required for a poll to pass

threshold*

Decimal

Minimum percentage of yes votes required for a poll to pass

voting_period*

u64

Number of blocks during which votes can be cast

effective_delay*

u64

Number of blocks after a poll passes to apply changes

expiration_period*

u64

Number of blocks after a poll's voting period during which the poll can be executed

proposal_deposit*

Uint128

Minimum MIR deposit required for a new poll to be submitted

voter_weight*

Decimal

Ratio of protocol fee which will be distributed among the governance poll voters

snapshot_period*

u64

Minimum number of blocks before end of voting period which snapshot could be taken to lock the current quorum for a poll

* = optional

CastVote

Submits a user's vote for an active poll. Once a user has voted, they cannot change their vote with subsequent messages (increasing voting power, changing vote option, cancelling vote, etc.)

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    CastVote {
        poll_id: u64,
        vote: VoteOption,
        amount: Uint128,
    },
}
Key
Type
Description

amount

Uint128

Amount of voting power (staked MIR) to allocate

poll_id

u64

Poll ID

vote

VoteOption

Can be yes,no or abstain

WithdrawVotingTokens

Removes deposited MIR tokens from a staking position and returns them to a user's balance.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    WithdrawVotingTokens {
        amount: Option<Uint128>,
    }
}
Key
Type
Description

amount*

Uint128

Amount of MIR tokens to withdraw. If empty, all staked MIR tokens are withdrawn.

* = optional

WithdrawVotingRewards

Withdraws a user’s voting reward for user’s voted governance poll after end_poll has happened.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    WithdrawVotingRewards {},
}

StakeVotingRewards

Immediately re-stakes user's voting rewards to Gov Contract.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    StakeVotingRewards {},
}

EndPoll

Can be issued by anyone to end the voting for an active poll. Triggers tally the results to determine whether the poll has passed. The current block height must exceed the end height of voting phase.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    EndPoll {
        poll_id: u64,
    }
}
Key
Type
Description

poll_id

u64

Poll ID

ExecutePoll

Can be issued by anyone to implement into action the contents of a passed poll. The current block height must exceed the end height of the poll's effective delay.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    ExecutePoll {
        poll_id: u64,
    }
}
Key
Type
Description

poll_id

u64

Poll ID

SnapshotPoll

Snapshot of poll’s current quorum status is saved when the block height enters snapshot_period.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum HandleMsg {
    SnapshotPoll {
        poll_id: u64,
    }
}
Key
Type
Description

poll_id

u64

Poll ID

Receive Hooks

StakeVotingTokens

WARNING

If you send MIR tokens to the Gov contract without issuing this hook, they will not be staked and will be irrevocably donated to the reward pool for stakers.

Issued when sending MIR tokens to the Gov contract to add them to their MIR staking position.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum Cw20HookMsg {
    StakeVotingTokens {}
}

CreatePoll

Issued when sending MIR tokens to the Gov contract to create a new poll. Will only succeed if the amount of tokens sent meets the configuredproposal_depositamount. Contains a generic message to be issued by the Gov contract if it passes (can invoke messages in other contracts it owns).

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
#[allow(clippy::large_enum_variant)]
pub enum Cw20HookMsg {
   CreatePoll {
        title: String,
        description: String,
        link: Option<String>,
        execute_msg: Option<PollExecuteMsg>,
        admin_action: Option<PollAdminAction>,
    },

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub struct PollExecuteMsg {
    pub contract: String,
    pub msg: Binary,
}
Key
Type
Description

description

string

Poll description

execute_msg*

ExecuteMsg

Message to be executed by Gov contract

link*

string

URL to external post about poll (forum, PDF, etc.)

title

string

Poll title

admin_action*

PollAdminAction

Messages to be executed for migration and authorize polls

* = optional

DepositReward

Reward is distributed between MIR stakers and governance poll voters based on voter_weight when rewards are sent from Mirror Collector.

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum Cw20HookMsg {
    DepositReward {},
}

QueryMsg

Config

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    Config {}
}

Response

#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema)]
pub struct ConfigResponse {
    pub owner: String,
    pub mirror_token: String,
    pub effective_delay: u64,
    pub default_poll_config: PollConfig,
    pub migration_poll_config: PollConfig,
    pub auth_admin_poll_config: PollConfig,
    pub voter_weight: Decimal,
    pub snapshot_period: u64,
    pub admin_manager: String,
    pub poll_gas_limit: u64,
}
Key
Type
Description

mirror_token

HumanAddr

Contract address of Mirror Token (MIR)

default_poll_config

PollConfig

PollConfig for default polls

migration_poll_config

PollConfig

PollConfig for migration polls

auth_admin_poll_config

PollConfig

PollConfig for Authorize polls

effective_delay

u64

Number of blocks after a poll passes to apply changes

proposal_deposit

Uint128

Minimum MIR deposit required for a new poll to be submitted

voter_weight

Decimal

Ratio of protocol fee which will be distributed among the governance poll voters

snapshot_period

u64

Minimum number of blocks before end of voting period which snapshot could be taken to lock the current quorum for a poll

admin_manager

String

Address of admin manager contract

poll_gas_limit

u64

Maximum amount of gas which a poll can consume during its execution

State

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    State {}
}

Response

#[derive(Serialize, Deserialize, Clone, PartialEq, JsonSchema)]
pub struct StateResponse {
    pub poll_count: u64,
    pub total_share: Uint128,
    pub total_deposit: Uint128,
    pub pending_voting_rewards: Uint128,
}
Key
Type
Description

poll_count

u64

Total number of polls that have been created on Mirror Protocol

total_share

Uint128

Amount of staked MIR to governance contract

total_deposit

Uint128

Amount of locked MIR to governance polls

pending_voting_rewards

Uint128

Amount of voting rewards that are not claimed yet

Staker

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    Staker {
        address: HumanAddr,
    }
}
Key
Type
Description

address

HumanAddr

Address of staker

Response

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, JsonSchema)]
pub struct StakerResponse {
    pub balance: Uint128,
    pub share: Uint128,
    pub locked_balance: Vec<(u64, VoterInfo)>,
    pub pending_voting_rewards: Uint128,
}
Key
Type
Description

balance

Uint128

Amount of MIR staked by the user

share

Uint128

Weight of the user's staked MIR

locked_balance

Vec<(u64, VoterInfo)>

Total number of staked MIR used as votes, and chosen vote option

pending_voting_rewards

u64

Amount of voting rewards that are not claimed yet

Poll

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    Poll {
        poll_id: u64,
    }
}
Key
Type
Description

poll_id

u64

Poll ID

Response

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, JsonSchema)]
pub struct PollResponse {
    pub id: u64,
    pub creator: HumanAddr,
    pub status: PollStatus,
    pub end_height: u64,
    pub title: String,
    pub description: String,
    pub link: Option<String>,
    pub deposit_amount: Uint128,
    pub execute_data: Option<ExecuteMsg>,
    pub yes_votes: Uint128, // balance
    pub no_votes: Uint128,  // balance
    pub abstain_votes: Uint128, // balance
    pub total_balance_at_end_poll: Option<Uint128>,
    pub voters_reward: Uint128,
    pub staked_amount: Option<Uint128>,
}
Key
Type
Description

id

u64

Poll ID

creator

HumanAddr

Address of the poll creator

status

PollStatus

Could be one of "in progress", "rejected", "passed", "executed", "expired"

end_height

u64

Amount of voting rewards that are not claimed yet

title

String

Title of the poll

description

String

Description submitted by the creator

link*

Uint128

URL link

deposit_amount

binary

Initial MIR deposit at poll creation

execute_data*

ExecuteMsg

Message to be executed by Gov contract

yes_votes

Uint128

Amount of yes votes

no_votes

Uint128

Amount of no votes

abstain_votes

Uint128

Amount of abstain votes

total_balance_at_end_poll*

Uint128

Total balanced used as yes, no, or abstain votes at the end of the poll

voters_reward

Uint128

Amount of MIR reward accumulated to be distributed to the voters

staked_amount*

Uint128

Total number of MIR staked on governance contract (used when Snapshot Poll has been taken)

Polls

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    Polls {
        filter: Option<PollStatus>,
        limit: Option<u32>,
        start_after: Option<u64>,
    }
}
Key
Type
Description

filter*

PollStatus

Can be yes or no

limit*

u32

Limit of results to fetch

start_after*

u64

Begins search query at specific ID

* = optional

Response

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, JsonSchema)]
pub struct PollsResponse {
    pub polls: Vec<PollResponse>,
}
Key
Type
Description

polls

Vec<PollResponse>

Array of poll query responses

Voters

#[derive(Serialize, Deserialize, Clone, Debug, PartialEq, JsonSchema)]
#[serde(rename_all = "snake_case")]
pub enum QueryMsg {
    Voters {
        limit: Option<u32>,
        poll_id: u64,
        start_after: Option<HumanAddr>,
    }
}
Key
Type
Description

limit*

u32

Limit of results to fetch

poll_id

u64

Poll ID

start_after*

HumanAddr

Begins search query with prefix

* = optional

Response

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq, JsonSchema)]
pub struct VotersResponseItem {
    pub voter: HumanAddr,
    pub vote: VoteOption,
    pub balance: Uint128,
}
Key
Type
Description

voter

HumanAddr

Address of the voter

vote

VoteOption

Could be one of yes, no, abstain

balance

Uint128

Amount of staked MIR used for voting

Last updated