Skip to content

Latest commit

 

History

History
2330 lines (1455 loc) · 84.2 KB

File metadata and controls

2330 lines (1455 loc) · 84.2 KB

Module 0x1::Dao

Resource DaoGlobalInfo

global DAO info of the specified token type Token.

struct DaoGlobalInfo<Token: store> has key
Fields
next_proposal_id: u64
next proposal id.
proposal_create_event: Event::EventHandle<Dao::ProposalCreatedEvent>
proposal creating event.
vote_changed_event: Event::EventHandle<Dao::VoteChangedEvent>
voting event.

Struct DaoConfig

Configuration of the Token's DAO.

struct DaoConfig<TokenT: copy, drop, store> has copy, drop, store
Fields
voting_delay: u64
after proposal created, how long use should wait before he can vote (in milliseconds)
voting_period: u64
how long the voting window is (in milliseconds).
voting_quorum_rate: u8
the quorum rate to agree on the proposal. if 50% votes needed, then the voting_quorum_rate should be 50. it should between (0, 100].
min_action_delay: u64
how long the proposal should wait before it can be executed (in milliseconds).
Specification
invariant voting_quorum_rate > 0 && voting_quorum_rate <= 100;
invariant voting_delay > 0;
invariant voting_period > 0;
invariant min_action_delay > 0;

Struct ProposalCreatedEvent

emitted when proposal created.

struct ProposalCreatedEvent has drop, store
Fields
proposal_id: u64
the proposal id.
proposer: address
proposer is the user who create the proposal.

Struct VoteChangedEvent

emitted when user vote/revoke_vote.

struct VoteChangedEvent has drop, store
Fields
proposal_id: u64
the proposal id.
voter: address
the voter.
proposer: address
creator of the proposal.
agree: bool
agree with the proposal or not
vote: u128
latest vote count of the voter.

Resource Proposal

Proposal data struct.

struct Proposal<Token: store, Action: store> has key
Fields
id: u64
id of the proposal
proposer: address
creator of the proposal
start_time: u64
when voting begins.
end_time: u64
when voting ends.
for_votes: u128
count of voters who agree with the proposal
against_votes: u128
count of voters who're against the proposal
eta: u64
executable after this time.
action_delay: u64
after how long, the agreed proposal can be executed.
quorum_votes: u128
how many votes to reach to make the proposal pass.
action: Option::Option<Action>
proposal action.

Resource Vote

User vote info.

struct Vote<TokenT: store> has key
Fields
proposer: address
vote for the proposal under the proposer.
id: u64
proposal id.
stake: Token::Token<TokenT>
how many tokens to stake.
agree: bool
vote for or vote against.

Constants

const ERR_NOT_AUTHORIZED: u64 = 1401;

const ACTIVE: u8 = 2;

const AGREED: u8 = 4;

const DEFEATED: u8 = 3;

const ERR_ACTION_DELAY_TOO_SMALL: u64 = 1402;

const ERR_ACTION_MUST_EXIST: u64 = 1409;

const ERR_CONFIG_PARAM_INVALID: u64 = 1407;

const ERR_PROPOSAL_ID_MISMATCH: u64 = 1404;

const ERR_PROPOSAL_STATE_INVALID: u64 = 1403;

const ERR_PROPOSER_MISMATCH: u64 = 1405;

const ERR_QUORUM_RATE_INVALID: u64 = 1406;

const ERR_VOTED_OTHERS_ALREADY: u64 = 1410;

const ERR_VOTE_STATE_MISMATCH: u64 = 1408;

const EXECUTABLE: u8 = 6;

const EXTRACTED: u8 = 7;

Proposal state

const PENDING: u8 = 1;

const QUEUED: u8 = 5;

Function plugin

plugin function, can only be called by token issuer. Any token who wants to have gov functionality can optin this module by call this register function.

public fun plugin<TokenT: copy, drop, store>(signer: &signer, voting_delay: u64, voting_period: u64, voting_quorum_rate: u8, min_action_delay: u64)
Implementation
public fun plugin<TokenT: copy + drop + store>(
    signer: &signer,
    voting_delay: u64,
    voting_period: u64,
    voting_quorum_rate: u8,
    min_action_delay: u64,
) {
    let token_issuer = Token::token_address<TokenT>();
    assert!(Signer::address_of(signer) == token_issuer, Errors::requires_address(ERR_NOT_AUTHORIZED));
    // let proposal_id = ProposalId {next: 0};
    let gov_info = DaoGlobalInfo<TokenT> {
        next_proposal_id: 0,
        proposal_create_event: Event::new_event_handle<ProposalCreatedEvent>(signer),
        vote_changed_event: Event::new_event_handle<VoteChangedEvent>(signer),
    };
    move_to(signer, gov_info);
    let config = new_dao_config<TokenT>(
        voting_delay,
        voting_period,
        voting_quorum_rate,
        min_action_delay,
    );
    Config::publish_new_config(signer, config);
}
Specification
let sender = Signer::address_of(signer);
aborts_if sender != Token::SPEC_TOKEN_TEST_ADDRESS();
include NewDaoConfigParamSchema<TokenT>;
include Config::PublishNewConfigAbortsIf<DaoConfig<TokenT>>{account: signer};
aborts_if exists<DaoGlobalInfo<TokenT>>(sender);

