568 lines
17 KiB
Rust
568 lines
17 KiB
Rust
#![cfg(feature = "test-bpf")]
|
|
|
|
mod program_test;
|
|
|
|
use solana_program_test::tokio;
|
|
|
|
use program_test::*;
|
|
use spl_governance::{error::GovernanceError, instruction::Vote, state::enums::ProposalState};
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Act
|
|
let vote_record_cookie = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
let vote_record_account = governance_test
|
|
.get_vote_record_account(&vote_record_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(vote_record_cookie.account, vote_record_account);
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(
|
|
token_owner_record_cookie
|
|
.account
|
|
.governing_token_deposit_amount,
|
|
proposal_account.yes_votes_count
|
|
);
|
|
|
|
assert_eq!(proposal_account.state, ProposalState::Succeeded);
|
|
assert_eq!(proposal_account.voting_completed_at, Some(1));
|
|
|
|
let token_owner_record = governance_test
|
|
.get_token_owner_record_account(&token_owner_record_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(1, token_owner_record.unrelinquished_votes_count);
|
|
assert_eq!(1, token_owner_record.total_votes_count);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_invalid_governance_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let mut proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Setup Governance for a different account
|
|
let governed_account_cookie2 = governance_test.with_governed_account().await;
|
|
|
|
let account_governance_cookie2 = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie2)
|
|
.await
|
|
.unwrap();
|
|
|
|
proposal_cookie.account.governance = account_governance_cookie2.address;
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
assert_eq!(err, GovernanceError::InvalidGovernanceForProposal.into());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_invalid_mint_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let mut proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Try to use Council Mint with Community Proposal
|
|
proposal_cookie.account.governing_token_mint = realm_cookie.account.council_mint.unwrap();
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
assert_eq!(err, GovernanceError::InvalidGoverningMintForProposal.into());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_invalid_token_owner_record_mint_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let mut token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Try to use token_owner_record for Council Mint with Community Proposal
|
|
let token_owner_record_cookie2 = governance_test
|
|
.with_council_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
token_owner_record_cookie.address = token_owner_record_cookie2.address;
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
err,
|
|
GovernanceError::InvalidGoverningMintForTokenOwnerRecord.into()
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_invalid_token_owner_record_from_different_realm_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let mut token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Try to use token_owner_record from another Realm for the same mint
|
|
let realm_cookie2 = governance_test.with_realm_using_mints(&realm_cookie).await;
|
|
|
|
let token_owner_record_cookie2 = governance_test
|
|
.with_community_token_deposit(&realm_cookie2)
|
|
.await;
|
|
|
|
token_owner_record_cookie.address = token_owner_record_cookie2.address;
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
assert_eq!(err, GovernanceError::InvalidRealmForTokenOwnerRecord.into());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_governance_authority_must_sign_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let mut token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Try to use a different owner to sign
|
|
let token_owner_record_cookie2 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
token_owner_record_cookie.token_owner = token_owner_record_cookie2.token_owner;
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
err,
|
|
GovernanceError::GoverningTokenOwnerOrDelegateMustSign.into()
|
|
);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_vote_tipped_to_succeeded() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie1 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let token_owner_record_cookie2 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let token_owner_record_cookie3 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
governance_test
|
|
.mint_community_tokens(&realm_cookie, 20)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie1, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie1, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Voting, proposal_account.state);
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie2, Vote::No)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Voting, proposal_account.state);
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie3, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Succeeded, proposal_account.state);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_vote_tipped_to_defeated() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// 100 votes
|
|
let token_owner_record_cookie1 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
// 100 votes
|
|
let token_owner_record_cookie2 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
// 100 votes
|
|
let token_owner_record_cookie3 = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
// Total 320 votes
|
|
governance_test
|
|
.mint_community_tokens(&realm_cookie, 20)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie1, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie1, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Voting, proposal_account.state);
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie2, Vote::No)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Voting, proposal_account.state);
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie3, Vote::No)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Defeated, proposal_account.state);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_threshold_below_50_and_vote_not_tipped() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut governance_config =
|
|
governance_test.get_default_governance_config(&realm_cookie, &governed_account_cookie);
|
|
|
|
governance_config.yes_vote_threshold_percentage = 40;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance_using_config(
|
|
&realm_cookie,
|
|
&governed_account_cookie,
|
|
&governance_config,
|
|
)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
// Total 210 tokens
|
|
governance_test
|
|
.mint_community_tokens(&realm_cookie, 110)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Act
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
assert_eq!(ProposalState::Voting, proposal_account.state);
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_voting_time_expired_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let proposal_account = governance_test
|
|
.get_proposal_account(&proposal_cookie.address)
|
|
.await;
|
|
|
|
let vote_expired_at_slot = account_governance_cookie.account.config.max_voting_time
|
|
+ proposal_account.voting_at.unwrap()
|
|
+ 1;
|
|
governance_test
|
|
.context
|
|
.warp_to_slot(vote_expired_at_slot)
|
|
.unwrap();
|
|
|
|
// Act
|
|
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::No)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
// Assert
|
|
|
|
assert_eq!(err, GovernanceError::ProposalVotingTimeExpired.into());
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn test_cast_vote_with_cast_twice_error() {
|
|
// Arrange
|
|
let mut governance_test = GovernanceProgramTest::start_new().await;
|
|
|
|
let realm_cookie = governance_test.with_realm().await;
|
|
let governed_account_cookie = governance_test.with_governed_account().await;
|
|
|
|
let mut account_governance_cookie = governance_test
|
|
.with_account_governance(&realm_cookie, &governed_account_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
let token_owner_record_cookie = governance_test
|
|
.with_community_token_deposit(&realm_cookie)
|
|
.await;
|
|
|
|
governance_test
|
|
.mint_community_tokens(&realm_cookie, 200)
|
|
.await;
|
|
|
|
let proposal_cookie = governance_test
|
|
.with_signed_off_proposal(&token_owner_record_cookie, &mut account_governance_cookie)
|
|
.await
|
|
.unwrap();
|
|
|
|
governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.unwrap();
|
|
|
|
governance_test.context.warp_to_slot(5).unwrap();
|
|
|
|
// Act
|
|
let err = governance_test
|
|
.with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Yes)
|
|
.await
|
|
.err()
|
|
.unwrap();
|
|
|
|
// Assert
|
|
assert_eq!(err, GovernanceError::VoteAlreadyExists.into());
|
|
}
|