#![cfg(feature = "test-sbf")] mod program_test; use { crate::program_test::args::{PluginSetupArgs, RealmSetupArgs}, program_test::*, solana_program::instruction::AccountMeta, solana_program_test::tokio, spl_governance::{ error::GovernanceError, state::{ enums::{ProposalState, VoteThreshold}, vote_record::Vote, }, }, spl_governance_test_sdk::tools::clone_keypair, }; #[tokio::test] async fn test_cast_council_veto_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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 120 governance_test.mint_council_tokens(&realm_cookie, 20).await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); let clock = governance_test.bench.get_clock().await; // Act let vote_record_cookie = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .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.veto_vote_weight ); assert_eq!(proposal_account.state, ProposalState::Vetoed); assert_eq!( proposal_account.voting_completed_at, Some(clock.unix_timestamp) ); assert_eq!(Some(120), proposal_account.max_vote_weight); assert_eq!( Some(governance_cookie.account.config.council_veto_vote_threshold), proposal_account.vote_threshold ); 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); } #[tokio::test] async fn test_cast_community_veto_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 token_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra community tokens for total supply of 120 governance_test .mint_community_tokens(&realm_cookie, 20) .await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act let vote_record_cookie = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .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.veto_vote_weight ); assert_eq!(proposal_account.state, ProposalState::Vetoed); assert_eq!(Some(120), proposal_account.max_vote_weight); assert_eq!( Some( governance_cookie .account .config .community_veto_vote_threshold ), proposal_account.vote_threshold ); } #[tokio::test] async fn test_cast_community_veto_vote_with_community_veto_disabled_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 token_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let mut governance_config = governance_test.get_default_governance_config(); governance_config.community_veto_vote_threshold = VoteThreshold::Disabled; let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act let err = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .err() .unwrap(); // Assert assert_eq!( err, GovernanceError::GoverningTokenMintNotAllowedToVote.into() ); } #[tokio::test] async fn test_cast_veto_vote_with_invalid_voting_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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act // Try to use Council Veto on Council vote Proposal let err = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .err() .unwrap(); // Assert assert_eq!(err, GovernanceError::InvalidGoverningMintForProposal.into()); } #[tokio::test] async fn test_cast_veto_vote_with_council_veto_vote_disabled_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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let mut governance_config = governance_test.get_default_governance_config(); governance_config.council_veto_vote_threshold = VoteThreshold::Disabled; let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act let err = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .err() .unwrap(); // Assert assert_eq!( err, GovernanceError::GoverningTokenMintNotAllowedToVote.into() ); } #[tokio::test] async fn test_cast_veto_vote_without_tipping() { // 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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 201 to prevent tipping governance_test .mint_council_tokens(&realm_cookie, 101) .await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act let vote_record_cookie = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .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.veto_vote_weight ); assert_eq!(proposal_account.state, ProposalState::Voting); } #[tokio::test] async fn test_cast_multiple_veto_votes_for_partially_approved_proposal() { // 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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let token_owner_record_cookie2 = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 210 to prevent single vote // tipping governance_test.mint_council_tokens(&realm_cookie, 10).await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 200 to prevent single vote // tipping governance_test .mint_community_tokens(&realm_cookie, 100) .await; let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Partially approve Proposal governance_test .with_cast_yes_no_vote( &proposal_cookie, &proposal_owner_record_cookie, YesNoVote::Yes, ) .await .unwrap(); // Partially Veto Proposal governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie2, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(200, proposal_account.veto_vote_weight); assert_eq!(proposal_account.state, ProposalState::Vetoed); } #[tokio::test] async fn test_cast_veto_vote_with_no_council_error() { // Arrange let mut governance_test = GovernanceProgramTest::start_new().await; let mut realm_cookie = governance_test.with_realm().await; let governed_account_cookie = governance_test.with_governed_account().await; let token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let mut governance_config = governance_test.get_default_governance_config(); governance_config.council_veto_vote_threshold = VoteThreshold::Disabled; let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Remove Council let realm_setup_args = RealmSetupArgs { use_council_mint: false, ..Default::default() }; governance_test .set_realm_config(&mut realm_cookie, &realm_setup_args) .await .unwrap(); // Act let err = governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .err() .unwrap(); // Assert assert_eq!(err, GovernanceError::InvalidGoverningTokenMint.into()); } #[tokio::test] async fn test_relinquish_veto_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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 201 to prevent tipping governance_test .mint_council_tokens(&realm_cookie, 101) .await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Act governance_test .relinquish_vote(&proposal_cookie, &token_owner_record_cookie) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(0, proposal_account.veto_vote_weight); assert_eq!(proposal_account.state, ProposalState::Voting); } #[tokio::test] async fn test_relinquish_veto_vote_with_vote_record_for_different_voting_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 council_token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 210 governance_test .mint_council_tokens(&realm_cookie, 110) .await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &council_token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); governance_test .with_cast_vote( &proposal_cookie, &council_token_owner_record_cookie, Vote::Veto, ) .await .unwrap(); // Create Community TokenOwnerRecord for council_token_owner and Cast Community // vote let community_token_owner_record_cookie = governance_test .with_community_token_deposit_by_owner( &realm_cookie, 100, clone_keypair(&council_token_owner_record_cookie.token_owner), ) .await .unwrap(); // Mint extra council tokens for total supply of 250 governance_test .mint_community_tokens(&realm_cookie, 150) .await; let community_vote_record_cookie = governance_test .with_cast_yes_no_vote( &proposal_cookie, &community_token_owner_record_cookie, YesNoVote::Yes, ) .await .unwrap(); // Act let err = governance_test .relinquish_vote_using_instruction( &proposal_cookie, &council_token_owner_record_cookie, |i| { // Try to use a vote_record from community Yes vote to relinquish council Veto // vote i.accounts[4] = AccountMeta::new(community_vote_record_cookie.address, false) }, ) .await .err() .unwrap(); // Assert assert_eq!(err, GovernanceError::InvalidGoverningMintForProposal.into()); } #[tokio::test] async fn test_cast_veto_vote_with_council_only_allowed_to_veto() { // 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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Allow Council to cast only Veto votes let mut governance_config = governance_test.get_default_governance_config(); governance_config.council_vote_threshold = VoteThreshold::Disabled; let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(proposal_account.state, ProposalState::Vetoed); } #[tokio::test] async fn test_cast_yes_and_veto_votes_with_yes_as_winning_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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 210 to prevent single vote // tipping governance_test .mint_council_tokens(&realm_cookie, 110) .await; let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Partially Veto Proposal governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Act // Approve Proposal governance_test .with_cast_yes_no_vote( &proposal_cookie, &proposal_owner_record_cookie, YesNoVote::Yes, ) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(100, proposal_account.veto_vote_weight); assert_eq!(proposal_account.state, ProposalState::Succeeded); } #[tokio::test] async fn test_veto_vote_with_community_voter_weight_addin() { // Arrange let mut governance_test = GovernanceProgramTest::start_with_voter_weight_addin().await; let governed_account_cookie = governance_test.with_governed_account().await; let realm_cookie = governance_test .with_realm_using_addins(PluginSetupArgs::COMMUNITY_VOTER_WEIGHT) .await; let mut token_owner_record_cookie = governance_test .with_community_token_owner_record(&realm_cookie) .await; governance_test .with_voter_weight_addin_record(&mut token_owner_record_cookie) .await .unwrap(); let mut governance_cookie = governance_test .with_governance( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, ) .await .unwrap(); // Create Proposal for Council vote let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(proposal_account.state, ProposalState::Vetoed); } #[tokio::test] async fn test_veto_vote_with_community_max_voter_weight_addin() { // Arrange let mut governance_test = GovernanceProgramTest::start_with_max_voter_weight_addin().await; let governed_account_cookie = governance_test.with_governed_account().await; let realm_cookie = governance_test .with_realm_using_addins(PluginSetupArgs::COMMUNITY_MAX_VOTER_WEIGHT) .await; // TokenOwnerRecord with voting power of 100 let mut token_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); // Bump MaxVoterWeight to 200 governance_test .with_max_voter_weight_addin_record(&mut token_owner_record_cookie) .await .unwrap(); let mut governance_config = governance_test.get_default_governance_config(); governance_config.community_veto_vote_threshold = VoteThreshold::YesVotePercentage(50); // 50% Veto let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); // Create Proposal for Council vote let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(proposal_account.state, ProposalState::Vetoed); } #[tokio::test] async fn test_veto_vote_with_community_max_voter_weight_addin_and_veto_not_tipped() { // Arrange let mut governance_test = GovernanceProgramTest::start_with_max_voter_weight_addin().await; let governed_account_cookie = governance_test.with_governed_account().await; let realm_cookie = governance_test .with_realm_using_addins(PluginSetupArgs::COMMUNITY_MAX_VOTER_WEIGHT) .await; // TokenOwnerRecord with voting power of 100 let mut token_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); // Bump MaxVoterWeight to 200 governance_test .with_max_voter_weight_addin_record(&mut token_owner_record_cookie) .await .unwrap(); let mut governance_config = governance_test.get_default_governance_config(); governance_config.community_veto_vote_threshold = VoteThreshold::YesVotePercentage(51); // 51% Veto let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); // Create Proposal for Council vote let proposal_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(proposal_account.state, ProposalState::Voting); } #[tokio::test] async fn test_cast_council_veto_vote_within_cool_off_time() { // 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 token_owner_record_cookie = governance_test .with_council_token_deposit(&realm_cookie) .await .unwrap(); // Mint extra council tokens for total supply of 120 governance_test.mint_council_tokens(&realm_cookie, 20).await; // Set none default voting cool off time let mut governance_config = governance_test.get_default_governance_config(); governance_config.voting_cool_off_time = 50; let mut governance_cookie = governance_test .with_governance_using_config( &realm_cookie, &governed_account_cookie, &token_owner_record_cookie, &governance_config, ) .await .unwrap(); let proposal_owner_record_cookie = governance_test .with_community_token_deposit(&realm_cookie) .await .unwrap(); let proposal_cookie = governance_test .with_signed_off_proposal(&proposal_owner_record_cookie, &mut governance_cookie) .await .unwrap(); // Advance timestamp into voting_cool_off_time let clock = governance_test.bench.get_clock().await; governance_test .advance_clock_past_timestamp( clock.unix_timestamp + governance_cookie.account.config.voting_base_time as i64, ) .await; // Act governance_test .with_cast_vote(&proposal_cookie, &token_owner_record_cookie, Vote::Veto) .await .unwrap(); // Assert let proposal_account = governance_test .get_proposal_account(&proposal_cookie.address) .await; assert_eq!(proposal_account.state, ProposalState::Vetoed); }