schema RequirePluginDao<TokenT> {
    let token_addr = Token::SPEC_TOKEN_TEST_ADDRESS();
    aborts_if !exists<DaoGlobalInfo<TokenT>>(token_addr);
    aborts_if !exists<Config::Config<DaoConfig<TokenT>>>(token_addr);
}

schema AbortIfDaoInfoNotExist<TokenT> {
    let token_addr = Token::SPEC_TOKEN_TEST_ADDRESS();
    aborts_if !exists<DaoGlobalInfo<TokenT>>(token_addr);
}

schema AbortIfDaoConfigNotExist<TokenT> {
    let token_addr = Token::SPEC_TOKEN_TEST_ADDRESS();
    aborts_if !exists<Config::Config<DaoConfig<TokenT>>>(token_addr);
}

apply
    AbortIfDaoInfoNotExist<TokenT>
to
    generate_next_proposal_id<TokenT>;
apply
    AbortIfDaoConfigNotExist<TokenT>
to
    get_config<TokenT>,
    voting_delay<TokenT>,
    voting_period<TokenT>,
    voting_quorum_rate<TokenT>,
    min_action_delay<TokenT>,
    quorum_votes<TokenT>;

Function new_dao_config

create a dao config

public fun new_dao_config<TokenT: copy, drop, store>(voting_delay: u64, voting_period: u64, voting_quorum_rate: u8, min_action_delay: u64): Dao::DaoConfig<TokenT>
Implementation
public fun new_dao_config<TokenT: copy + drop + store>(
    voting_delay: u64,
    voting_period: u64,
    voting_quorum_rate: u8,
    min_action_delay: u64,
): DaoConfig<TokenT> {
    assert!(voting_delay > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    assert!(voting_period > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    assert!(
        voting_quorum_rate > 0 && voting_quorum_rate <= 100,
        Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID),
    );
    assert!(min_action_delay > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    DaoConfig { voting_delay, voting_period, voting_quorum_rate, min_action_delay }
}
Specification
include NewDaoConfigParamSchema<TokenT>;

schema NewDaoConfigParamSchema<TokenT> {
    voting_delay: u64;
    voting_period: u64;
    voting_quorum_rate: u8;
    min_action_delay: u64;
    aborts_if voting_delay == 0;
    aborts_if voting_period == 0;
    aborts_if voting_quorum_rate == 0 || voting_quorum_rate > 100;
    aborts_if min_action_delay == 0;
}

Function propose

propose a proposal. action: the actual action to execute. action_delay: the delay to execute after the proposal is agreed

public fun propose<TokenT: copy, drop, store, ActionT: copy, drop, store>(signer: &signer, action: ActionT, action_delay: u64)
Implementation
public fun propose<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    signer: &signer,
    action: ActionT,
    action_delay: u64,
) acquires DaoGlobalInfo {
    if (action_delay == 0) {
        action_delay = min_action_delay<TokenT>();
    } else {
        assert!(action_delay >= min_action_delay<TokenT>(), Errors::invalid_argument(ERR_ACTION_DELAY_TOO_SMALL));
    };
    let proposal_id = generate_next_proposal_id<TokenT>();
    let proposer = Signer::address_of(signer);
    let start_time = Timestamp::now_milliseconds() + voting_delay<TokenT>();
    let quorum_votes = quorum_votes<TokenT>();
    let proposal = Proposal<TokenT, ActionT> {
        id: proposal_id,
        proposer,
        start_time,
        end_time: start_time + voting_period<TokenT>(),
        for_votes: 0,
        against_votes: 0,
        eta: 0,
        action_delay,
        quorum_votes,
        action: Option::some(action),
    };
    move_to(signer, proposal);
    // emit event
    let gov_info = borrow_global_mut<DaoGlobalInfo<TokenT>>(Token::token_address<TokenT>());
    Event::emit_event(
        &mut gov_info.proposal_create_event,
        ProposalCreatedEvent { proposal_id, proposer },
    );
}
Specification
pragma verify = false;
let proposer = Signer::address_of(signer);
include GenerateNextProposalIdSchema<TokenT>;
pragma addition_overflow_unchecked = true;
include AbortIfDaoConfigNotExist<TokenT>;
include AbortIfDaoInfoNotExist<TokenT>;
aborts_if !exists<Timestamp::CurrentTimeMilliseconds>(CoreAddresses::GENESIS_ADDRESS());
aborts_if action_delay > 0 && action_delay < spec_dao_config<TokenT>().min_action_delay;
include CheckQuorumVotes<TokenT>;
let sender = Signer::address_of(signer);
aborts_if exists<Proposal<TokenT, ActionT>>(sender);
modifies global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS());
ensures exists<Proposal<TokenT, ActionT>>(sender);

Function cast_vote

votes for a proposal. User can only vote once, then the stake is locked, which can only be unstaked by user after the proposal is expired, or cancelled, or executed. So think twice before casting vote.

public fun cast_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(signer: &signer, proposer_address: address, proposal_id: u64, stake: Token::Token<TokenT>, agree: bool)
Implementation
public fun cast_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    signer: &signer,
    proposer_address: address,
    proposal_id: u64,
    stake: Token::Token<TokenT>,
    agree: bool,
) acquires Proposal, DaoGlobalInfo, Vote {
    {
        let state = proposal_state<TokenT, ActionT>(proposer_address, proposal_id);
        // only when proposal is active, use can cast vote.
        assert!(state == ACTIVE, Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID));
    };
    let proposal = borrow_global_mut<Proposal<TokenT, ActionT>>(proposer_address);
    assert!(proposal.id == proposal_id, Errors::invalid_argument(ERR_PROPOSAL_ID_MISMATCH));
    let sender = Signer::address_of(signer);
    let total_voted = if (exists<Vote<TokenT>>(sender)) {
        let my_vote = borrow_global_mut<Vote<TokenT>>(sender);
        assert!(my_vote.id == proposal_id, Errors::invalid_argument(ERR_VOTED_OTHERS_ALREADY));
        assert!(my_vote.agree == agree, Errors::invalid_state(ERR_VOTE_STATE_MISMATCH));

        do_cast_vote(proposal, my_vote, stake);
        Token::value(&my_vote.stake)
    } else {
        let my_vote = Vote<TokenT> {
            proposer: proposer_address,
            id: proposal_id,
            stake: Token::zero(),
            agree,
        };
        do_cast_vote(proposal, &mut my_vote, stake);
        let total_voted = Token::value(&my_vote.stake);
        move_to(signer, my_vote);
        total_voted
    };

    // emit event
    let gov_info = borrow_global_mut<DaoGlobalInfo<TokenT>>(Token::token_address<TokenT>());
    Event::emit_event(
        &mut gov_info.vote_changed_event,
        VoteChangedEvent {
            proposal_id,
            proposer: proposer_address,
            voter: sender,
            agree,
            vote: total_voted,
        },
    );
}
Specification
pragma addition_overflow_unchecked = true;
include AbortIfDaoInfoNotExist<TokenT>;
let expected_states = vec(ACTIVE);
include CheckProposalStates<TokenT, ActionT> {expected_states};
let sender = Signer::address_of(signer);
let vote_exists = exists<Vote<TokenT>>(sender);
include vote_exists ==> CheckVoteOnCast<TokenT, ActionT> {
    voter: sender,
    proposal_id: proposal_id,
    agree: agree,
    stake_value: stake.value,
};
modifies global<Proposal<TokenT, ActionT>>(proposer_address);
ensures !vote_exists ==> global<Vote<TokenT>>(sender).stake.value == stake.value;

Function do_cast_vote

fun do_cast_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposal: &mut Dao::Proposal<TokenT, ActionT>, vote: &mut Dao::Vote<TokenT>, stake: Token::Token<TokenT>)
Implementation
fun do_cast_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(proposal: &mut Proposal<TokenT, ActionT>, vote: &mut Vote<TokenT>, stake: Token::Token<TokenT>) {
    let stake_value = Token::value(&stake);
    Token::deposit(&mut vote.stake, stake);
    if (vote.agree) {
        proposal.for_votes = proposal.for_votes + stake_value;
    } else {
        proposal.against_votes = proposal.against_votes + stake_value;
    };
}
Specification
pragma addition_overflow_unchecked = true;
aborts_if vote.stake.value + stake.value > MAX_U128;
ensures vote.stake.value == old(vote).stake.value + stake.value;
ensures vote.agree ==> old(proposal).for_votes + stake.value == proposal.for_votes;
ensures vote.agree ==> old(proposal).against_votes == proposal.against_votes;
ensures !vote.agree ==> old(proposal).against_votes + stake.value == proposal.against_votes;
ensures !vote.agree ==> old(proposal).for_votes == proposal.for_votes;

Function change_vote

Let user change their vote during the voting time.

public fun change_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(signer: &signer, proposer_address: address, proposal_id: u64, agree: bool)
Implementation
public fun change_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    signer: &signer,
    proposer_address: address,
    proposal_id: u64,
    agree: bool,
) acquires Proposal, DaoGlobalInfo, Vote {
    {
        let state = proposal_state<TokenT, ActionT>(proposer_address, proposal_id);
        // only when proposal is active, user can change vote.
        assert!(state == ACTIVE, Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID));
    };
    let proposal = borrow_global_mut<Proposal<TokenT, ActionT>>(proposer_address);
    assert!(proposal.id == proposal_id, Errors::invalid_argument(ERR_PROPOSAL_ID_MISMATCH));
    let my_vote = borrow_global_mut<Vote<TokenT>>(Signer::address_of(signer));
    {
        assert!(my_vote.proposer == proposer_address, Errors::invalid_argument(ERR_PROPOSER_MISMATCH));
        assert!(my_vote.id == proposal_id, Errors::invalid_argument(ERR_VOTED_OTHERS_ALREADY));
    };

    // flip the vote
    if (my_vote.agree != agree) {
        let total_voted = do_flip_vote(my_vote, proposal);
        // emit event
        let gov_info = borrow_global_mut<DaoGlobalInfo<TokenT>>(Token::token_address<TokenT>());
        Event::emit_event(
            &mut gov_info.vote_changed_event,
            VoteChangedEvent {
                proposal_id,
                proposer: proposer_address,
                voter: Signer::address_of(signer),
                agree,
                vote: total_voted,
            },
        );
    };
}
Specification
pragma verify = false;
let expected_states = vec(ACTIVE);
include CheckProposalStates<TokenT, ActionT>{expected_states};
let sender = Signer::address_of(signer);
aborts_if !exists<Vote<TokenT>>(sender);
let vote = global<Vote<TokenT>>(sender);
include CheckVoteOnProposal<TokenT>{vote, proposer_address, proposal_id};
include vote.agree != agree ==> CheckChangeVote<TokenT, ActionT>{vote, proposer_address};
ensures vote.agree != agree ==> vote.agree == agree;

Function do_flip_vote

fun do_flip_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(my_vote: &mut Dao::Vote<TokenT>, proposal: &mut Dao::Proposal<TokenT, ActionT>): u128
Implementation
fun do_flip_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(my_vote: &mut Vote<TokenT>, proposal: &mut Proposal<TokenT, ActionT>): u128 {
    my_vote.agree = !my_vote.agree;
    let total_voted = Token::value(&my_vote.stake);
    if (my_vote.agree) {
        proposal.for_votes = proposal.for_votes + total_voted;
        proposal.against_votes = proposal.against_votes - total_voted;
    } else {
        proposal.for_votes = proposal.for_votes - total_voted;
        proposal.against_votes = proposal.against_votes + total_voted;
    };
    total_voted
}
Specification
include CheckFlipVote<TokenT, ActionT>;
ensures my_vote.agree == !old(my_vote).agree;

Function revoke_vote

Revoke some voting powers from vote on proposal_id of proposer_address.

public fun revoke_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(signer: &signer, proposer_address: address, proposal_id: u64, voting_power: u128): Token::Token<TokenT>
Implementation
public fun revoke_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    signer: &signer,
    proposer_address: address,
    proposal_id: u64,
    voting_power: u128,
): Token::Token<TokenT> acquires Proposal, Vote, DaoGlobalInfo {
    {
        let state = proposal_state<TokenT, ActionT>(proposer_address, proposal_id);
        // only when proposal is active, user can revoke vote.
        assert!(state == ACTIVE, Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID));
    };
    // get proposal
    let proposal = borrow_global_mut<Proposal<TokenT, ActionT>>(proposer_address);

    // get vote
    let my_vote = move_from<Vote<TokenT>>(Signer::address_of(signer));
    {
        assert!(my_vote.proposer == proposer_address, Errors::invalid_argument(ERR_PROPOSER_MISMATCH));
        assert!(my_vote.id == proposal_id, Errors::invalid_argument(ERR_VOTED_OTHERS_ALREADY));
    };
    // revoke vote on proposal
    let reverted_stake =do_revoke_vote(proposal, &mut my_vote, voting_power);
    // emit vote changed event
    let gov_info = borrow_global_mut<DaoGlobalInfo<TokenT>>(Token::token_address<TokenT>());
    Event::emit_event(
        &mut gov_info.vote_changed_event,
        VoteChangedEvent {
            proposal_id,
            proposer: proposer_address,
            voter: Signer::address_of(signer),
            agree: my_vote.agree,
            vote: Token::value(&my_vote.stake),
        },
    );

    // if user has no stake, destroy his vote. resolve https://github.com/starcoinorg/starcoin/issues/2925.
    if (Token::value(&my_vote.stake) == 0u128) {
        let Vote {stake, proposer: _, id: _, agree: _} = my_vote;
        Token::destroy_zero(stake);
    } else {
        move_to(signer, my_vote);
    };

    reverted_stake
}
Specification
pragma verify = false;
include AbortIfDaoInfoNotExist<TokenT>;
let expected_states = vec(ACTIVE);
include CheckProposalStates<TokenT, ActionT> {expected_states};
let sender = Signer::address_of(signer);
aborts_if !exists<Vote<TokenT>>(sender);
let vote = global<Vote<TokenT>>(sender);
include CheckVoteOnProposal<TokenT> {vote, proposer_address, proposal_id};
include CheckRevokeVote<TokenT, ActionT> {
    vote,
    proposal: global<Proposal<TokenT, ActionT>>(proposer_address),
    to_revoke: voting_power,
};
modifies global<Vote<TokenT>>(sender);
modifies global<Proposal<TokenT, ActionT>>(proposer_address);
modifies global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS());
ensures global<Vote<TokenT>>(sender).stake.value + result.value == old(global<Vote<TokenT>>(sender)).stake.value;
ensures result.value == voting_power;

Function do_revoke_vote

fun do_revoke_vote<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposal: &mut Dao::Proposal<TokenT, ActionT>, vote: &mut Dao::Vote<TokenT>, to_revoke: u128): Token::Token<TokenT>
Implementation
fun do_revoke_vote<TokenT: copy + drop + store, ActionT: copy + drop + store>(proposal: &mut Proposal<TokenT, ActionT>, vote: &mut Vote<TokenT>, to_revoke: u128): Token::Token<TokenT> {
    spec {
        assume vote.stake.value >= to_revoke;
    };
    let reverted_stake = Token::withdraw(&mut vote.stake, to_revoke);
    if (vote.agree) {
        proposal.for_votes = proposal.for_votes - to_revoke;
    } else {
        proposal.against_votes = proposal.against_votes - to_revoke;
    };
    spec {
        assert Token::value(reverted_stake) == to_revoke;
    };
    reverted_stake
}
Specification
include CheckRevokeVote<TokenT, ActionT>;
ensures vote.agree ==> old(proposal).for_votes == proposal.for_votes + to_revoke;
ensures !vote.agree ==> old(proposal).against_votes == proposal.against_votes + to_revoke;
ensures result.value == to_revoke;

Function unstake_votes

Retrieve back my staked token voted for a proposal.

public fun unstake_votes<TokenT: copy, drop, store, ActionT: copy, drop, store>(signer: &signer, proposer_address: address, proposal_id: u64): Token::Token<TokenT>
Implementation
public fun unstake_votes<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    signer: &signer,
    proposer_address: address,
    proposal_id: u64,
): Token::Token<TokenT> acquires Proposal, Vote {
    // only check state when proposal exists.
    // because proposal can be destroyed after it ends in DEFEATED or EXTRACTED state.
    if (proposal_exists<TokenT, ActionT>(proposer_address, proposal_id)) {
        let state = proposal_state<TokenT, ActionT>(proposer_address, proposal_id);
        // Only after vote period end, user can unstake his votes.
        assert!(state > ACTIVE, Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID));
    };
    let Vote { proposer, id, stake, agree: _ } = move_from<Vote<TokenT>>(
        Signer::address_of(signer),
    );
    // these checks are still required.
    assert!(proposer == proposer_address, Errors::requires_address(ERR_PROPOSER_MISMATCH));
    assert!(id == proposal_id, Errors::invalid_argument(ERR_VOTED_OTHERS_ALREADY));
    stake
}
Specification
pragma verify = false;
let expected_states = vec(DEFEATED);
let expected_states1 = concat(expected_states,vec(AGREED));
let expected_states2 = concat(expected_states1,vec(QUEUED));
let expected_states3 = concat(expected_states2,vec(EXECUTABLE));
let expected_states4 = concat(expected_states3,vec(EXTRACTED));
aborts_if expected_states4[0] != DEFEATED;
aborts_if expected_states4[1] != AGREED;
aborts_if expected_states4[2] != QUEUED;
aborts_if expected_states4[3] != EXECUTABLE;
aborts_if expected_states4[4] != EXTRACTED;
include spec_proposal_exists<TokenT, ActionT>(proposer_address, proposal_id) ==>
            CheckProposalStates<TokenT, ActionT>{expected_states: expected_states4};
let sender = Signer::address_of(signer);
aborts_if !exists<Vote<TokenT>>(sender);
let vote = global<Vote<TokenT>>(sender);
include CheckVoteOnProposal<TokenT>{vote, proposer_address, proposal_id};
ensures !exists<Vote<TokenT>>(sender);
ensures result.value == old(vote).stake.value;

Function queue_proposal_action

queue agreed proposal to execute.

public entry fun queue_proposal_action<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address, proposal_id: u64)
Implementation
public entry fun queue_proposal_action<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
    proposal_id: u64,
) acquires Proposal {
    // Only agreed proposal can be submitted.
    assert!(
        proposal_state<TokenT, ActionT>(proposer_address, proposal_id) == AGREED,
        Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID)
    );
    let proposal = borrow_global_mut<Proposal<TokenT, ActionT>>(proposer_address);
    proposal.eta = Timestamp::now_milliseconds() + proposal.action_delay;
}
Specification
pragma verify = false;
let expected_states = vec(AGREED);
include CheckProposalStates<TokenT, ActionT>{expected_states};
let proposal = global<Proposal<TokenT, ActionT>>(proposer_address);
aborts_if Timestamp::spec_now_millseconds() + proposal.action_delay > MAX_U64;
ensures proposal.eta >= Timestamp::spec_now_millseconds();

Function extract_proposal_action

extract proposal action to execute.

public fun extract_proposal_action<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address, proposal_id: u64): ActionT
Implementation
public fun extract_proposal_action<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
    proposal_id: u64,
): ActionT acquires Proposal {
    // Only executable proposal's action can be extracted.
    assert!(
        proposal_state<TokenT, ActionT>(proposer_address, proposal_id) == EXECUTABLE,
        Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID),
    );
    let proposal = borrow_global_mut<Proposal<TokenT, ActionT>>(proposer_address);
    let action: ActionT = Option::extract(&mut proposal.action);
    action
}
Specification
pragma aborts_if_is_partial = false;
let expected_states = vec(EXECUTABLE);
include CheckProposalStates<TokenT, ActionT>{expected_states};
modifies global<Proposal<TokenT, ActionT>>(proposer_address);
ensures Option::is_none(global<Proposal<TokenT, ActionT>>(proposer_address).action);

Function destroy_terminated_proposal

remove terminated proposal from proposer

public entry fun destroy_terminated_proposal<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address, proposal_id: u64)
Implementation
public entry fun destroy_terminated_proposal<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
    proposal_id: u64,
) acquires Proposal {
    let proposal_state = proposal_state<TokenT, ActionT>(proposer_address, proposal_id);
    assert!(
        proposal_state == DEFEATED || proposal_state == EXTRACTED,
        Errors::invalid_state(ERR_PROPOSAL_STATE_INVALID),
    );
    let Proposal {
        id: _,
        proposer: _,
        start_time: _,
        end_time: _,
        for_votes: _,
        against_votes: _,
        eta: _,
        action_delay: _,
        quorum_votes: _,
        action,
    } = move_from<Proposal<TokenT, ActionT>>(proposer_address);
    if (proposal_state == DEFEATED) {
        let _ = Option::extract(&mut action);
    };
    Option::destroy_none(action);
}
Specification
let expected_states = concat(vec(DEFEATED), vec(EXTRACTED));
aborts_if len(expected_states) != 2;
aborts_if expected_states[0] != DEFEATED;
aborts_if expected_states[1] != EXTRACTED;
aborts_if !exists<Proposal<TokenT, ActionT>>(proposer_address);
let proposal = global<Proposal<TokenT, ActionT>>(proposer_address);
aborts_if proposal.id != proposal_id;
include AbortIfTimestampNotExist;
let current_time = Timestamp::spec_now_millseconds();
let state = do_proposal_state(proposal, current_time);
aborts_if (forall s in expected_states : s != state);
aborts_if state == DEFEATED && Option::is_none(global<Proposal<TokenT, ActionT>>(proposer_address).action);
aborts_if state == EXTRACTED && Option::is_some(global<Proposal<TokenT, ActionT>>(proposer_address).action);
modifies global<Proposal<TokenT, ActionT>>(proposer_address);

Function proposal_exists

check whether a proposal exists in proposer_address with id proposal_id.

public fun proposal_exists<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address, proposal_id: u64): bool
Implementation
public fun proposal_exists<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
    proposal_id: u64,
): bool acquires Proposal {
    if (exists<Proposal<TokenT, ActionT>>(proposer_address)) {
        let proposal = borrow_global<Proposal<TokenT, ActionT>>(proposer_address);
        return proposal.id == proposal_id
    };
    false
}
Specification
ensures exists<Proposal<TokenT, ActionT>>(proposer_address) &&
            borrow_global<Proposal<TokenT, ActionT>>(proposer_address).id == proposal_id ==>
            result;

fun spec_proposal_exists<TokenT: copy + drop + store, ActionT: copy + drop + store>(
   proposer_address: address,
   proposal_id: u64,
): bool {
   if (exists<Proposal<TokenT, ActionT>>(proposer_address)) {
       let proposal = global<Proposal<TokenT, ActionT>>(proposer_address);
       proposal.id == proposal_id
   } else {
       false
   }
}

Function proposal_state

Get the proposal state.

public fun proposal_state<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address, proposal_id: u64): u8
Implementation
public fun proposal_state<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
    proposal_id: u64,
): u8 acquires Proposal {
    let proposal = borrow_global<Proposal<TokenT, ActionT>>(proposer_address);
    assert!(proposal.id == proposal_id, Errors::invalid_argument(ERR_PROPOSAL_ID_MISMATCH));
    let current_time = Timestamp::now_milliseconds();
    do_proposal_state(proposal, current_time)
}
Specification
include AbortIfTimestampNotExist;
aborts_if !exists<Timestamp::CurrentTimeMilliseconds>(CoreAddresses::GENESIS_ADDRESS());
aborts_if !exists<Proposal<TokenT, ActionT>>(proposer_address);
let proposal = global<Proposal<TokenT, ActionT>>(proposer_address);
aborts_if proposal.id != proposal_id;

Function do_proposal_state

fun do_proposal_state<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposal: &Dao::Proposal<TokenT, ActionT>, current_time: u64): u8
Implementation
fun do_proposal_state<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposal: &Proposal<TokenT, ActionT>,
    current_time: u64,
): u8 {
    if (current_time < proposal.start_time) {
        // Pending
        PENDING
    } else if (current_time <= proposal.end_time) {
        // Active
        ACTIVE
    } else if (proposal.for_votes <= proposal.against_votes ||
        proposal.for_votes < proposal.quorum_votes) {
        // Defeated
        DEFEATED
    } else if (proposal.eta == 0) {
        // Agreed.
        AGREED
    } else if (current_time < proposal.eta) {
        // Queued, waiting to execute
        QUEUED
    } else if (Option::is_some(&proposal.action)) {
        EXECUTABLE
    } else {
        EXTRACTED
    }
}

Function proposal_info

get proposal's information. return: (id, start_time, end_time, for_votes, against_votes).

public fun proposal_info<TokenT: copy, drop, store, ActionT: copy, drop, store>(proposer_address: address): (u64, u64, u64, u128, u128)
Implementation
public fun proposal_info<TokenT: copy + drop + store, ActionT: copy + drop + store>(
    proposer_address: address,
): (u64, u64, u64, u128, u128) acquires Proposal {
    let proposal = borrow_global<Proposal<TokenT, ActionT>>(proposer_address);
    (proposal.id, proposal.start_time, proposal.end_time, proposal.for_votes, proposal.against_votes)
}
Specification
aborts_if !exists<Proposal<TokenT, ActionT>>(proposer_address);

Function vote_of

Get voter's vote info on proposal with proposal_id of proposer_address.

public fun vote_of<TokenT: copy, drop, store>(voter: address, proposer_address: address, proposal_id: u64): (bool, u128)
Implementation
public fun vote_of<TokenT: copy + drop + store>(
    voter: address,
    proposer_address: address,
    proposal_id: u64,
): (bool, u128) acquires Vote {
    let vote = borrow_global<Vote<TokenT>>(voter);
    assert!(vote.proposer == proposer_address, Errors::requires_address(ERR_PROPOSER_MISMATCH));
    assert!(vote.id == proposal_id, Errors::invalid_argument(ERR_VOTED_OTHERS_ALREADY));
    (vote.agree, Token::value(&vote.stake))
}
Specification
aborts_if !exists<Vote<TokenT>>(voter);
let vote = global<Vote<TokenT>>(voter);
include CheckVoteOnProposal<TokenT>{vote, proposer_address, proposal_id};

Function has_vote

Check whether voter has voted on proposal with proposal_id of proposer_address.

public fun has_vote<TokenT: copy, drop, store>(voter: address, proposer_address: address, proposal_id: u64): bool
Implementation
public fun has_vote<TokenT: copy + drop + store>(
    voter: address,
    proposer_address: address,
    proposal_id: u64,
): bool acquires Vote {
    if (!exists<Vote<TokenT>>(voter)) {
        return false
    };

    let vote = borrow_global<Vote<TokenT>>(voter);
    vote.proposer == proposer_address && vote.id == proposal_id
}

Function generate_next_proposal_id

fun generate_next_proposal_id<TokenT: store>(): u64
Implementation
fun generate_next_proposal_id<TokenT: store>(): u64 acquires DaoGlobalInfo {
    let gov_info = borrow_global_mut<DaoGlobalInfo<TokenT>>(Token::token_address<TokenT>());
    let proposal_id = gov_info.next_proposal_id;
    gov_info.next_proposal_id = proposal_id + 1;
    proposal_id
}
Specification
include GenerateNextProposalIdSchema<TokenT>;
ensures result == old(global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS()).next_proposal_id);

schema GenerateNextProposalIdSchema<TokenT> {
    aborts_if global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS()).next_proposal_id >= MAX_U64;
    modifies global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS());
    ensures
        global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS()).next_proposal_id ==
        old(global<DaoGlobalInfo<TokenT>>(Token::SPEC_TOKEN_TEST_ADDRESS()).next_proposal_id) + 1;
}

Function voting_delay

get default voting delay of the DAO.

public fun voting_delay<TokenT: copy, drop, store>(): u64
Implementation
public fun voting_delay<TokenT: copy + drop + store>(): u64 {
    get_config<TokenT>().voting_delay
}
Specification
aborts_if false;

Function voting_period

get the default voting period of the DAO.

public fun voting_period<TokenT: copy, drop, store>(): u64
Implementation
public fun voting_period<TokenT: copy + drop + store>(): u64 {
    get_config<TokenT>().voting_period
}
Specification
aborts_if false;

Function quorum_votes

Quorum votes to make proposal pass.

public fun quorum_votes<TokenT: copy, drop, store>(): u128
Implementation
public fun quorum_votes<TokenT: copy + drop + store>(): u128 {
    let market_cap = Token::market_cap<TokenT>();
    let balance_in_treasury = Treasury::balance<TokenT>();
    let supply = market_cap - balance_in_treasury;
    let rate = voting_quorum_rate<TokenT>();
    let rate = (rate as u128);
    supply * rate / 100
}
Specification
pragma verify = false;
include CheckQuorumVotes<TokenT>;

fun spec_quorum_votes<TokenT: copy + drop + store>(): u128 {
   let supply = Token::spec_abstract_total_value<TokenT>() - Treasury::spec_balance<TokenT>();
   supply * spec_dao_config<TokenT>().voting_quorum_rate / 100
}

Function voting_quorum_rate

Get the quorum rate in percent.

public fun voting_quorum_rate<TokenT: copy, drop, store>(): u8
Implementation
public fun voting_quorum_rate<TokenT: copy + drop + store>(): u8 {
    get_config<TokenT>().voting_quorum_rate
}
Specification
aborts_if false;
ensures result == global<Config::Config<DaoConfig<TokenT>>>((Token::SPEC_TOKEN_TEST_ADDRESS())).payload.voting_quorum_rate;

Function min_action_delay

Get the min_action_delay of the DAO.

public fun min_action_delay<TokenT: copy, drop, store>(): u64
Implementation
public fun min_action_delay<TokenT: copy + drop + store>(): u64 {
    get_config<TokenT>().min_action_delay
}
Specification
aborts_if false;
ensures result == spec_dao_config<TokenT>().min_action_delay;

Function get_config

fun get_config<TokenT: copy, drop, store>(): Dao::DaoConfig<TokenT>
Implementation
fun get_config<TokenT: copy + drop + store>(): DaoConfig<TokenT> {
    let token_issuer = Token::token_address<TokenT>();
    Config::get_by_address<DaoConfig<TokenT>>(token_issuer)
}
Specification
aborts_if false;
ensures result == global<Config::Config<DaoConfig<TokenT>>>(Token::SPEC_TOKEN_TEST_ADDRESS()).payload;

fun spec_dao_config<TokenT: copy + drop + store>(): DaoConfig<TokenT> {
   global<Config::Config<DaoConfig<TokenT>>>((Token::SPEC_TOKEN_TEST_ADDRESS())).payload
}

schema CheckModifyConfigWithCap<TokenT> {
    cap: Config::ModifyConfigCapability<DaoConfig<TokenT>>;
    aborts_if cap.account_address != Token::SPEC_TOKEN_TEST_ADDRESS();
    aborts_if !exists<Config::Config<DaoConfig<TokenT>>>(cap.account_address);
}

Function modify_dao_config

update function, modify dao config. if any param is 0, it means no change to that param.

public fun modify_dao_config<TokenT: copy, drop, store>(cap: &mut Config::ModifyConfigCapability<Dao::DaoConfig<TokenT>>, voting_delay: u64, voting_period: u64, voting_quorum_rate: u8, min_action_delay: u64)
Implementation
public fun modify_dao_config<TokenT: copy + drop + store>(
    cap: &mut Config::ModifyConfigCapability<DaoConfig<TokenT>>,
    voting_delay: u64,
    voting_period: u64,
    voting_quorum_rate: u8,
    min_action_delay: u64,
) {
    assert!(Config::account_address(cap) == Token::token_address<TokenT>(), Errors::invalid_argument(ERR_NOT_AUTHORIZED));
    let config = get_config<TokenT>();
    if (voting_period > 0) {
        config.voting_period = voting_period;
    };
    if (voting_delay > 0) {
        config.voting_delay = voting_delay;
    };
    if (voting_quorum_rate > 0) {
        assert!(voting_quorum_rate <= 100, Errors::invalid_argument(ERR_QUORUM_RATE_INVALID));
        config.voting_quorum_rate = voting_quorum_rate;
    };
    if (min_action_delay > 0) {
        config.min_action_delay = min_action_delay;
    };
    Config::set_with_capability<DaoConfig<TokenT>>(cap, config);
}
Specification
include CheckModifyConfigWithCap<TokenT>;
aborts_if voting_quorum_rate > 0 && voting_quorum_rate > 100;

Function set_voting_delay

set voting delay

public fun set_voting_delay<TokenT: copy, drop, store>(cap: &mut Config::ModifyConfigCapability<Dao::DaoConfig<TokenT>>, value: u64)
Implementation
public fun set_voting_delay<TokenT: copy + drop + store>(
    cap: &mut Config::ModifyConfigCapability<DaoConfig<TokenT>>,
    value: u64,
) {
    assert!(Config::account_address(cap) == Token::token_address<TokenT>(), Errors::invalid_argument(ERR_NOT_AUTHORIZED));
    assert!(value > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    let config = get_config<TokenT>();
    config.voting_delay = value;
    Config::set_with_capability<DaoConfig<TokenT>>(cap, config);
}
Specification
include CheckModifyConfigWithCap<TokenT>;
aborts_if value == 0;

Function set_voting_period

set voting period

public fun set_voting_period<TokenT: copy, drop, store>(cap: &mut Config::ModifyConfigCapability<Dao::DaoConfig<TokenT>>, value: u64)
Implementation
public fun set_voting_period<TokenT: copy + drop + store>(
    cap: &mut Config::ModifyConfigCapability<DaoConfig<TokenT>>,
    value: u64,
) {
    assert!(Config::account_address(cap) == Token::token_address<TokenT>(), Errors::invalid_argument(ERR_NOT_AUTHORIZED));
    assert!(value > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    let config = get_config<TokenT>();
    config.voting_period = value;
    Config::set_with_capability<DaoConfig<TokenT>>(cap, config);
}
Specification
include CheckModifyConfigWithCap<TokenT>;
aborts_if value == 0;

Function set_voting_quorum_rate

set voting quorum rate

public fun set_voting_quorum_rate<TokenT: copy, drop, store>(cap: &mut Config::ModifyConfigCapability<Dao::DaoConfig<TokenT>>, value: u8)
Implementation
public fun set_voting_quorum_rate<TokenT: copy + drop + store>(
    cap: &mut Config::ModifyConfigCapability<DaoConfig<TokenT>>,
    value: u8,
) {
    assert!(Config::account_address(cap) == Token::token_address<TokenT>(), Errors::invalid_argument(ERR_NOT_AUTHORIZED));
    assert!(value <= 100 && value > 0, Errors::invalid_argument(ERR_QUORUM_RATE_INVALID));
    let config = get_config<TokenT>();
    config.voting_quorum_rate = value;
    Config::set_with_capability<DaoConfig<TokenT>>(cap, config);
}
Specification
aborts_if !(value > 0 && value <= 100);
include CheckModifyConfigWithCap<TokenT>;

Function set_min_action_delay

set min action delay

public fun set_min_action_delay<TokenT: copy, drop, store>(cap: &mut Config::ModifyConfigCapability<Dao::DaoConfig<TokenT>>, value: u64)
Implementation
public fun set_min_action_delay<TokenT: copy + drop + store>(
    cap: &mut Config::ModifyConfigCapability<DaoConfig<TokenT>>,
    value: u64,
) {
    assert!(Config::account_address(cap) == Token::token_address<TokenT>(), Errors::invalid_argument(ERR_NOT_AUTHORIZED));
    assert!(value > 0, Errors::invalid_argument(ERR_CONFIG_PARAM_INVALID));
    let config = get_config<TokenT>();
    config.min_action_delay = value;
    Config::set_with_capability<DaoConfig<TokenT>>(cap, config);
}
Specification
aborts_if value == 0;
include CheckModifyConfigWithCap<TokenT>;

Module Specification

pragma verify;
pragma aborts_if_is_strict;