From d8f3c22a28a500621ffe7be4acdb839e71a07641 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Fri, 13 Dec 2024 20:00:29 +0900 Subject: [PATCH 01/56] Dao contract --- pop-api/examples/dao/Cargo.toml | 34 +++++ pop-api/examples/dao/src/lib.rs | 229 ++++++++++++++++++++++++++++++++ 2 files changed, 263 insertions(+) create mode 100644 pop-api/examples/dao/Cargo.toml create mode 100644 pop-api/examples/dao/src/lib.rs diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml new file mode 100644 index 000000000..a46f14cd9 --- /dev/null +++ b/pop-api/examples/dao/Cargo.toml @@ -0,0 +1,34 @@ +[package] +name = "dao" +version = "0.1.0" +edition = "2021" + +[dependencies] +ink = { version = "=5.0.0", default-features = false, features = ["ink-debug"] } +pop-api = { path = "../../../pop-api", default-features = false, features = [ + "fungibles", +] } +scale = { package = "parity-scale-codec", version = "3", default-features = false, features = ["derive"] } +scale-info = { version = "2.3", default-features = false, features = ["derive"], optional = true } + +[dev-dependencies] +drink = { package = "pop-drink", git = "https://github.com/r0gue-io/pop-drink" } +env_logger = { version = "0.11.3" } +serde_json = "1.0.114" + +# TODO: due to compilation issues caused by `sp-runtime`, `frame-support-procedural` and `staging-xcm` this dependency +# (with specific version) has to be added. Will be tackled by #348, please ignore for now. +frame-support-procedural = { version = "=30.0.1", default-features = false } +sp-runtime = { version = "=38.0.0", default-features = false } +staging-xcm = { version = "=14.1.0", default-features = false } + +[features] +default = ["std"] +e2e-tests = [] +ink-as-dependency = [] +std = [ + "ink/std", + "pop-api/std", + "scale-info/std", + "scale/std", +] diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs new file mode 100644 index 000000000..636695c96 --- /dev/null +++ b/pop-api/examples/dao/src/lib.rs @@ -0,0 +1,229 @@ +#[ink::contract] +mod dao { + use ink::{ + prelude::{string::String, vec::Vec}, + storage::Mapping, + }; + use pop_api::{ + primitives::TokenId, + v0::fungibles::{ + self as api, + events::{Approval, Created, Transfer}, + Psp22Error, + }, + }; + + /// Structure of the proposal used by the Dao governance sysytem + #[derive(scale::Decode, scale::Encode)] + #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] + pub struct Proposal { + // Description of the proposal + description: String, + + // Beginnning of the voting period for this proposal + vote_start: BlockNumber, + + // End of the voting period for this proposal + vote_end: BlockNumber, + + // Balance representing the total votes for this proposal + yes_votes: Balance, + + // Balance representing the total votes against this proposal + no_votes: Balance, + + // Flag that indicates if the proposal was executed + executed: bool, + + // AccountId of the recipient of the proposal + beneficiary: AccountId, + + // Amount of tokens to be awarded to the beneficiary + amount: Balance, + } + + #[derive(scale::Decode, scale::Encode)] + #[cfg_attr(feature = "std", derive(scale_info::TypeInfo, ink::storage::traits::StorageLayout))] + pub struct Member { + voting_power: Balance, + last_vote: BlockNumber, + } + + /// Structure of a DAO (Decentralized Autonomous Organization) + /// that uses Psp22 to manage the Dao treasury and funds projects + /// selected by the members through governance + #[ink(storage)] + pub struct Dao { + // Funding proposals + proposals: Vec, + + // Mapping of AccountId to Member structs, representing DAO membership. + members: Mapping, + + // Mapping tracking the last time each account voted. + last_votes: Mapping, + + // Duration of the voting period + voting_period: BlockNumber, + + // Identifier of the Psp22 token associated with this DAO + token_id: TokenId, + } + + impl Dao { + #[ink(constructor, payable)] + pub fn new( + token_id: TokenId, + voting_period: BlockNumber, + min_balance: Balance, + ) -> Result { + let instance = Self { + proposals: Vec::new(), + members: Mapping::default(), + last_votes: Mapping::default(), + voting_period, + token_id: token_id.clone(), + }; + let contract_id = instance.env().account_id(); + api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; + instance.env(); + + Ok(instance) + } + + #[ink(message)] + pub fn create_proposal( + &mut self, + beneficiary: AccountId, + amount: Balance, + description: String, + ) -> Result<(), Error> { + let _caller = self.env().caller(); + let current_block = self.env().block_number(); + let proposal = Proposal { + description, + vote_start: current_block, + vote_end: current_block + self.voting_period, + yes_votes: 0, + no_votes: 0, + executed: false, + beneficiary, + amount, + }; + + self.proposals.push(proposal); + Ok(()) + } + + #[ink(message)] + pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { + let caller = self.env().caller(); + let current_block = self.env().block_number(); + + let proposal = + self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; + + if current_block < proposal.vote_start || current_block > proposal.vote_end { + return Err(Error::VotingPeriodEnded); + } + + let member = self.members.get(caller).ok_or(Error::NotAMember)?; + + if member.last_vote >= proposal.vote_start { + return Err(Error::AlreadyVoted); + } + + if approve { + proposal.yes_votes += member.voting_power; + } else { + proposal.no_votes += member.voting_power; + } + + self.members.insert( + caller, + &Member { voting_power: member.voting_power, last_vote: current_block }, + ); + + Ok(()) + } + + #[ink(message)] + pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { + let vote_end = self + .proposals + .get(proposal_id as usize) + .ok_or(Error::ProposalNotFound)? + .vote_end; + + // Check the voting period + if self.env().block_number() <= vote_end { + return Err(Error::VotingPeriodNotEnded); + } + + // If we've passed the checks, now we can mutably borrow the proposal + let proposal = + self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; + + if proposal.executed { + return Err(Error::ProposalAlreadyExecuted); + } + + if proposal.yes_votes > proposal.no_votes { + // ToDo: Check that there is enough funds in the treasury + // Execute the proposal + api::transfer(self.token_id, proposal.beneficiary, proposal.amount) + .map_err(Psp22Error::from)?; + + proposal.executed = true; + Ok(()) + } else { + Err(Error::ProposalRejected) + } + } + + #[ink(message)] + pub fn join(&mut self, amount: Balance) -> Result<(), Error> { + let caller = self.env().caller(); + api::transfer_from(self.token_id, caller, self.env().account_id(), amount) + .map_err(Psp22Error::from)?; + + let member = + self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); + + self.members.insert( + caller, + &Member { voting_power: member.voting_power + amount, last_vote: member.last_vote }, + ); + + Ok(()) + } + } + + #[derive(Debug, PartialEq, Eq, scale::Encode, scale::Decode)] + #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] + pub enum Error { + ProposalNotFound, + VotingPeriodEnded, + NotAMember, + AlreadyVoted, + VotingPeriodNotEnded, + ProposalAlreadyExecuted, + ProposalRejected, + Psp22(Psp22Error), + } + + impl From for Error { + fn from(error: Psp22Error) -> Self { + Error::Psp22(error) + } + } + + impl From for Psp22Error { + fn from(error: Error) -> Self { + match error { + Error::Psp22(psp22_error) => psp22_error, + _ => Psp22Error::Custom(String::from("Unknown error")), + } + } + } +} From 5cbb44a19173caea09d29d709962cb4a05fe2d35 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 09:40:41 +0900 Subject: [PATCH 02/56] Added some events --- pop-api/examples/dao/src/lib.rs | 38 ++++++++++++++++++++++++++------- 1 file changed, 30 insertions(+), 8 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 636695c96..8ce95e90d 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,3 +1,6 @@ +#[cfg(test)] +mod tests; + #[ink::contract] mod dao { use ink::{ @@ -8,13 +11,14 @@ mod dao { primitives::TokenId, v0::fungibles::{ self as api, - events::{Approval, Created, Transfer}, + events::{Created, Transfer}, Psp22Error, }, }; + /// Structure of the proposal used by the Dao governance sysytem - #[derive(scale::Decode, scale::Encode)] + #[derive(scale::Decode, scale::Encode, Debug)] #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] pub struct Proposal { // Description of the proposal @@ -86,7 +90,11 @@ mod dao { }; let contract_id = instance.env().account_id(); api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; - instance.env(); + instance.env().emit_event(Created { + id: token_id, + creator: contract_id, + admin: contract_id, + }); Ok(instance) } @@ -161,20 +169,28 @@ mod dao { } // If we've passed the checks, now we can mutably borrow the proposal - let proposal = - self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; + let proposal_id_usize = proposal_id as usize; + let proposal = self.proposals.get(proposal_id_usize).ok_or(Error::ProposalNotFound)?; if proposal.executed { return Err(Error::ProposalAlreadyExecuted); } if proposal.yes_votes > proposal.no_votes { + let contract = self.env().account_id(); // ToDo: Check that there is enough funds in the treasury // Execute the proposal api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; - - proposal.executed = true; + self.env().emit_event(Transfer { + from: Some(contract), + to: Some(proposal.beneficiary), + value: proposal.amount, + }); + + if let Some(proposal) = self.proposals.get_mut(proposal_id_usize) { + proposal.executed = true; + } Ok(()) } else { Err(Error::ProposalRejected) @@ -184,8 +200,14 @@ mod dao { #[ink(message)] pub fn join(&mut self, amount: Balance) -> Result<(), Error> { let caller = self.env().caller(); - api::transfer_from(self.token_id, caller, self.env().account_id(), amount) + let contract = self.env().account_id(); + api::transfer_from(self.token_id, caller.clone(), contract.clone(), amount) .map_err(Psp22Error::from)?; + self.env().emit_event(Transfer { + from: Some(caller), + to: Some(contract), + value: amount, + }); let member = self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); From bb060f1144fecd44cc9164b82b44e92deb91d2c6 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 09:41:55 +0900 Subject: [PATCH 03/56] removed absent module --- pop-api/examples/dao/src/lib.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 8ce95e90d..c0679c6db 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,5 +1,4 @@ #[cfg(test)] -mod tests; #[ink::contract] mod dao { From 30637ae5bcddfbb85f6729658083df48684bd9b3 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 10:21:23 +0900 Subject: [PATCH 04/56] Added some in-code documentation --- pop-api/examples/dao/src/lib.rs | 42 +++++++++++++++++++++++++++++++++ 1 file changed, 42 insertions(+) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c0679c6db..93f4e1109 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -43,12 +43,19 @@ mod dao { // Amount of tokens to be awarded to the beneficiary amount: Balance, + + // Identifier of the proposal + proposal_id: u32, } + /// Representation of a member in the voting system #[derive(scale::Decode, scale::Encode)] #[cfg_attr(feature = "std", derive(scale_info::TypeInfo, ink::storage::traits::StorageLayout))] pub struct Member { + // Stores the member's voting influence by using his balance voting_power: Balance, + + // Keeps track of the last vote casted by the member last_vote: BlockNumber, } @@ -74,6 +81,14 @@ mod dao { } impl Dao { + /// Instantiate a new Dao contract and create the associated token + /// + /// # Parameters: + /// - `token_id` - The identifier of the token to be created + /// - `voting_period` - Amount of blocks during which members can cast their votes + /// - `min_balance` - The minimum balance required for accounts holding this token. + // The `min_balance` ensures accounts hold a minimum amount of tokens, preventing tiny, + // inactive balances from bloating the blockchain state and slowing down the network. #[ink(constructor, payable)] pub fn new( token_id: TokenId, @@ -98,6 +113,13 @@ mod dao { Ok(instance) } + /// Allows members to create new spending proposals + /// + /// # Parameters + /// - `beneficiary` - The account that will receive the payment + /// if the proposal is accepted. + /// - `amount` - Amount requested for this proposal + /// - `description` - Description of the proposal #[ink(message)] pub fn create_proposal( &mut self, @@ -107,6 +129,7 @@ mod dao { ) -> Result<(), Error> { let _caller = self.env().caller(); let current_block = self.env().block_number(); + let proposal_id = self.proposals.len() - 1; let proposal = Proposal { description, vote_start: current_block, @@ -116,12 +139,18 @@ mod dao { executed: false, beneficiary, amount, + proposal_id, }; self.proposals.push(proposal); Ok(()) } + /// Allows Dao's members to vote for a proposal + /// + /// # Parameters + /// - `proposal_id` - Identifier of the proposal + /// - `approve` - Indicates whether the vote is in favor (true) or against (false) the proposal. #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); @@ -154,6 +183,10 @@ mod dao { Ok(()) } + /// Enact a proposal approved by the Dao members + /// + /// # Parameters + /// - `proposal_id` - Identifier of the proposal #[ink(message)] pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { let vote_end = self @@ -196,6 +229,15 @@ mod dao { } } + /// Allows a user to become a member of the Dao + /// by transferring some tokens to the DAO's treasury. + /// The amount of tokens transferred will be stored as the + /// voting power of this member. + /// + /// # Parameters + /// - `amount` - Balance transferred to the Dao and representing + /// the voting power of the member. + #[ink(message)] pub fn join(&mut self, amount: Balance) -> Result<(), Error> { let caller = self.env().caller(); From e6c9edd638983026148fa445fe905d5cff036110 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 10:36:37 +0900 Subject: [PATCH 05/56] Corrected index error --- pop-api/examples/dao/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 93f4e1109..2e35ceab2 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -129,7 +129,7 @@ mod dao { ) -> Result<(), Error> { let _caller = self.env().caller(); let current_block = self.env().block_number(); - let proposal_id = self.proposals.len() - 1; + let proposal_id = self.proposals.len(); let proposal = Proposal { description, vote_start: current_block, From 058e53b1e036ec9e1239cf2cf19bff51c554303c Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 12:09:44 +0900 Subject: [PATCH 06/56] Problem with first test --- pop-api/examples/dao/src/lib.rs | 61 ++++++++++++++++++++++++++------- 1 file changed, 48 insertions(+), 13 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 2e35ceab2..fdea23b70 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,5 +1,4 @@ #[cfg(test)] - #[ink::contract] mod dao { use ink::{ @@ -10,12 +9,11 @@ mod dao { primitives::TokenId, v0::fungibles::{ self as api, - events::{Created, Transfer}, + events::{Approval, Created, Transfer}, Psp22Error, }, }; - /// Structure of the proposal used by the Dao governance sysytem #[derive(scale::Decode, scale::Encode, Debug)] #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] @@ -81,6 +79,7 @@ mod dao { } impl Dao { + /// Instantiate a new Dao contract and create the associated token /// /// # Parameters: @@ -114,12 +113,12 @@ mod dao { } /// Allows members to create new spending proposals - /// + /// /// # Parameters - /// - `beneficiary` - The account that will receive the payment + /// - `beneficiary` - The account that will receive the payment /// if the proposal is accepted. /// - `amount` - Amount requested for this proposal - /// - `description` - Description of the proposal + /// - `description` - Description of the proposal #[ink(message)] pub fn create_proposal( &mut self, @@ -129,7 +128,7 @@ mod dao { ) -> Result<(), Error> { let _caller = self.env().caller(); let current_block = self.env().block_number(); - let proposal_id = self.proposals.len(); + let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let proposal = Proposal { description, vote_start: current_block, @@ -147,10 +146,11 @@ mod dao { } /// Allows Dao's members to vote for a proposal - /// + /// /// # Parameters /// - `proposal_id` - Identifier of the proposal - /// - `approve` - Indicates whether the vote is in favor (true) or against (false) the proposal. + /// - `approve` - Indicates whether the vote is in favor (true) or against (false) the + /// proposal. #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); @@ -184,7 +184,7 @@ mod dao { } /// Enact a proposal approved by the Dao members - /// + /// /// # Parameters /// - `proposal_id` - Identifier of the proposal #[ink(message)] @@ -219,6 +219,11 @@ mod dao { to: Some(proposal.beneficiary), value: proposal.amount, }); + self.env().emit_event(Approval { + owner: contract, + spender: contract, + value: proposal.amount, + }); if let Some(proposal) = self.proposals.get_mut(proposal_id_usize) { proposal.executed = true; @@ -231,13 +236,13 @@ mod dao { /// Allows a user to become a member of the Dao /// by transferring some tokens to the DAO's treasury. - /// The amount of tokens transferred will be stored as the + /// The amount of tokens transferred will be stored as the /// voting power of this member. /// /// # Parameters - /// - `amount` - Balance transferred to the Dao and representing + /// - `amount` - Balance transferred to the Dao and representing /// the voting power of the member. - + #[ink(message)] pub fn join(&mut self, amount: Balance) -> Result<(), Error> { let caller = self.env().caller(); @@ -289,4 +294,34 @@ mod dao { } } } + + #[cfg(test)] + mod tests { + use super::*; + const UNIT: Balance = 10_000_000_000; + const INIT_AMOUNT: Balance = 100_000_000 * UNIT; + const INIT_VALUE: Balance = 100 * UNIT; + const AMOUNT: Balance = MIN_BALANCE * 4; + const MIN_BALANCE: Balance = 10_000; + const TOKEN: TokenId = 1; + + #[ink::test] + fn test_join() { + // Setup + let accounts = ink::env::test::default_accounts::(); + // Test joining the DAO + ink::env::test::set_caller::(accounts.alice); + + if let Ok(mut dao) = Dao::new(TOKEN, 10, MIN_BALANCE) { + // Give some tokens to Alice + let _ = api::transfer(TOKEN, accounts.alice, INIT_AMOUNT); + + assert_eq!(dao.join(AMOUNT), Ok(())); + // Verify member was added correctly + let member = dao.members.get(accounts.alice).unwrap(); + assert_eq!(member.voting_power, AMOUNT); + assert_eq!(member.last_vote, 0); + } + } + } } From ff2b300b553608225701c09d9cf21e782a3fdc01 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 15 Dec 2024 21:08:16 +0900 Subject: [PATCH 07/56] Making sense of pop-drink for testing --- pop-api/examples/dao/src/lib.rs | 72 ++++++++++------------------- pop-api/examples/dao/src/tests.rs | 77 +++++++++++++++++++++++++++++++ 2 files changed, 101 insertions(+), 48 deletions(-) create mode 100644 pop-api/examples/dao/src/tests.rs diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index fdea23b70..44c8c893f 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,18 +1,22 @@ +use ink::{ + prelude::{string::String, vec::Vec}, + storage::Mapping, +}; +use pop_api::{ + primitives::TokenId, + v0::fungibles::{ + self as api, + events::{Approval, Created, Transfer}, + Psp22Error, + }, +}; + #[cfg(test)] +mod tests; + #[ink::contract] mod dao { - use ink::{ - prelude::{string::String, vec::Vec}, - storage::Mapping, - }; - use pop_api::{ - primitives::TokenId, - v0::fungibles::{ - self as api, - events::{Approval, Created, Transfer}, - Psp22Error, - }, - }; + use super::*; /// Structure of the proposal used by the Dao governance sysytem #[derive(scale::Decode, scale::Encode, Debug)] @@ -79,7 +83,6 @@ mod dao { } impl Dao { - /// Instantiate a new Dao contract and create the associated token /// /// # Parameters: @@ -99,7 +102,7 @@ mod dao { members: Mapping::default(), last_votes: Mapping::default(), voting_period, - token_id: token_id.clone(), + token_id: token_id, }; let contract_id = instance.env().account_id(); api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; @@ -129,10 +132,11 @@ mod dao { let _caller = self.env().caller(); let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); + let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; let proposal = Proposal { description, vote_start: current_block, - vote_end: current_block + self.voting_period, + vote_end, yes_votes: 0, no_votes: 0, executed: false, @@ -170,9 +174,9 @@ mod dao { } if approve { - proposal.yes_votes += member.voting_power; + proposal.yes_votes.checked_add(member.voting_power).ok_or(Error::ArithmeticOverflow)?; } else { - proposal.no_votes += member.voting_power; + proposal.no_votes.checked_add(member.voting_power).ok_or(Error::ArithmeticOverflow)?; } self.members.insert( @@ -258,9 +262,10 @@ mod dao { let member = self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); + let voting_power = member.voting_power.checked_add(amount).ok_or(Error::ArithmeticOverflow)?; self.members.insert( caller, - &Member { voting_power: member.voting_power + amount, last_vote: member.last_vote }, + &Member { voting_power, last_vote: member.last_vote }, ); Ok(()) @@ -270,6 +275,7 @@ mod dao { #[derive(Debug, PartialEq, Eq, scale::Encode, scale::Decode)] #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] pub enum Error { + ArithmeticOverflow, ProposalNotFound, VotingPeriodEnded, NotAMember, @@ -294,34 +300,4 @@ mod dao { } } } - - #[cfg(test)] - mod tests { - use super::*; - const UNIT: Balance = 10_000_000_000; - const INIT_AMOUNT: Balance = 100_000_000 * UNIT; - const INIT_VALUE: Balance = 100 * UNIT; - const AMOUNT: Balance = MIN_BALANCE * 4; - const MIN_BALANCE: Balance = 10_000; - const TOKEN: TokenId = 1; - - #[ink::test] - fn test_join() { - // Setup - let accounts = ink::env::test::default_accounts::(); - // Test joining the DAO - ink::env::test::set_caller::(accounts.alice); - - if let Ok(mut dao) = Dao::new(TOKEN, 10, MIN_BALANCE) { - // Give some tokens to Alice - let _ = api::transfer(TOKEN, accounts.alice, INIT_AMOUNT); - - assert_eq!(dao.join(AMOUNT), Ok(())); - // Verify member was added correctly - let member = dao.members.get(accounts.alice).unwrap(); - assert_eq!(member.voting_power, AMOUNT); - assert_eq!(member.last_vote, 0); - } - } - } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs new file mode 100644 index 000000000..9c8d46b83 --- /dev/null +++ b/pop-api/examples/dao/src/tests.rs @@ -0,0 +1,77 @@ +use drink::{ + assert_err, assert_last_contract_event, assert_ok, call, deploy, + devnet::{ + account_id_from_slice, + error::{ + v0::{ApiError::*, ArithmeticError::*, Error}, + Assets, + AssetsError::*, + }, + AccountId, Balance, Runtime, + }, + last_contract_event, + session::Session, + AssetsAPI, TestExternalities, NO_SALT, +}; +use ink::scale::Encode; +use pop_api::{ + primitives::TokenId, + v0::fungibles::events::{Approval, Created, Transfer}, +}; + +use super::*; +const UNIT: Balance = 10_000_000_000; +const INIT_AMOUNT: Balance = 100_000_000 * UNIT; +const INIT_VALUE: Balance = 100 * UNIT; +const ALICE: AccountId = AccountId::new([1u8; 32]); +const BOB: AccountId = AccountId::new([2_u8; 32]); +const CHARLIE: AccountId = AccountId::new([3_u8; 32]); +const AMOUNT: Balance = MIN_BALANCE * 4; +const MIN_BALANCE: Balance = 10_000; +const TOKEN: TokenId = 1; + +#[drink::contract_bundle_provider] +enum BundleProvider {} + +/// Sandbox environment for Pop Devnet Runtime. +pub struct Pop { + ext: TestExternalities, +} + +impl Default for Pop { + fn default() -> Self { + // Initialising genesis state, providing accounts with an initial balance. + let balances: Vec<(AccountId, u128)> = + vec![(ALICE, INIT_AMOUNT), (BOB, INIT_AMOUNT), (CHARLIE, INIT_AMOUNT)]; + let ext = BlockBuilder::::new_ext(balances); + Self { ext } + } +} + +// Implement core functionalities for the `Pop` sandbox. +drink::impl_sandbox!(Pop, Runtime, ALICE); + +// Deployment and constructor method tests. + +fn deploy_with_default(session: &mut Session) -> Result { + deploy(session, "new", vec![TOKEN.to_string(), 10.to_string(), MIN_BALANCE.to_string()]) +} + +#[drink::test(sandbox = Pop)] +fn new_constructor_works(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + println!("{:?}", contract); + // Token exists after the deployment. + assert!(session.sandbox().asset_exists(&TOKEN)); + // Successfully emit event. + assert_last_contract_event!( + &session, + Created { + id: TOKEN, + creator: account_id_from_slice(&contract), + admin: account_id_from_slice(&contract), + } + ); +} From 2505f899da01b4211749b1b4bda22bb8ccf0ae2e Mon Sep 17 00:00:00 2001 From: ndkazu Date: Mon, 16 Dec 2024 20:26:03 +0900 Subject: [PATCH 08/56] Added lib.rs to Cargo, but still... --- pop-api/examples/dao/Cargo.toml | 5 +++-- pop-api/examples/dao/src/lib.rs | 18 ++++++++++-------- 2 files changed, 13 insertions(+), 10 deletions(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index a46f14cd9..95fc4219b 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -22,6 +22,9 @@ frame-support-procedural = { version = "=30.0.1", default-features = false } sp-runtime = { version = "=38.0.0", default-features = false } staging-xcm = { version = "=14.1.0", default-features = false } +[lib] +path = "src/lib.rs" + [features] default = ["std"] e2e-tests = [] @@ -29,6 +32,4 @@ ink-as-dependency = [] std = [ "ink/std", "pop-api/std", - "scale-info/std", - "scale/std", ] diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 44c8c893f..aafed92fa 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,3 +1,4 @@ +#![cfg_attr(not(feature = "std"), no_std, no_main)] use ink::{ prelude::{string::String, vec::Vec}, storage::Mapping, @@ -19,8 +20,9 @@ mod dao { use super::*; /// Structure of the proposal used by the Dao governance sysytem - #[derive(scale::Decode, scale::Encode, Debug)] - #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] + #[derive(Debug)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Proposal { // Description of the proposal description: String, @@ -51,8 +53,8 @@ mod dao { } /// Representation of a member in the voting system - #[derive(scale::Decode, scale::Encode)] - #[cfg_attr(feature = "std", derive(scale_info::TypeInfo, ink::storage::traits::StorageLayout))] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Member { // Stores the member's voting influence by using his balance voting_power: Balance, @@ -102,7 +104,7 @@ mod dao { members: Mapping::default(), last_votes: Mapping::default(), voting_period, - token_id: token_id, + token_id, }; let contract_id = instance.env().account_id(); api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; @@ -251,7 +253,7 @@ mod dao { pub fn join(&mut self, amount: Balance) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - api::transfer_from(self.token_id, caller.clone(), contract.clone(), amount) + api::transfer_from(self.token_id, caller, contract, amount) .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { from: Some(caller), @@ -272,8 +274,8 @@ mod dao { } } - #[derive(Debug, PartialEq, Eq, scale::Encode, scale::Decode)] - #[cfg_attr(feature = "std", derive(scale_info::TypeInfo))] + #[derive(Debug, PartialEq, Eq)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] pub enum Error { ArithmeticOverflow, ProposalNotFound, From 6f653a9fae5cbd1295fbd022949cc8de33993b0c Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 17 Dec 2024 11:01:25 +0900 Subject: [PATCH 09/56] Added the correct deploy() function in tests --- pop-api/examples/dao/src/tests.rs | 22 +++++++++++++++++++++- pop-api/examples/dao/src/tests0.rs | 0 2 files changed, 21 insertions(+), 1 deletion(-) create mode 100644 pop-api/examples/dao/src/tests0.rs diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 9c8d46b83..6029843f3 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -1,5 +1,5 @@ use drink::{ - assert_err, assert_last_contract_event, assert_ok, call, deploy, + assert_err, assert_last_contract_event, assert_ok, call, devnet::{ account_id_from_slice, error::{ @@ -20,6 +20,7 @@ use pop_api::{ }; use super::*; + const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; const INIT_VALUE: Balance = 100 * UNIT; @@ -75,3 +76,22 @@ fn new_constructor_works(mut session: Session) { } ); } + + +// Deploy the contract with `NO_SALT and `INIT_VALUE`. +fn deploy( + session: &mut Session, + method: &str, + input: Vec, +) -> Result { + drink::deploy::( + session, + // The local contract (i.e. `fungibles`). + BundleProvider::local().unwrap(), + method, + input, + NO_SALT, + Some(INIT_VALUE), + ) +} + diff --git a/pop-api/examples/dao/src/tests0.rs b/pop-api/examples/dao/src/tests0.rs new file mode 100644 index 000000000..e69de29bb From aacbf8785838f8d0794d35e3e47d6a93ec143b2d Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 17 Dec 2024 12:11:58 +0900 Subject: [PATCH 10/56] Put a limitation of description string length --- pop-api/examples/dao/Cargo.toml | 2 -- pop-api/examples/dao/src/lib.rs | 11 +++++++++-- pop-api/examples/dao/src/tests0.rs | 0 3 files changed, 9 insertions(+), 4 deletions(-) delete mode 100644 pop-api/examples/dao/src/tests0.rs diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 95fc4219b..68bf2a1de 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -8,8 +8,6 @@ ink = { version = "=5.0.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } -scale = { package = "parity-scale-codec", version = "3", default-features = false, features = ["derive"] } -scale-info = { version = "2.3", default-features = false, features = ["derive"], optional = true } [dev-dependencies] drink = { package = "pop-drink", git = "https://github.com/r0gue-io/pop-drink" } diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index aafed92fa..659d04919 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,4 +1,5 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] + use ink::{ prelude::{string::String, vec::Vec}, storage::Mapping, @@ -12,6 +13,7 @@ use pop_api::{ }, }; +pub const STRINGLIMIT: u8 = u8::MAX; #[cfg(test)] mod tests; @@ -25,7 +27,7 @@ mod dao { #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Proposal { // Description of the proposal - description: String, + description: Vec, // Beginnning of the voting period for this proposal vote_start: BlockNumber, @@ -135,8 +137,12 @@ mod dao { let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; + let description_vec: Vec = description.as_bytes().to_vec(); + if description_vec.len() >= STRINGLIMIT.into(){ + return Err(Error::StringLimitReached); + } let proposal = Proposal { - description, + description: description_vec, vote_start: current_block, vote_end, yes_votes: 0, @@ -285,6 +291,7 @@ mod dao { VotingPeriodNotEnded, ProposalAlreadyExecuted, ProposalRejected, + StringLimitReached, Psp22(Psp22Error), } diff --git a/pop-api/examples/dao/src/tests0.rs b/pop-api/examples/dao/src/tests0.rs deleted file mode 100644 index e69de29bb..000000000 From f746400af5c31c8a674fab588d68f2a8fdaf23d2 Mon Sep 17 00:00:00 2001 From: chungquantin <56880684+chungquantin@users.noreply.github.com> Date: Tue, 17 Dec 2024 10:38:05 +0700 Subject: [PATCH 11/56] chore: add missing authors --- pop-api/examples/dao/Cargo.toml | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 68bf2a1de..9c00efea0 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -1,10 +1,11 @@ [package] +authors = [ "R0GUE " ] +edition = "2021" name = "dao" version = "0.1.0" -edition = "2021" [dependencies] -ink = { version = "=5.0.0", default-features = false, features = ["ink-debug"] } +ink = { version = "=5.0.0", default-features = false, features = [ "ink-debug" ] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } @@ -24,9 +25,9 @@ staging-xcm = { version = "=14.1.0", default-features = false } path = "src/lib.rs" [features] -default = ["std"] -e2e-tests = [] -ink-as-dependency = [] +default = [ "std" ] +e2e-tests = [ ] +ink-as-dependency = [ ] std = [ "ink/std", "pop-api/std", From 285c1781a0624d9fe0706908b515a4223a5ca541 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 17 Dec 2024 16:07:42 +0900 Subject: [PATCH 12/56] new member test added --- pop-api/examples/dao/src/lib.rs | 69 ++++++++++++++++++------------- pop-api/examples/dao/src/tests.rs | 42 ++++++++++++++++++- 2 files changed, 81 insertions(+), 30 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 659d04919..d9ca27499 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -23,46 +23,47 @@ mod dao { /// Structure of the proposal used by the Dao governance sysytem #[derive(Debug)] - #[ink::scale_derive(Encode, Decode, TypeInfo)] - #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Proposal { // Description of the proposal - description: Vec, + pub description: Vec, // Beginnning of the voting period for this proposal - vote_start: BlockNumber, + pub vote_start: BlockNumber, // End of the voting period for this proposal - vote_end: BlockNumber, + pub vote_end: BlockNumber, // Balance representing the total votes for this proposal - yes_votes: Balance, + pub yes_votes: Balance, // Balance representing the total votes against this proposal - no_votes: Balance, + pub no_votes: Balance, // Flag that indicates if the proposal was executed - executed: bool, + pub executed: bool, // AccountId of the recipient of the proposal - beneficiary: AccountId, + pub beneficiary: AccountId, // Amount of tokens to be awarded to the beneficiary - amount: Balance, + pub amount: Balance, // Identifier of the proposal - proposal_id: u32, + pub proposal_id: u32, } /// Representation of a member in the voting system - #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[derive(Debug)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Member { // Stores the member's voting influence by using his balance - voting_power: Balance, + pub voting_power: Balance, // Keeps track of the last vote casted by the member - last_vote: BlockNumber, + pub last_vote: BlockNumber, } /// Structure of a DAO (Decentralized Autonomous Organization) @@ -136,9 +137,10 @@ mod dao { let _caller = self.env().caller(); let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); - let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; + let vote_end = + current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; let description_vec: Vec = description.as_bytes().to_vec(); - if description_vec.len() >= STRINGLIMIT.into(){ + if description_vec.len() >= STRINGLIMIT.into() { return Err(Error::StringLimitReached); } let proposal = Proposal { @@ -182,9 +184,15 @@ mod dao { } if approve { - proposal.yes_votes.checked_add(member.voting_power).ok_or(Error::ArithmeticOverflow)?; + proposal + .yes_votes + .checked_add(member.voting_power) + .ok_or(Error::ArithmeticOverflow)?; } else { - proposal.no_votes.checked_add(member.voting_power).ok_or(Error::ArithmeticOverflow)?; + proposal + .no_votes + .checked_add(member.voting_power) + .ok_or(Error::ArithmeticOverflow)?; } self.members.insert( @@ -261,27 +269,31 @@ mod dao { let contract = self.env().account_id(); api::transfer_from(self.token_id, caller, contract, amount) .map_err(Psp22Error::from)?; + let member = + self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); + + let voting_power = + member.voting_power.checked_add(amount).ok_or(Error::ArithmeticOverflow)?; + self.members + .insert(caller, &Member { voting_power, last_vote: member.last_vote }); + self.env().emit_event(Transfer { from: Some(caller), to: Some(contract), value: amount, }); - let member = - self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); - - let voting_power = member.voting_power.checked_add(amount).ok_or(Error::ArithmeticOverflow)?; - self.members.insert( - caller, - &Member { voting_power, last_vote: member.last_vote }, - ); - Ok(()) } + + #[ink(message)] + pub fn get_member(&mut self, account: AccountId) -> Member { + self.members.get(account).unwrap_or(Member { voting_power: 0, last_vote: 0 }) + } } #[derive(Debug, PartialEq, Eq)] - #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] pub enum Error { ArithmeticOverflow, ProposalNotFound, @@ -292,6 +304,7 @@ mod dao { ProposalAlreadyExecuted, ProposalRejected, StringLimitReached, + None, Psp22(Psp22Error), } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 6029843f3..4a654e57c 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -3,7 +3,7 @@ use drink::{ devnet::{ account_id_from_slice, error::{ - v0::{ApiError::*, ArithmeticError::*, Error}, + v0::{ApiError::*, ArithmeticError::*}, Assets, AssetsError::*, }, @@ -20,6 +20,7 @@ use pop_api::{ }; use super::*; +use crate::dao::{Error, Member}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -60,7 +61,7 @@ fn deploy_with_default(session: &mut Session) -> Result, value: Balance) -> Result<(), Error> { + call::(session, "join", vec![value.to_string()], None) +} + +fn members(session: &mut Session, account: AccountId) -> Result { + call::(session, "get_member", vec![account.to_string()], None) +} From 34002d00edac085317df5ada3bac69e8dc10d33a Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 17 Dec 2024 17:39:37 +0900 Subject: [PATCH 13/56] create_proposal test --- pop-api/examples/dao/src/lib.rs | 10 +++++- pop-api/examples/dao/src/tests.rs | 60 +++++++++++++++++++++++++++++-- 2 files changed, 67 insertions(+), 3 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index d9ca27499..9db7efc98 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -134,7 +134,8 @@ mod dao { amount: Balance, description: String, ) -> Result<(), Error> { - let _caller = self.env().caller(); + let caller = self.env().caller(); + let contract = self.env().account_id(); let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = @@ -156,6 +157,13 @@ mod dao { }; self.proposals.push(proposal); + + self.env().emit_event(Created { + id: proposal_id, + creator: caller, + admin: contract, + }); + Ok(()) } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 4a654e57c..2675fea15 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -104,11 +104,28 @@ fn join_dao_works(mut session: Session) { value, } ); + + // We check that Alice is a member with a voting power of 20000 if let Ok(member) = members(&mut session, ALICE) { assert_eq!(member.voting_power, 20000); } } +#[drink::test(sandbox = Pop)] +fn member_create_proposal_works(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + assert_ok!(prepare_dao(&mut session, contract)); + + // Alice create a proposal + let description: String = "Funds for creation of a Dao contract".to_string(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); +} + // Deploy the contract with `NO_SALT and `INIT_VALUE`. fn deploy( session: &mut Session, @@ -127,9 +144,48 @@ fn deploy( } fn join(session: &mut Session, value: Balance) -> Result<(), Error> { - call::(session, "join", vec![value.to_string()], None) + call::( + session, + "join", + vec![value.to_string()], + None, + ) } fn members(session: &mut Session, account: AccountId) -> Result { - call::(session, "get_member", vec![account.to_string()], None) + call::( + session, + "get_member", + vec![account.to_string()], + None, + ) +} + +fn create_proposal( + session: &mut Session, + beneficiary: AccountId, + amount: Balance, + description: String, +) -> Result<(), Error> { + call::( + session, + "create_proposal", + vec![beneficiary.to_string(), amount.to_string(), description], + None, + ) +} +fn prepare_dao(session: &mut Session, contract: AccountId) -> Result<(), Error> { + assert_ok!(session.sandbox().mint_into(&TOKEN, &ALICE, AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &ALICE, &contract.clone(), AMOUNT)); + assert_ok!(session.sandbox().mint_into(&TOKEN, &BOB, AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &BOB, &contract.clone(), AMOUNT)); + assert_ok!(session.sandbox().mint_into(&TOKEN, &CHARLIE, AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &CHARLIE, &contract.clone(), AMOUNT)); + session.set_actor(ALICE); + assert_ok!(join(session, AMOUNT / 2)); + session.set_actor(BOB); + assert_ok!(join(session, AMOUNT / 4)); + session.set_actor(CHARLIE); + assert_ok!(join(session, AMOUNT / 3)); + Ok(()) } From d17096cd96f2e66ef21d1351a9df6d637e2d19e5 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 17 Dec 2024 18:49:10 +0900 Subject: [PATCH 14/56] Calls prepared for testing --- pop-api/examples/dao/src/tests.rs | 23 +++++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 2675fea15..e9ad3efcf 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -117,13 +117,13 @@ fn member_create_proposal_works(mut session: Session) { // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); // Prepare voters accounts - assert_ok!(prepare_dao(&mut session, contract)); + let _ = prepare_dao(&mut session, contract); // Alice create a proposal let description: String = "Funds for creation of a Dao contract".to_string(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + //assert_ok!(create_proposal(&mut session, BOB, amount, description)); } // Deploy the contract with `NO_SALT and `INIT_VALUE`. @@ -174,6 +174,25 @@ fn create_proposal( None, ) } + +fn vote( session: &mut Session, proposal_id: u32, approve: bool) -> Result<(), Error> { + call:: ( + session, + "vote", + vec![proposal_id.to_string(), approve.to_string()], + None, + ) +} + +fn execute_proposal( session: &mut Session, proposal_id: u32) -> Result<(), Error> { + call:: ( + session, + "execute_proposal", + vec![proposal_id.to_string()], + None, + ) +} + fn prepare_dao(session: &mut Session, contract: AccountId) -> Result<(), Error> { assert_ok!(session.sandbox().mint_into(&TOKEN, &ALICE, AMOUNT)); assert_ok!(session.sandbox().approve(&TOKEN, &ALICE, &contract.clone(), AMOUNT)); From 3f2c114099b82d06a65cf94e46d7586e6704982f Mon Sep 17 00:00:00 2001 From: ndkazu Date: Wed, 18 Dec 2024 21:01:55 +0900 Subject: [PATCH 15/56] ReadMe --- pop-api/examples/dao/README.md | 53 ++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) create mode 100644 pop-api/examples/dao/README.md diff --git a/pop-api/examples/dao/README.md b/pop-api/examples/dao/README.md new file mode 100644 index 000000000..776cfcc73 --- /dev/null +++ b/pop-api/examples/dao/README.md @@ -0,0 +1,53 @@ +# PSP22 DAO contract example + +## Description +This contract implements a Decentralized Autonomous Organization using Psp22. +The key functionalities include: +- **Membership Management**: It maintains a registry of DAO members. +- **Proposal Lifecycle**: The contract manages the creation, voting, and execution of proposals. Each proposal includes details like description, voting period, vote tallies, execution status, beneficiary, and amount to be awarded. +- **Voting Mechanism**: It implements a voting system where members can vote with their balance on proposals. The contract tracks voting periods and maintains vote counts for each proposal. +- **Token Integration**: The DAO is associated with a specific Psp22 token_id. +- **Governance Parameters**: governance parameters such as voting periods are customizable. +- **Vote Tracking**: The contract keeps track of when members last voted. +- **Proposal Execution**: Once a proposal's voting period ends and passes, the contract handles its execution: transferring funds to the chosen beneficiary. + + +## Dao Workflow +- **Subscription**: The first step is membership: users use their funds* to join the Dao, become members, and determine their voting power, as membership gives them the right to vote on the use of the Dao Treasury. +- **Proposal**: Dao members can create spending proposals for the treasury. At the moment, the voting period for the proposal is given by the proposal creator, but this could be a Dao parameter, determined by the creator of the Dao contract. +- **Vote**: Members of the Dao can vote for or against a given proposal, through the selection of the corresponding proposal ID. The vote has to be cast within the voting period of the selected proposal. +- **Proposal enactment**: After the end of the voting period, If the proposal has been accepted by the Dao Members, the proposal can be enacted, i.e. funds can be claimed/transferred to the account specified as the beneficiary in the proposal. Any member can claim the reward for the winning proposal. + +*It is assumed that after Dao & associated token creation, potential members own the Dao Token through airdrop, token purchase, etc... (These mechanisms are not part of the contract). + +### Flowchart + +```mermaid +flowchart LR + A[Subscriber A] -->|Joins DAO| M[Membership Process] + B[Subscriber B] -->|Joins DAO| M + C[Subscriber C] -->|Joins DAO| M + D[Subscriber D] -->|Joins DAO| M + M --> E[Subscriber B creates Proposal] + E --> F[Voting Process] + F -->|Votes by| G[Subscribers A, C, D] + G --> H{Proposal Accepted?} + H -- Yes --> I[Proposal Enactment] + H -- No --> J[End of Process] + I --> K[Subscriber A claims reward for B] + K --> L[Funds transferred to beneficiary] + + style A fill:#f9f,stroke:#333,stroke-width:2px + style B fill:#f9f,stroke:#333,stroke-width:2px + style C fill:#f9f,stroke:#333,stroke-width:2px + style D fill:#f9f,stroke:#333,stroke-width:2px + style M fill:#bbf,stroke:#333,stroke-width:2px + style E fill:#bbf,stroke:#333,stroke-width:2px + style F fill:#bbf,stroke:#333,stroke-width:2px + style G fill:#bbf,stroke:#333,stroke-width:2px + style H fill:#ff0,stroke:#333,stroke-width:2px + style I fill:#bfb,stroke:#333,stroke-width:2px + style J fill:#f99,stroke:#333,stroke-width:2px + style K fill:#bfb,stroke:#333,stroke-width:2px + style L fill:#bfb,stroke:#333,stroke-width:2px +``` \ No newline at end of file From 22d46e3ed37a50fb21b42ebc0263debbcd787f5f Mon Sep 17 00:00:00 2001 From: ndkazu Date: Wed, 18 Dec 2024 21:20:36 +0900 Subject: [PATCH 16/56] ReadME --- pop-api/examples/dao/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pop-api/examples/dao/README.md b/pop-api/examples/dao/README.md index 776cfcc73..609d2ca90 100644 --- a/pop-api/examples/dao/README.md +++ b/pop-api/examples/dao/README.md @@ -34,7 +34,7 @@ flowchart LR G --> H{Proposal Accepted?} H -- Yes --> I[Proposal Enactment] H -- No --> J[End of Process] - I --> K[Subscriber A claims reward for B] + I --> K[Subscriber A claims reward for proposal's beneficiary] K --> L[Funds transferred to beneficiary] style A fill:#f9f,stroke:#333,stroke-width:2px From b02f232d356efa2e6c89da947374f373db5adcac Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 12:47:32 +0900 Subject: [PATCH 17/56] create_proposal_test --- pop-api/examples/dao/src/lib.rs | 7 +++---- pop-api/examples/dao/src/tests.rs | 20 +++++++++++++++----- 2 files changed, 18 insertions(+), 9 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 9db7efc98..418396871 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -132,7 +132,7 @@ mod dao { &mut self, beneficiary: AccountId, amount: Balance, - description: String, + description: Vec, ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); @@ -140,12 +140,11 @@ mod dao { let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; - let description_vec: Vec = description.as_bytes().to_vec(); - if description_vec.len() >= STRINGLIMIT.into() { + if description.len() >= STRINGLIMIT.into() { return Err(Error::StringLimitReached); } let proposal = Proposal { - description: description_vec, + description, vote_start: current_block, vote_end, yes_votes: 0, diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index e9ad3efcf..a6cb4055b 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -117,13 +117,22 @@ fn member_create_proposal_works(mut session: Session) { // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); // Prepare voters accounts - let _ = prepare_dao(&mut session, contract); + let _ = prepare_dao(&mut session, contract.clone()); // Alice create a proposal - let description: String = "Funds for creation of a Dao contract".to_string(); + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - //assert_ok!(create_proposal(&mut session, BOB, amount, description)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + assert_last_contract_event!( + &session, + Created { + id: 0, + creator: account_id_from_slice(&ALICE), + admin: account_id_from_slice(&contract), + } + ); } // Deploy the contract with `NO_SALT and `INIT_VALUE`. @@ -165,12 +174,13 @@ fn create_proposal( session: &mut Session, beneficiary: AccountId, amount: Balance, - description: String, + description: Vec, ) -> Result<(), Error> { + let desc: &[u8] = &description; call::( session, "create_proposal", - vec![beneficiary.to_string(), amount.to_string(), description], + vec![beneficiary.to_string(), amount.to_string(), serde_json::to_string::<[u8]>(desc).unwrap()], None, ) } From c08b335ff5a1b953bc6f559d513ddd2288f783e0 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 13:14:04 +0900 Subject: [PATCH 18/56] Another test --- pop-api/examples/dao/src/lib.rs | 22 ++++++++---- pop-api/examples/dao/src/tests.rs | 60 +++++++++++++++++++------------ 2 files changed, 52 insertions(+), 30 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 418396871..dc55b6d0c 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -87,6 +87,14 @@ mod dao { token_id: TokenId, } + /// Defines an event that is emitted + /// every time a member voted. + #[ink(event)] + pub struct Voted { + who: Option, + when: Option, + } + impl Dao { /// Instantiate a new Dao contract and create the associated token /// @@ -157,11 +165,8 @@ mod dao { self.proposals.push(proposal); - self.env().emit_event(Created { - id: proposal_id, - creator: caller, - admin: contract, - }); + self.env() + .emit_event(Created { id: proposal_id, creator: caller, admin: contract }); Ok(()) } @@ -184,7 +189,7 @@ mod dao { return Err(Error::VotingPeriodEnded); } - let member = self.members.get(caller).ok_or(Error::NotAMember)?; + let member = self.members.get(caller.clone()).ok_or(Error::NotAMember)?; if member.last_vote >= proposal.vote_start { return Err(Error::AlreadyVoted); @@ -203,10 +208,13 @@ mod dao { } self.members.insert( - caller, + caller.clone(), &Member { voting_power: member.voting_power, last_vote: current_block }, ); + let now = self.env().block_number(); + self.env().emit_event(Voted { who: Some(caller), when: Some(now) }); + Ok(()) } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index a6cb4055b..4c41d9a67 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -124,7 +124,7 @@ fn member_create_proposal_works(mut session: Session) { let amount = AMOUNT * 3; session.set_actor(ALICE); assert_ok!(create_proposal(&mut session, BOB, amount, description)); - + assert_last_contract_event!( &session, Created { @@ -135,6 +135,31 @@ fn member_create_proposal_works(mut session: Session) { ); } +#[drink::test(sandbox = Pop)] +fn members_vote_system_works(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + session.set_actor(CHARLIE); + // Charlie vote + assert_ok!(vote(&mut session, 0, true)); + let now = ink::env::block_number::(); + + assert_last_contract_event!( + &session, + Voted { who: account_id_from_slice(&CHARLIE), when: now } + ); +} + // Deploy the contract with `NO_SALT and `INIT_VALUE`. fn deploy( session: &mut Session, @@ -153,21 +178,11 @@ fn deploy( } fn join(session: &mut Session, value: Balance) -> Result<(), Error> { - call::( - session, - "join", - vec![value.to_string()], - None, - ) + call::(session, "join", vec![value.to_string()], None) } fn members(session: &mut Session, account: AccountId) -> Result { - call::( - session, - "get_member", - vec![account.to_string()], - None, - ) + call::(session, "get_member", vec![account.to_string()], None) } fn create_proposal( @@ -180,13 +195,17 @@ fn create_proposal( call::( session, "create_proposal", - vec![beneficiary.to_string(), amount.to_string(), serde_json::to_string::<[u8]>(desc).unwrap()], + vec![ + beneficiary.to_string(), + amount.to_string(), + serde_json::to_string::<[u8]>(desc).unwrap(), + ], None, ) } -fn vote( session: &mut Session, proposal_id: u32, approve: bool) -> Result<(), Error> { - call:: ( +fn vote(session: &mut Session, proposal_id: u32, approve: bool) -> Result<(), Error> { + call::( session, "vote", vec![proposal_id.to_string(), approve.to_string()], @@ -194,13 +213,8 @@ fn vote( session: &mut Session, proposal_id: u32, approve: bool) -> Result< ) } -fn execute_proposal( session: &mut Session, proposal_id: u32) -> Result<(), Error> { - call:: ( - session, - "execute_proposal", - vec![proposal_id.to_string()], - None, - ) +fn execute_proposal(session: &mut Session, proposal_id: u32) -> Result<(), Error> { + call::(session, "execute_proposal", vec![proposal_id.to_string()], None) } fn prepare_dao(session: &mut Session, contract: AccountId) -> Result<(), Error> { From 7a1563b6dad8132abd5af4b4a082c73ab89d691a Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 15:05:16 +0900 Subject: [PATCH 19/56] Enactment test --- pop-api/examples/dao/src/lib.rs | 12 +++++++--- pop-api/examples/dao/src/tests.rs | 40 ++++++++++++++++++++++++++----- 2 files changed, 43 insertions(+), 9 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index dc55b6d0c..178d616fa 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -89,10 +89,11 @@ mod dao { /// Defines an event that is emitted /// every time a member voted. + #[derive(Debug)] #[ink(event)] pub struct Voted { - who: Option, - when: Option, + pub who: Option, + pub when: Option, } impl Dao { @@ -181,6 +182,7 @@ mod dao { pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); let current_block = self.env().block_number(); + let now = self.env().block_number(); let proposal = self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; @@ -212,7 +214,6 @@ mod dao { &Member { voting_power: member.voting_power, last_vote: current_block }, ); - let now = self.env().block_number(); self.env().emit_event(Voted { who: Some(caller), when: Some(now) }); Ok(()) @@ -305,6 +306,11 @@ mod dao { pub fn get_member(&mut self, account: AccountId) -> Member { self.members.get(account).unwrap_or(Member { voting_power: 0, last_vote: 0 }) } + + #[ink(message)] + pub fn get_block_number(&mut self) -> BlockNumber { + self.env().block_number() + } } #[derive(Debug, PartialEq, Eq)] diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 4c41d9a67..ab2ffd1ac 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -18,9 +18,8 @@ use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, }; - use super::*; -use crate::dao::{Error, Member}; +use crate::dao::{Error, Member, Voted}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -31,6 +30,7 @@ const CHARLIE: AccountId = AccountId::new([3_u8; 32]); const AMOUNT: Balance = MIN_BALANCE * 4; const MIN_BALANCE: Balance = 10_000; const TOKEN: TokenId = 1; +const VOTING_PERIOD: u32 = 10; #[drink::contract_bundle_provider] enum BundleProvider {} @@ -56,7 +56,7 @@ drink::impl_sandbox!(Pop, Runtime, ALICE); // Deployment and constructor method tests. fn deploy_with_default(session: &mut Session) -> Result { - deploy(session, "new", vec![TOKEN.to_string(), 10.to_string(), MIN_BALANCE.to_string()]) + deploy(session, "new", vec![TOKEN.to_string(), VOTING_PERIOD.to_string(), MIN_BALANCE.to_string()]) } #[drink::test(sandbox = Pop)] @@ -93,7 +93,6 @@ fn join_dao_works(mut session: Session) { // Alice joins the dao assert_ok!(join(&mut session, value)); - // assert_ok!(members(&mut session, ALICE)); // Successfully emit event. assert_last_contract_event!( @@ -151,15 +150,40 @@ fn members_vote_system_works(mut session: Session) { session.set_actor(CHARLIE); // Charlie vote + let now = block(&mut session); assert_ok!(vote(&mut session, 0, true)); - let now = ink::env::block_number::(); + assert_last_contract_event!( &session, - Voted { who: account_id_from_slice(&CHARLIE), when: now } + Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); } +#[drink::test(sandbox = Pop)] +fn proposal_enactment_works(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + session.set_actor(CHARLIE); + // Charlie vote + assert_ok!(vote(&mut session, 0, true)); + let next_block = block(&mut session).saturating_add(VOTING_PERIOD); + let mut now = block(&mut session); + + //ToDo: move to next block to end the voting_period + +} + // Deploy the contract with `NO_SALT and `INIT_VALUE`. fn deploy( session: &mut Session, @@ -185,6 +209,10 @@ fn members(session: &mut Session, account: AccountId) -> Result(session, "get_member", vec![account.to_string()], None) } +fn block(session: &mut Session) -> u32{ + call::(session, "get_block_number", vec![], None).unwrap() +} + fn create_proposal( session: &mut Session, beneficiary: AccountId, From 9eca1b24ea68c2f7eab318e41612911f5a9f381d Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 16:37:42 +0900 Subject: [PATCH 20/56] one more test --- pop-api/examples/dao/src/lib.rs | 9 +++++---- pop-api/examples/dao/src/tests.rs | 30 +++++++++++++++++++++++++++++- 2 files changed, 34 insertions(+), 5 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 178d616fa..c6d09ea6d 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -145,7 +145,7 @@ mod dao { ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let current_block = self.env().block_number(); + let current_block = ink::env::block_number::(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; @@ -181,8 +181,8 @@ mod dao { #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); - let current_block = self.env().block_number(); - let now = self.env().block_number(); + let current_block = ink::env::block_number::(); + let now = ink::env::block_number::(); let proposal = self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; @@ -232,7 +232,7 @@ mod dao { .vote_end; // Check the voting period - if self.env().block_number() <= vote_end { + if ink::env::block_number::() <= vote_end { return Err(Error::VotingPeriodNotEnded); } @@ -309,6 +309,7 @@ mod dao { #[ink(message)] pub fn get_block_number(&mut self) -> BlockNumber { + //ink::env::block_number::() self.env().block_number() } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index ab2ffd1ac..143b06f8e 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -27,6 +27,7 @@ const INIT_VALUE: Balance = 100 * UNIT; const ALICE: AccountId = AccountId::new([1u8; 32]); const BOB: AccountId = AccountId::new([2_u8; 32]); const CHARLIE: AccountId = AccountId::new([3_u8; 32]); +const NON_MEMBER: AccountId = AccountId::new([4_u8; 32]); const AMOUNT: Balance = MIN_BALANCE * 4; const MIN_BALANCE: Balance = 10_000; const TOKEN: TokenId = 1; @@ -43,8 +44,10 @@ pub struct Pop { impl Default for Pop { fn default() -> Self { // Initialising genesis state, providing accounts with an initial balance. + let balances: Vec<(AccountId, u128)> = vec![(ALICE, INIT_AMOUNT), (BOB, INIT_AMOUNT), (CHARLIE, INIT_AMOUNT)]; + ink::env::test::advance_block::(); let ext = BlockBuilder::::new_ext(balances); Self { ext } } @@ -160,6 +163,26 @@ fn members_vote_system_works(mut session: Session) { ); } +#[drink::test(sandbox = Pop)] +fn vote_fails_if_not_a_member(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + session.set_actor(NON_MEMBER); + assert_eq!(vote(&mut session, 0, true), Err(Error::NotAMember) ); + //assert_eq!(last_contract_event(&session), None); + +} + #[drink::test(sandbox = Pop)] fn proposal_enactment_works(mut session: Session) { let _ = env_logger::try_init(); @@ -178,7 +201,12 @@ fn proposal_enactment_works(mut session: Session) { // Charlie vote assert_ok!(vote(&mut session, 0, true)); let next_block = block(&mut session).saturating_add(VOTING_PERIOD); - let mut now = block(&mut session); + let mut now = ink::env::block_number::();//block(&mut session); + println!("Current block number: {:?}", now); + ink::env::test::set_block_number::(next_block); + let block = block(&mut session); + now = ink::env::block_number::(); + println!("new block number_1: {:?}\nnew block number_2: {:?}", block,now); //ToDo: move to next block to end the voting_period From b0d6194a3c8f49d9c5c59eb742417f3b69a97764 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 17:24:34 +0900 Subject: [PATCH 21/56] Reverted some changes --- pop-api/examples/dao/src/lib.rs | 12 ++++++------ pop-api/examples/dao/src/tests.rs | 5 +++++ 2 files changed, 11 insertions(+), 6 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c6d09ea6d..9409666fc 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -145,7 +145,7 @@ mod dao { ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let current_block = ink::env::block_number::(); + let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; @@ -181,8 +181,8 @@ mod dao { #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); - let current_block = ink::env::block_number::(); - let now = ink::env::block_number::(); + let current_block = self.env().block_number(); + let now = self.env().block_number(); let proposal = self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; @@ -232,7 +232,7 @@ mod dao { .vote_end; // Check the voting period - if ink::env::block_number::() <= vote_end { + if self.env().block_number() <= vote_end { return Err(Error::VotingPeriodNotEnded); } @@ -246,7 +246,7 @@ mod dao { if proposal.yes_votes > proposal.no_votes { let contract = self.env().account_id(); - // ToDo: Check that there is enough funds in the treasury + // Execute the proposal api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; @@ -309,7 +309,7 @@ mod dao { #[ink(message)] pub fn get_block_number(&mut self) -> BlockNumber { - //ink::env::block_number::() + //self.env().block_number() self.env().block_number() } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 143b06f8e..5a0b8f874 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -200,11 +200,16 @@ fn proposal_enactment_works(mut session: Session) { session.set_actor(CHARLIE); // Charlie vote assert_ok!(vote(&mut session, 0, true)); + let next_block = block(&mut session).saturating_add(VOTING_PERIOD); let mut now = ink::env::block_number::();//block(&mut session); println!("Current block number: {:?}", now); + ink::env::test::set_block_number::(next_block); + + // this variable is coming from the contract, but is not changed by set_block_number let block = block(&mut session); + now = ink::env::block_number::(); println!("new block number_1: {:?}\nnew block number_2: {:?}", block,now); From f8910ce9be77e1dd2297fdf654ab8033db743a67 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 17:25:32 +0900 Subject: [PATCH 22/56] Reverted some changes --- pop-api/examples/dao/src/lib.rs | 1 - 1 file changed, 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 9409666fc..c8f936276 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -309,7 +309,6 @@ mod dao { #[ink(message)] pub fn get_block_number(&mut self) -> BlockNumber { - //self.env().block_number() self.env().block_number() } } From 94d884efe62bca3d91a0b6d778c4d58059e4030f Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 17:31:11 +0900 Subject: [PATCH 23/56] Documented the failing test: proposal_enactment_works --- pop-api/examples/dao/src/tests.rs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 5a0b8f874..f712d93bf 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -205,13 +205,14 @@ fn proposal_enactment_works(mut session: Session) { let mut now = ink::env::block_number::();//block(&mut session); println!("Current block number: {:?}", now); + // Changing block number ink::env::test::set_block_number::(next_block); - // this variable is coming from the contract, but is not changed by set_block_number + // This variable is coming from the contract, but is not changed by set_block_number let block = block(&mut session); - + now = ink::env::block_number::(); - println!("new block number_1: {:?}\nnew block number_2: {:?}", block,now); + println!("Non updated blocknumber: {:?}\nExpected updated blocknumber_2: {:?}", block,now); //ToDo: move to next block to end the voting_period From 00d76c06962fa63979ae07d29df303f8704e8a03 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 18:52:12 +0900 Subject: [PATCH 24/56] Another test... --- pop-api/examples/dao/src/tests.rs | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index f712d93bf..391df7963 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -163,6 +163,26 @@ fn members_vote_system_works(mut session: Session) { ); } +#[drink::test(sandbox = Pop)] +fn double_vote_fails(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + session.set_actor(CHARLIE); + // Charlie tries to vote twice for the same proposal + assert_ok!(vote(&mut session, 0, true)); + assert_eq!(vote(&mut session, 0, false), Err(Error::AlreadyVoted)); +} + #[drink::test(sandbox = Pop)] fn vote_fails_if_not_a_member(mut session: Session) { let _ = env_logger::try_init(); From 8de7a7d2d21456a2d5bb3ee5f7aaa0f73dea7551 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 19 Dec 2024 19:25:39 +0900 Subject: [PATCH 25/56] tests --- pop-api/examples/dao/src/lib.rs | 24 ++++++++++++------------ pop-api/examples/dao/src/tests.rs | 26 ++++++++++++++------------ 2 files changed, 26 insertions(+), 24 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c8f936276..2700511ca 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -30,10 +30,10 @@ mod dao { pub description: Vec, // Beginnning of the voting period for this proposal - pub vote_start: BlockNumber, + pub vote_start: Timestamp, // End of the voting period for this proposal - pub vote_end: BlockNumber, + pub vote_end: Timestamp, // Balance representing the total votes for this proposal pub yes_votes: Balance, @@ -63,7 +63,7 @@ mod dao { pub voting_power: Balance, // Keeps track of the last vote casted by the member - pub last_vote: BlockNumber, + pub last_vote: Timestamp, } /// Structure of a DAO (Decentralized Autonomous Organization) @@ -81,7 +81,7 @@ mod dao { last_votes: Mapping, // Duration of the voting period - voting_period: BlockNumber, + voting_period: Timestamp, // Identifier of the Psp22 token associated with this DAO token_id: TokenId, @@ -93,7 +93,7 @@ mod dao { #[ink(event)] pub struct Voted { pub who: Option, - pub when: Option, + pub when: Option, } impl Dao { @@ -108,7 +108,7 @@ mod dao { #[ink(constructor, payable)] pub fn new( token_id: TokenId, - voting_period: BlockNumber, + voting_period: Timestamp, min_balance: Balance, ) -> Result { let instance = Self { @@ -145,7 +145,7 @@ mod dao { ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let current_block = self.env().block_number(); + let current_block = self.env().block_timestamp(); let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; @@ -181,8 +181,8 @@ mod dao { #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); - let current_block = self.env().block_number(); - let now = self.env().block_number(); + let current_block = self.env().block_timestamp(); + let now = self.env().block_timestamp(); let proposal = self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; @@ -232,7 +232,7 @@ mod dao { .vote_end; // Check the voting period - if self.env().block_number() <= vote_end { + if self.env().block_timestamp() <= vote_end { return Err(Error::VotingPeriodNotEnded); } @@ -308,8 +308,8 @@ mod dao { } #[ink(message)] - pub fn get_block_number(&mut self) -> BlockNumber { - self.env().block_number() + pub fn get_block_timestamp(&mut self) -> Option { + Some(self.env().block_timestamp()) } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 391df7963..6a399a68e 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -31,7 +31,7 @@ const NON_MEMBER: AccountId = AccountId::new([4_u8; 32]); const AMOUNT: Balance = MIN_BALANCE * 4; const MIN_BALANCE: Balance = 10_000; const TOKEN: TokenId = 1; -const VOTING_PERIOD: u32 = 10; +const VOTING_PERIOD: u64 = 10; #[drink::contract_bundle_provider] enum BundleProvider {} @@ -47,7 +47,7 @@ impl Default for Pop { let balances: Vec<(AccountId, u128)> = vec![(ALICE, INIT_AMOUNT), (BOB, INIT_AMOUNT), (CHARLIE, INIT_AMOUNT)]; - ink::env::test::advance_block::(); + //ink::env::test::advance_block::(); let ext = BlockBuilder::::new_ext(balances); Self { ext } } @@ -153,7 +153,7 @@ fn members_vote_system_works(mut session: Session) { session.set_actor(CHARLIE); // Charlie vote - let now = block(&mut session); + let now = block(&mut session).unwrap(); assert_ok!(vote(&mut session, 0, true)); @@ -221,20 +221,22 @@ fn proposal_enactment_works(mut session: Session) { // Charlie vote assert_ok!(vote(&mut session, 0, true)); - let next_block = block(&mut session).saturating_add(VOTING_PERIOD); - let mut now = ink::env::block_number::();//block(&mut session); - println!("Current block number: {:?}", now); + let next_block = block(&mut session).unwrap().saturating_add(VOTING_PERIOD); + let mut now = ink::env::block_timestamp::();//block(&mut session); + let block1 = block(&mut session); + println!("Non updated blocknumber: {:?}\nExpected updated blocknumber_2: {:?}", block1,now); + // Changing block number - ink::env::test::set_block_number::(next_block); + ink::env::test::set_block_timestamp::(next_block); - // This variable is coming from the contract, but is not changed by set_block_number + // This variable is coming from the contract, but is not changed by set_block_timestamp let block = block(&mut session); - now = ink::env::block_number::(); + now = ink::env::block_timestamp::(); println!("Non updated blocknumber: {:?}\nExpected updated blocknumber_2: {:?}", block,now); - //ToDo: move to next block to end the voting_period + //assert_ok!(execute_proposal(&mut session, 0)); } @@ -263,8 +265,8 @@ fn members(session: &mut Session, account: AccountId) -> Result(session, "get_member", vec![account.to_string()], None) } -fn block(session: &mut Session) -> u32{ - call::(session, "get_block_number", vec![], None).unwrap() +fn block(session: &mut Session) -> Option{ + call::, Error>(session, "get_block_timestamp", vec![], None).unwrap() } fn create_proposal( From 5e916798880ef9de924c0a1586dc1c5f5125998d Mon Sep 17 00:00:00 2001 From: ndkazu Date: Fri, 20 Dec 2024 00:26:04 +0900 Subject: [PATCH 26/56] refactored the code & added another test --- pop-api/examples/dao/Cargo.toml | 10 +-- pop-api/examples/dao/src/lib.rs | 86 ++++++++++++++----------- pop-api/examples/dao/src/tests.rs | 103 ++++++++++++++++++++---------- 3 files changed, 125 insertions(+), 74 deletions(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 9c00efea0..dce2d8a5d 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -1,11 +1,11 @@ [package] -authors = [ "R0GUE " ] +authors = ["R0GUE "] edition = "2021" name = "dao" version = "0.1.0" [dependencies] -ink = { version = "=5.0.0", default-features = false, features = [ "ink-debug" ] } +ink = { version = "=5.0.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } @@ -25,9 +25,9 @@ staging-xcm = { version = "=14.1.0", default-features = false } path = "src/lib.rs" [features] -default = [ "std" ] -e2e-tests = [ ] -ink-as-dependency = [ ] +default = ["std"] +e2e-tests = [] +ink-as-dependency = [] std = [ "ink/std", "pop-api/std", diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 2700511ca..389eeee30 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -22,7 +22,7 @@ mod dao { use super::*; /// Structure of the proposal used by the Dao governance sysytem - #[derive(Debug)] + #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Proposal { @@ -30,10 +30,10 @@ mod dao { pub description: Vec, // Beginnning of the voting period for this proposal - pub vote_start: Timestamp, + pub vote_start: BlockNumber, // End of the voting period for this proposal - pub vote_end: Timestamp, + pub vote_end: BlockNumber, // Balance representing the total votes for this proposal pub yes_votes: Balance, @@ -63,7 +63,7 @@ mod dao { pub voting_power: Balance, // Keeps track of the last vote casted by the member - pub last_vote: Timestamp, + pub last_vote: BlockNumber, } /// Structure of a DAO (Decentralized Autonomous Organization) @@ -72,19 +72,22 @@ mod dao { #[ink(storage)] pub struct Dao { // Funding proposals - proposals: Vec, + proposals: Mapping, // Mapping of AccountId to Member structs, representing DAO membership. members: Mapping, // Mapping tracking the last time each account voted. - last_votes: Mapping, + last_votes: Mapping, // Duration of the voting period - voting_period: Timestamp, + voting_period: BlockNumber, // Identifier of the Psp22 token associated with this DAO token_id: TokenId, + + // Proposals created in the history of the Dao + proposals_created: u32, } /// Defines an event that is emitted @@ -93,7 +96,7 @@ mod dao { #[ink(event)] pub struct Voted { pub who: Option, - pub when: Option, + pub when: Option, } impl Dao { @@ -108,15 +111,16 @@ mod dao { #[ink(constructor, payable)] pub fn new( token_id: TokenId, - voting_period: Timestamp, + voting_period: BlockNumber, min_balance: Balance, ) -> Result { let instance = Self { - proposals: Vec::new(), + proposals: Mapping::default(), members: Mapping::default(), last_votes: Mapping::default(), voting_period, token_id, + proposals_created: 0, }; let contract_id = instance.env().account_id(); api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; @@ -145,8 +149,8 @@ mod dao { ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let current_block = self.env().block_timestamp(); - let proposal_id: u32 = self.proposals.len().try_into().unwrap_or(0u32); + let current_block = self.env().block_number(); + let proposal_id: u32 = self.proposals_created.saturating_add(1); let vote_end = current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; if description.len() >= STRINGLIMIT.into() { @@ -164,7 +168,7 @@ mod dao { proposal_id, }; - self.proposals.push(proposal); + self.proposals.insert(proposal_id, &proposal); self.env() .emit_event(Created { id: proposal_id, creator: caller, admin: contract }); @@ -181,11 +185,11 @@ mod dao { #[ink(message)] pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); - let current_block = self.env().block_timestamp(); - let now = self.env().block_timestamp(); + let current_block = self.env().block_number(); + let now = self.env().block_number(); + // let props = self.proposals.clone(); - let proposal = - self.proposals.get_mut(proposal_id as usize).ok_or(Error::ProposalNotFound)?; + let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; if current_block < proposal.vote_start || current_block > proposal.vote_end { return Err(Error::VotingPeriodEnded); @@ -198,17 +202,16 @@ mod dao { } if approve { - proposal - .yes_votes - .checked_add(member.voting_power) - .ok_or(Error::ArithmeticOverflow)?; + proposal.yes_votes = proposal.yes_votes.saturating_add(member.voting_power); } else { - proposal - .no_votes - .checked_add(member.voting_power) - .ok_or(Error::ArithmeticOverflow)?; + proposal.no_votes = proposal.no_votes.saturating_add(member.voting_power); } + debug_assert!(proposal.yes_votes > 0); + + self.proposals.remove(proposal_id); + self.proposals.insert(proposal_id, &proposal); + self.members.insert( caller.clone(), &Member { voting_power: member.voting_power, last_vote: current_block }, @@ -225,20 +228,16 @@ mod dao { /// - `proposal_id` - Identifier of the proposal #[ink(message)] pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { - let vote_end = self - .proposals - .get(proposal_id as usize) - .ok_or(Error::ProposalNotFound)? - .vote_end; + let vote_end = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?.vote_end; // Check the voting period - if self.env().block_timestamp() <= vote_end { + if self.env().block_number() <= vote_end { return Err(Error::VotingPeriodNotEnded); } // If we've passed the checks, now we can mutably borrow the proposal - let proposal_id_usize = proposal_id as usize; - let proposal = self.proposals.get(proposal_id_usize).ok_or(Error::ProposalNotFound)?; + // let proposal_id_usize = proposal_id as usize; + let proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; if proposal.executed { return Err(Error::ProposalAlreadyExecuted); @@ -246,8 +245,9 @@ mod dao { if proposal.yes_votes > proposal.no_votes { let contract = self.env().account_id(); - + // Execute the proposal + // ToDo: Check that the contract has enough tokens api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { @@ -261,7 +261,7 @@ mod dao { value: proposal.amount, }); - if let Some(proposal) = self.proposals.get_mut(proposal_id_usize) { + if let Some(mut proposal) = self.proposals.get(proposal_id) { proposal.executed = true; } Ok(()) @@ -308,8 +308,20 @@ mod dao { } #[ink(message)] - pub fn get_block_timestamp(&mut self) -> Option { - Some(self.env().block_timestamp()) + pub fn get_block_number(&mut self) -> Option { + Some(self.env().block_number()) + } + + #[ink(message)] + pub fn positive_votes(&mut self, proposal_id: u32) -> Option { + let proposal = &self.proposals.get(proposal_id).unwrap(); + Some(proposal.yes_votes) + } + + #[ink(message)] + pub fn negative_votes(&mut self, proposal_id: u32) -> Option { + let proposal = &self.proposals.get(proposal_id).unwrap(); + Some(proposal.no_votes) } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 6a399a68e..977f9165d 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -10,6 +10,7 @@ use drink::{ AccountId, Balance, Runtime, }, last_contract_event, + sandbox_api::system_api::SystemAPI, session::Session, AssetsAPI, TestExternalities, NO_SALT, }; @@ -18,6 +19,7 @@ use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, }; + use super::*; use crate::dao::{Error, Member, Voted}; @@ -31,7 +33,7 @@ const NON_MEMBER: AccountId = AccountId::new([4_u8; 32]); const AMOUNT: Balance = MIN_BALANCE * 4; const MIN_BALANCE: Balance = 10_000; const TOKEN: TokenId = 1; -const VOTING_PERIOD: u64 = 10; +const VOTING_PERIOD: u32 = 10; #[drink::contract_bundle_provider] enum BundleProvider {} @@ -44,10 +46,10 @@ pub struct Pop { impl Default for Pop { fn default() -> Self { // Initialising genesis state, providing accounts with an initial balance. - + let balances: Vec<(AccountId, u128)> = vec![(ALICE, INIT_AMOUNT), (BOB, INIT_AMOUNT), (CHARLIE, INIT_AMOUNT)]; - //ink::env::test::advance_block::(); + // ink::env::test::advance_block::(); let ext = BlockBuilder::::new_ext(balances); Self { ext } } @@ -59,7 +61,11 @@ drink::impl_sandbox!(Pop, Runtime, ALICE); // Deployment and constructor method tests. fn deploy_with_default(session: &mut Session) -> Result { - deploy(session, "new", vec![TOKEN.to_string(), VOTING_PERIOD.to_string(), MIN_BALANCE.to_string()]) + deploy( + session, + "new", + vec![TOKEN.to_string(), VOTING_PERIOD.to_string(), MIN_BALANCE.to_string()], + ) } #[drink::test(sandbox = Pop)] @@ -67,7 +73,6 @@ fn new_constructor_works(mut session: Session) { let _ = env_logger::try_init(); // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); - println!("{:?}", contract); // Token exists after the deployment. assert!(session.sandbox().asset_exists(&TOKEN)); // Successfully emit event. @@ -130,7 +135,7 @@ fn member_create_proposal_works(mut session: Session) { assert_last_contract_event!( &session, Created { - id: 0, + id: 1, creator: account_id_from_slice(&ALICE), admin: account_id_from_slice(&contract), } @@ -154,13 +159,13 @@ fn members_vote_system_works(mut session: Session) { session.set_actor(CHARLIE); // Charlie vote let now = block(&mut session).unwrap(); - assert_ok!(vote(&mut session, 0, true)); - + assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); + let val = yes_votes(&mut session, 1).unwrap(); } #[drink::test(sandbox = Pop)] @@ -179,8 +184,8 @@ fn double_vote_fails(mut session: Session) { session.set_actor(CHARLIE); // Charlie tries to vote twice for the same proposal - assert_ok!(vote(&mut session, 0, true)); - assert_eq!(vote(&mut session, 0, false), Err(Error::AlreadyVoted)); + assert_ok!(vote(&mut session, 1, true)); + assert_eq!(vote(&mut session, 1, false), Err(Error::AlreadyVoted)); } #[drink::test(sandbox = Pop)] @@ -198,9 +203,7 @@ fn vote_fails_if_not_a_member(mut session: Session) { assert_ok!(create_proposal(&mut session, BOB, amount, description)); session.set_actor(NON_MEMBER); - assert_eq!(vote(&mut session, 0, true), Err(Error::NotAMember) ); - //assert_eq!(last_contract_event(&session), None); - + assert_eq!(vote(&mut session, 1, true), Err(Error::NotAMember)); } #[drink::test(sandbox = Pop)] @@ -213,31 +216,47 @@ fn proposal_enactment_works(mut session: Session) { // Alice create a proposal let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); - let amount = AMOUNT * 3; + let amount = MIN_BALANCE; session.set_actor(ALICE); assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let now = block(&mut session).unwrap(); session.set_actor(CHARLIE); // Charlie vote - assert_ok!(vote(&mut session, 0, true)); + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + ); + // Alice vote + session.set_actor(ALICE); + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + ); + // BOB vote + session.set_actor(BOB); + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + ); + let val = yes_votes(&mut session, 1).unwrap(); let next_block = block(&mut session).unwrap().saturating_add(VOTING_PERIOD); - let mut now = ink::env::block_timestamp::();//block(&mut session); - let block1 = block(&mut session); - println!("Non updated blocknumber: {:?}\nExpected updated blocknumber_2: {:?}", block1,now); - - - // Changing block number - ink::env::test::set_block_timestamp::(next_block); - - // This variable is coming from the contract, but is not changed by set_block_timestamp - let block = block(&mut session); - - now = ink::env::block_timestamp::(); - println!("Non updated blocknumber: {:?}\nExpected updated blocknumber_2: {:?}", block,now); - - //assert_ok!(execute_proposal(&mut session, 0)); + session.sandbox().build_blocks(VOTING_PERIOD + 1); + assert_ok!(execute_proposal(&mut session, 1)); + // Successfully emit event. + assert_last_contract_event!( + &session, + Approval { + owner: account_id_from_slice(&contract), + spender: account_id_from_slice(&contract), + value: MIN_BALANCE, + } + ) } // Deploy the contract with `NO_SALT and `INIT_VALUE`. @@ -265,8 +284,8 @@ fn members(session: &mut Session, account: AccountId) -> Result(session, "get_member", vec![account.to_string()], None) } -fn block(session: &mut Session) -> Option{ - call::, Error>(session, "get_block_timestamp", vec![], None).unwrap() +fn block(session: &mut Session) -> Option { + call::, Error>(session, "get_block_number", vec![], None).unwrap() } fn create_proposal( @@ -301,6 +320,26 @@ fn execute_proposal(session: &mut Session, proposal_id: u32) -> Result<(), call::(session, "execute_proposal", vec![proposal_id.to_string()], None) } +fn yes_votes(session: &mut Session, proposal_id: u32) -> Option { + call::, Error>( + session, + "positive_votes", + vec![proposal_id.to_string()], + None, + ) + .unwrap() +} + +fn no_votes(session: &mut Session, proposal_id: u32) -> Option { + call::, Error>( + session, + "negative_votes", + vec![proposal_id.to_string()], + None, + ) + .unwrap() +} + fn prepare_dao(session: &mut Session, contract: AccountId) -> Result<(), Error> { assert_ok!(session.sandbox().mint_into(&TOKEN, &ALICE, AMOUNT)); assert_ok!(session.sandbox().approve(&TOKEN, &ALICE, &contract.clone(), AMOUNT)); From 77328dfd2740383eda83d51cc1e80d9e3764a598 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Fri, 20 Dec 2024 22:51:34 +0900 Subject: [PATCH 27/56] Treasury balance check & another test --- pop-api/examples/dao/src/lib.rs | 9 ++++++++- pop-api/examples/dao/src/tests.rs | 23 +++++++++++++++++++++++ 2 files changed, 31 insertions(+), 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 389eeee30..e46704d83 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -247,7 +247,13 @@ mod dao { let contract = self.env().account_id(); // Execute the proposal - // ToDo: Check that the contract has enough tokens + let treasury_balance = api::balance_of(self.token_id, contract.clone()).unwrap_or_default(); + if treasury_balance < proposal.amount { + // Not enough tokens in treasury to distribute + return Err(Error::TryAgainLater); + } + + api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { @@ -337,6 +343,7 @@ mod dao { ProposalAlreadyExecuted, ProposalRejected, StringLimitReached, + TryAgainLater, None, Psp22(Psp22Error), } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 977f9165d..3bf830932 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -188,6 +188,29 @@ fn double_vote_fails(mut session: Session) { assert_eq!(vote(&mut session, 1, false), Err(Error::AlreadyVoted)); } +#[drink::test(sandbox = Pop)] +fn vote_fails_if_voting_period_ended(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = AMOUNT * 3; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + // Moving to blocks beyond voting period + let next_block = block(&mut session).unwrap().saturating_add(VOTING_PERIOD); + session.sandbox().build_blocks(VOTING_PERIOD + 1); + + session.set_actor(CHARLIE); + // Charlie tries to vote + assert_eq!(vote(&mut session, 1, true), Err(Error::VotingPeriodEnded)); +} + #[drink::test(sandbox = Pop)] fn vote_fails_if_not_a_member(mut session: Session) { let _ = env_logger::try_init(); From a8206d44da20398dfb87682908dca45fa16ef031 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sat, 21 Dec 2024 00:46:24 +0900 Subject: [PATCH 28/56] Another test --- pop-api/examples/dao/src/lib.rs | 7 ++-- pop-api/examples/dao/src/tests.rs | 56 ++++++++++++++++++++++++------- 2 files changed, 47 insertions(+), 16 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index e46704d83..99f56c5a0 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -101,7 +101,6 @@ mod dao { impl Dao { /// Instantiate a new Dao contract and create the associated token - /// /// # Parameters: /// - `token_id` - The identifier of the token to be created /// - `voting_period` - Amount of blocks during which members can cast their votes @@ -195,7 +194,7 @@ mod dao { return Err(Error::VotingPeriodEnded); } - let member = self.members.get(caller.clone()).ok_or(Error::NotAMember)?; + let member = self.members.get(caller).ok_or(Error::NotAMember)?; if member.last_vote >= proposal.vote_start { return Err(Error::AlreadyVoted); @@ -213,7 +212,7 @@ mod dao { self.proposals.insert(proposal_id, &proposal); self.members.insert( - caller.clone(), + caller, &Member { voting_power: member.voting_power, last_vote: current_block }, ); @@ -247,7 +246,7 @@ mod dao { let contract = self.env().account_id(); // Execute the proposal - let treasury_balance = api::balance_of(self.token_id, contract.clone()).unwrap_or_default(); + let treasury_balance = api::balance_of(self.token_id, contract).unwrap_or_default(); if treasury_balance < proposal.amount { // Not enough tokens in treasury to distribute return Err(Error::TryAgainLater); diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 3bf830932..7211d6fdb 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -1,20 +1,13 @@ use drink::{ - assert_err, assert_last_contract_event, assert_ok, call, + assert_last_contract_event, assert_ok, call, devnet::{ account_id_from_slice, - error::{ - v0::{ApiError::*, ArithmeticError::*}, - Assets, - AssetsError::*, - }, AccountId, Balance, Runtime, }, - last_contract_event, sandbox_api::system_api::SystemAPI, session::Session, AssetsAPI, TestExternalities, NO_SALT, }; -use ink::scale::Encode; use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, @@ -165,7 +158,6 @@ fn members_vote_system_works(mut session: Session) { &session, Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); - let val = yes_votes(&mut session, 1).unwrap(); } #[drink::test(sandbox = Pop)] @@ -203,7 +195,6 @@ fn vote_fails_if_voting_period_ended(mut session: Session) { assert_ok!(create_proposal(&mut session, BOB, amount, description)); // Moving to blocks beyond voting period - let next_block = block(&mut session).unwrap().saturating_add(VOTING_PERIOD); session.sandbox().build_blocks(VOTING_PERIOD + 1); session.set_actor(CHARLIE); @@ -266,8 +257,6 @@ fn proposal_enactment_works(mut session: Session) { Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } ); - let val = yes_votes(&mut session, 1).unwrap(); - let next_block = block(&mut session).unwrap().saturating_add(VOTING_PERIOD); session.sandbox().build_blocks(VOTING_PERIOD + 1); assert_ok!(execute_proposal(&mut session, 1)); @@ -282,6 +271,49 @@ fn proposal_enactment_works(mut session: Session) { ) } +#[drink::test(sandbox = Pop)] +fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = MIN_BALANCE; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + let now = block(&mut session).unwrap(); + session.set_actor(CHARLIE); + // Charlie vote + assert_ok!(vote(&mut session, 1, false)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + ); + // Alice vote + session.set_actor(ALICE); + assert_ok!(vote(&mut session, 1, false)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + ); + // BOB vote + session.set_actor(BOB); + assert_ok!(vote(&mut session, 1, false)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + ); + + session.sandbox().build_blocks(VOTING_PERIOD + 1); + + assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalRejected)); + +} + // Deploy the contract with `NO_SALT and `INIT_VALUE`. fn deploy( session: &mut Session, From 56e622d9be0d8a27915f6bdc726af8407cd59025 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sat, 21 Dec 2024 00:47:45 +0900 Subject: [PATCH 29/56] cargo fmt --- pop-api/examples/dao/src/lib.rs | 7 +++---- pop-api/examples/dao/src/tests.rs | 8 ++------ 2 files changed, 5 insertions(+), 10 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 99f56c5a0..8bfb7d32c 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -248,11 +248,10 @@ mod dao { // Execute the proposal let treasury_balance = api::balance_of(self.token_id, contract).unwrap_or_default(); if treasury_balance < proposal.amount { - // Not enough tokens in treasury to distribute - return Err(Error::TryAgainLater); - } + // Not enough tokens in treasury to distribute + return Err(Error::TryAgainLater); + } - api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 7211d6fdb..382753630 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -1,9 +1,6 @@ use drink::{ assert_last_contract_event, assert_ok, call, - devnet::{ - account_id_from_slice, - AccountId, Balance, Runtime, - }, + devnet::{account_id_from_slice, AccountId, Balance, Runtime}, sandbox_api::system_api::SystemAPI, session::Session, AssetsAPI, TestExternalities, NO_SALT, @@ -195,7 +192,7 @@ fn vote_fails_if_voting_period_ended(mut session: Session) { assert_ok!(create_proposal(&mut session, BOB, amount, description)); // Moving to blocks beyond voting period - session.sandbox().build_blocks(VOTING_PERIOD + 1); + session.sandbox().build_blocks(VOTING_PERIOD + 1); session.set_actor(CHARLIE); // Charlie tries to vote @@ -311,7 +308,6 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { session.sandbox().build_blocks(VOTING_PERIOD + 1); assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalRejected)); - } // Deploy the contract with `NO_SALT and `INIT_VALUE`. From 4ad34f4d7380fafe92575fb77f8ec8a32b68370e Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sat, 21 Dec 2024 01:09:26 +0900 Subject: [PATCH 30/56] Final test --- pop-api/examples/dao/src/lib.rs | 11 +++--- pop-api/examples/dao/src/tests.rs | 56 ++++++++++++++++++++++++++++++- 2 files changed, 61 insertions(+), 6 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 8bfb7d32c..91f0bd4e7 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -236,9 +236,9 @@ mod dao { // If we've passed the checks, now we can mutably borrow the proposal // let proposal_id_usize = proposal_id as usize; - let proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; + let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - if proposal.executed { + if proposal.executed == true { return Err(Error::ProposalAlreadyExecuted); } @@ -265,9 +265,10 @@ mod dao { value: proposal.amount, }); - if let Some(mut proposal) = self.proposals.get(proposal_id) { - proposal.executed = true; - } + proposal.executed = true; + + self.proposals.remove(proposal_id); + self.proposals.insert(proposal_id, &proposal); Ok(()) } else { Err(Error::ProposalRejected) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 382753630..9bda725e3 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -265,7 +265,61 @@ fn proposal_enactment_works(mut session: Session) { spender: account_id_from_slice(&contract), value: MIN_BALANCE, } - ) + ); +} + +#[drink::test(sandbox = Pop)] +fn same_proposal_consecutive_claim_fails(mut session: Session) { + let _ = env_logger::try_init(); + // Deploy a new contract. + let contract = deploy_with_default(&mut session).unwrap(); + // Prepare voters accounts + let _ = prepare_dao(&mut session, contract.clone()); + + // Alice create a proposal + let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); + let amount = MIN_BALANCE; + session.set_actor(ALICE); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + let now = block(&mut session).unwrap(); + session.set_actor(CHARLIE); + // Charlie vote + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + ); + // Alice vote + session.set_actor(ALICE); + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + ); + // BOB vote + session.set_actor(BOB); + assert_ok!(vote(&mut session, 1, true)); + assert_last_contract_event!( + &session, + Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + ); + + session.sandbox().build_blocks(VOTING_PERIOD + 1); + + assert_ok!(execute_proposal(&mut session, 1)); + // Successfully emit event. + assert_last_contract_event!( + &session, + Approval { + owner: account_id_from_slice(&contract), + spender: account_id_from_slice(&contract), + value: MIN_BALANCE, + } + ); + session.sandbox().build_block(); + // Second consecutive claim for same proposal fails + assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalAlreadyExecuted)); } #[drink::test(sandbox = Pop)] From 0cd95731272e951aa8c93cf65100929fb6256add Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 24 Dec 2024 16:05:12 +0900 Subject: [PATCH 31/56] Applied some fixes related to the code review --- pop-api/examples/dao/src/lib.rs | 83 ++++++++++++++----------------- pop-api/examples/dao/src/tests.rs | 4 +- 2 files changed, 38 insertions(+), 49 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 91f0bd4e7..87d5a3cda 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -13,7 +13,6 @@ use pop_api::{ }, }; -pub const STRINGLIMIT: u8 = u8::MAX; #[cfg(test)] mod tests; @@ -44,7 +43,7 @@ mod dao { // Flag that indicates if the proposal was executed pub executed: bool, - // AccountId of the recipient of the proposal + // The recipient of the proposal pub beneficiary: AccountId, // Amount of tokens to be awarded to the beneficiary @@ -101,6 +100,7 @@ mod dao { impl Dao { /// Instantiate a new Dao contract and create the associated token + /// # Parameters: /// - `token_id` - The identifier of the token to be created /// - `voting_period` - Amount of blocks during which members can cast their votes @@ -151,9 +151,9 @@ mod dao { let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals_created.saturating_add(1); let vote_end = - current_block.checked_add(self.voting_period).ok_or(Error::ArithmeticOverflow)?; - if description.len() >= STRINGLIMIT.into() { - return Err(Error::StringLimitReached); + current_block.saturating_add(self.voting_period); + if description.len() >= u8::MAX.into() { + return Err(Error::ExceedeMaxDescriptionLength); } let proposal = Proposal { description, @@ -185,38 +185,34 @@ mod dao { pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); let current_block = self.env().block_number(); - let now = self.env().block_number(); - // let props = self.proposals.clone(); - - let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; + let proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; if current_block < proposal.vote_start || current_block > proposal.vote_end { return Err(Error::VotingPeriodEnded); } - let member = self.members.get(caller).ok_or(Error::NotAMember)?; + let member = self.members.get(caller).ok_or(Error::MemberNotFound)?; if member.last_vote >= proposal.vote_start { return Err(Error::AlreadyVoted); } - if approve { - proposal.yes_votes = proposal.yes_votes.saturating_add(member.voting_power); - } else { - proposal.no_votes = proposal.no_votes.saturating_add(member.voting_power); - } - - debug_assert!(proposal.yes_votes > 0); + let votes = match approve { + true => proposal.yes_votes, + false => proposal.no_votes + }; + + let _ = votes.saturating_add(member.voting_power); - self.proposals.remove(proposal_id); self.proposals.insert(proposal_id, &proposal); self.members.insert( caller, &Member { voting_power: member.voting_power, last_vote: current_block }, ); + self.last_votes.insert(caller, ¤t_block); - self.env().emit_event(Voted { who: Some(caller), when: Some(now) }); + self.env().emit_event(Voted { who: Some(caller), when: Some(current_block) }); Ok(()) } @@ -227,19 +223,15 @@ mod dao { /// - `proposal_id` - Identifier of the proposal #[ink(message)] pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { - let vote_end = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?.vote_end; + let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; // Check the voting period - if self.env().block_number() <= vote_end { + if self.env().block_number() <= proposal.vote_end { return Err(Error::VotingPeriodNotEnded); } - // If we've passed the checks, now we can mutably borrow the proposal - // let proposal_id_usize = proposal_id as usize; - let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - if proposal.executed == true { - return Err(Error::ProposalAlreadyExecuted); + return Err(Error::ProposalExecuted); } if proposal.yes_votes > proposal.no_votes { @@ -248,8 +240,7 @@ mod dao { // Execute the proposal let treasury_balance = api::balance_of(self.token_id, contract).unwrap_or_default(); if treasury_balance < proposal.amount { - // Not enough tokens in treasury to distribute - return Err(Error::TryAgainLater); + return Err(Error::NotEnoughFundsAvailable); } api::transfer(self.token_id, proposal.beneficiary, proposal.amount) @@ -267,7 +258,6 @@ mod dao { proposal.executed = true; - self.proposals.remove(proposal_id); self.proposals.insert(proposal_id, &proposal); Ok(()) } else { @@ -294,7 +284,7 @@ mod dao { self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); let voting_power = - member.voting_power.checked_add(amount).ok_or(Error::ArithmeticOverflow)?; + member.voting_power.saturating_add(amount); self.members .insert(caller, &Member { voting_power, last_vote: member.last_vote }); @@ -312,37 +302,36 @@ mod dao { self.members.get(account).unwrap_or(Member { voting_power: 0, last_vote: 0 }) } - #[ink(message)] - pub fn get_block_number(&mut self) -> Option { - Some(self.env().block_number()) - } #[ink(message)] pub fn positive_votes(&mut self, proposal_id: u32) -> Option { - let proposal = &self.proposals.get(proposal_id).unwrap(); - Some(proposal.yes_votes) + match &self.proposals.get(proposal_id){ + Some(x) => Some(x.yes_votes), + _ => None, + } + } #[ink(message)] pub fn negative_votes(&mut self, proposal_id: u32) -> Option { - let proposal = &self.proposals.get(proposal_id).unwrap(); - Some(proposal.no_votes) - } - } + match &self.proposals.get(proposal_id){ + Some(x) => Some(x.no_votes), + _ => None, + } + }} #[derive(Debug, PartialEq, Eq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] pub enum Error { - ArithmeticOverflow, ProposalNotFound, VotingPeriodEnded, - NotAMember, + MemberNotFound, AlreadyVoted, VotingPeriodNotEnded, - ProposalAlreadyExecuted, + ProposalExecuted, ProposalRejected, - StringLimitReached, - TryAgainLater, + ExceedeMaxDescriptionLength, + NotEnoughFundsAvailable, None, Psp22(Psp22Error), } @@ -353,12 +342,12 @@ mod dao { } } - impl From for Psp22Error { + /*impl From for Psp22Error { fn from(error: Error) -> Self { match error { Error::Psp22(psp22_error) => psp22_error, _ => Psp22Error::Custom(String::from("Unknown error")), } } - } + }*/ } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 9bda725e3..5ee3c4989 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -214,7 +214,7 @@ fn vote_fails_if_not_a_member(mut session: Session) { assert_ok!(create_proposal(&mut session, BOB, amount, description)); session.set_actor(NON_MEMBER); - assert_eq!(vote(&mut session, 1, true), Err(Error::NotAMember)); + assert_eq!(vote(&mut session, 1, true), Err(Error::MemberNotFound)); } #[drink::test(sandbox = Pop)] @@ -319,7 +319,7 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { ); session.sandbox().build_block(); // Second consecutive claim for same proposal fails - assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalAlreadyExecuted)); + assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalExecuted)); } #[drink::test(sandbox = Pop)] From be3759f7e4784e9ac5cf36776d3f1b2b5009c324 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 24 Dec 2024 17:08:41 +0900 Subject: [PATCH 32/56] Added ProposalStatus enum --- pop-api/examples/dao/src/lib.rs | 36 ++++++++++++++++++++++++++------- 1 file changed, 29 insertions(+), 7 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 87d5a3cda..c90f5df15 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -20,6 +20,16 @@ mod tests; mod dao { use super::*; + #[derive(Debug, Clone, PartialEq)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + pub enum ProposalStatus{ + SUBMITTED, + APPROVED, + REJECTED, + EXECUTED, + } + /// Structure of the proposal used by the Dao governance sysytem #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] @@ -41,7 +51,7 @@ mod dao { pub no_votes: Balance, // Flag that indicates if the proposal was executed - pub executed: bool, + pub status: ProposalStatus, // The recipient of the proposal pub beneficiary: AccountId, @@ -161,7 +171,7 @@ mod dao { vote_end, yes_votes: 0, no_votes: 0, - executed: false, + status: ProposalStatus::SUBMITTED, beneficiary, amount, proposal_id, @@ -185,15 +195,27 @@ mod dao { pub fn vote(&mut self, proposal_id: u32, approve: bool) -> Result<(), Error> { let caller = self.env().caller(); let current_block = self.env().block_number(); - let proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; + let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - if current_block < proposal.vote_start || current_block > proposal.vote_end { + if current_block > proposal.vote_end { + match proposal.status{ + ProposalStatus::SUBMITTED => { + if proposal.yes_votes > proposal.no_votes { + proposal.status = ProposalStatus::APPROVED; + }else{ + proposal.status = ProposalStatus::REJECTED; + } + }, + _ => () + }; + return Err(Error::VotingPeriodEnded); } let member = self.members.get(caller).ok_or(Error::MemberNotFound)?; if member.last_vote >= proposal.vote_start { + return Err(Error::AlreadyVoted); } @@ -201,7 +223,7 @@ mod dao { true => proposal.yes_votes, false => proposal.no_votes }; - + let _ = votes.saturating_add(member.voting_power); self.proposals.insert(proposal_id, &proposal); @@ -230,7 +252,7 @@ mod dao { return Err(Error::VotingPeriodNotEnded); } - if proposal.executed == true { + if proposal.status == ProposalStatus::EXECUTED { return Err(Error::ProposalExecuted); } @@ -256,7 +278,7 @@ mod dao { value: proposal.amount, }); - proposal.executed = true; + proposal.status = ProposalStatus::EXECUTED; self.proposals.insert(proposal_id, &proposal); Ok(()) From a34f0d1a28f87f1896a947662a24a56d7d2b31ff Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 24 Dec 2024 17:29:33 +0900 Subject: [PATCH 33/56] Added descriptions for errors --- pop-api/examples/dao/src/lib.rs | 20 +++++++++++++++++++- 1 file changed, 19 insertions(+), 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c90f5df15..8ae188555 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -345,16 +345,34 @@ mod dao { #[derive(Debug, PartialEq, Eq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] pub enum Error { + /// This proposal does not exists ProposalNotFound, + + /// The end of the voting period has been reached VotingPeriodEnded, + + /// User is not a member of this Dao MemberNotFound, + + /// User already voted for this proposal AlreadyVoted, + + /// The voting period for this proposal is still ongoing VotingPeriodNotEnded, + + /// This proposal has already been executed ProposalExecuted, + + /// This proposal has been rejected ProposalRejected, + + /// The proposal description is too long ExceedeMaxDescriptionLength, + + /// There are not enough funds in the Dao treasury NotEnoughFundsAvailable, - None, + + /// PSP22 specific error Psp22(Psp22Error), } From 5e8e8e46b241fbc85c86a2de531b47ec894aaf21 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 24 Dec 2024 17:39:17 +0900 Subject: [PATCH 34/56] Review correction --- pop-api/examples/dao/src/lib.rs | 73 ++++++++++++++++----------------- 1 file changed, 35 insertions(+), 38 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 8ae188555..c13e557ec 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -23,7 +23,7 @@ mod dao { #[derive(Debug, Clone, PartialEq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] - pub enum ProposalStatus{ + pub enum ProposalStatus { SUBMITTED, APPROVED, REJECTED, @@ -110,7 +110,7 @@ mod dao { impl Dao { /// Instantiate a new Dao contract and create the associated token - + /// # Parameters: /// - `token_id` - The identifier of the token to be created /// - `voting_period` - Amount of blocks during which members can cast their votes @@ -160,8 +160,7 @@ mod dao { let contract = self.env().account_id(); let current_block = self.env().block_number(); let proposal_id: u32 = self.proposals_created.saturating_add(1); - let vote_end = - current_block.saturating_add(self.voting_period); + let vote_end = current_block.saturating_add(self.voting_period); if description.len() >= u8::MAX.into() { return Err(Error::ExceedeMaxDescriptionLength); } @@ -198,33 +197,31 @@ mod dao { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; if current_block > proposal.vote_end { - match proposal.status{ - ProposalStatus::SUBMITTED => { + match proposal.status { + ProposalStatus::SUBMITTED => if proposal.yes_votes > proposal.no_votes { proposal.status = ProposalStatus::APPROVED; - }else{ + } else { proposal.status = ProposalStatus::REJECTED; - } - }, - _ => () + }, + _ => (), }; - + return Err(Error::VotingPeriodEnded); } let member = self.members.get(caller).ok_or(Error::MemberNotFound)?; if member.last_vote >= proposal.vote_start { - return Err(Error::AlreadyVoted); } - let votes = match approve { - true => proposal.yes_votes, - false => proposal.no_votes - }; - - let _ = votes.saturating_add(member.voting_power); + let votes = match approve { + true => proposal.yes_votes, + false => proposal.no_votes, + }; + + let _ = votes.saturating_add(member.voting_power); self.proposals.insert(proposal_id, &proposal); @@ -260,10 +257,12 @@ mod dao { let contract = self.env().account_id(); // Execute the proposal - let treasury_balance = api::balance_of(self.token_id, contract).unwrap_or_default(); - if treasury_balance < proposal.amount { - return Err(Error::NotEnoughFundsAvailable); - } + let _treasury_balance = match api::balance_of(self.token_id, contract) { + Ok(val) if val > proposal.amount => val, + _ => { + return Err(Error::NotEnoughFundsAvailable); + }, + }; api::transfer(self.token_id, proposal.beneficiary, proposal.amount) .map_err(Psp22Error::from)?; @@ -305,8 +304,7 @@ mod dao { let member = self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); - let voting_power = - member.voting_power.saturating_add(amount); + let voting_power = member.voting_power.saturating_add(amount); self.members .insert(caller, &Member { voting_power, last_vote: member.last_vote }); @@ -324,23 +322,22 @@ mod dao { self.members.get(account).unwrap_or(Member { voting_power: 0, last_vote: 0 }) } - #[ink(message)] pub fn positive_votes(&mut self, proposal_id: u32) -> Option { - match &self.proposals.get(proposal_id){ + match &self.proposals.get(proposal_id) { Some(x) => Some(x.yes_votes), _ => None, } - } #[ink(message)] pub fn negative_votes(&mut self, proposal_id: u32) -> Option { - match &self.proposals.get(proposal_id){ + match &self.proposals.get(proposal_id) { Some(x) => Some(x.no_votes), _ => None, } - }} + } + } #[derive(Debug, PartialEq, Eq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] @@ -369,7 +366,7 @@ mod dao { /// The proposal description is too long ExceedeMaxDescriptionLength, - /// There are not enough funds in the Dao treasury + /// There are not enough funds in the Dao treasury NotEnoughFundsAvailable, /// PSP22 specific error @@ -382,12 +379,12 @@ mod dao { } } - /*impl From for Psp22Error { - fn from(error: Error) -> Self { - match error { - Error::Psp22(psp22_error) => psp22_error, - _ => Psp22Error::Custom(String::from("Unknown error")), - } - } - }*/ + // impl From for Psp22Error { + // fn from(error: Error) -> Self { + // match error { + // Error::Psp22(psp22_error) => psp22_error, + // _ => Psp22Error::Custom(String::from("Unknown error")), + // } + // } + // } } From ede1283900f32aed7cc55f107111f37d3c82665b Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 24 Dec 2024 18:17:07 +0900 Subject: [PATCH 35/56] cargo clippy --- pop-api/examples/dao/src/lib.rs | 48 ++++++++++++++------------------- 1 file changed, 20 insertions(+), 28 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c13e557ec..1838afcc0 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -24,10 +24,10 @@ mod dao { #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub enum ProposalStatus { - SUBMITTED, - APPROVED, - REJECTED, - EXECUTED, + Submitted, + Approved, + Rejected, + Executed, } /// Structure of the proposal used by the Dao governance sysytem @@ -50,7 +50,7 @@ mod dao { // Balance representing the total votes against this proposal pub no_votes: Balance, - // Flag that indicates if the proposal was executed + // Flag that indicates if the proposal was Executed pub status: ProposalStatus, // The recipient of the proposal @@ -110,7 +110,7 @@ mod dao { impl Dao { /// Instantiate a new Dao contract and create the associated token - + /// /// # Parameters: /// - `token_id` - The identifier of the token to be created /// - `voting_period` - Amount of blocks during which members can cast their votes @@ -170,7 +170,7 @@ mod dao { vote_end, yes_votes: 0, no_votes: 0, - status: ProposalStatus::SUBMITTED, + status: ProposalStatus::Submitted, beneficiary, amount, proposal_id, @@ -197,14 +197,12 @@ mod dao { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; if current_block > proposal.vote_end { - match proposal.status { - ProposalStatus::SUBMITTED => - if proposal.yes_votes > proposal.no_votes { - proposal.status = ProposalStatus::APPROVED; - } else { - proposal.status = ProposalStatus::REJECTED; - }, - _ => (), + if proposal.status == ProposalStatus::Submitted { + if proposal.yes_votes > proposal.no_votes { + proposal.status = ProposalStatus::Approved; + } else { + proposal.status = ProposalStatus::Rejected; + } }; return Err(Error::VotingPeriodEnded); @@ -236,7 +234,7 @@ mod dao { Ok(()) } - /// Enact a proposal approved by the Dao members + /// Enact a proposal Approved by the Dao members /// /// # Parameters /// - `proposal_id` - Identifier of the proposal @@ -249,7 +247,7 @@ mod dao { return Err(Error::VotingPeriodNotEnded); } - if proposal.status == ProposalStatus::EXECUTED { + if proposal.status == ProposalStatus::Executed { return Err(Error::ProposalExecuted); } @@ -277,7 +275,7 @@ mod dao { value: proposal.amount, }); - proposal.status = ProposalStatus::EXECUTED; + proposal.status = ProposalStatus::Executed; self.proposals.insert(proposal_id, &proposal); Ok(()) @@ -324,18 +322,12 @@ mod dao { #[ink(message)] pub fn positive_votes(&mut self, proposal_id: u32) -> Option { - match &self.proposals.get(proposal_id) { - Some(x) => Some(x.yes_votes), - _ => None, - } + self.proposals.get(proposal_id).as_ref().map(|x| x.yes_votes) } #[ink(message)] pub fn negative_votes(&mut self, proposal_id: u32) -> Option { - match &self.proposals.get(proposal_id) { - Some(x) => Some(x.no_votes), - _ => None, - } + self.proposals.get(proposal_id).as_ref().map(|x| x.no_votes) } } @@ -357,10 +349,10 @@ mod dao { /// The voting period for this proposal is still ongoing VotingPeriodNotEnded, - /// This proposal has already been executed + /// This proposal has already been Executed ProposalExecuted, - /// This proposal has been rejected + /// This proposal has been Rejected ProposalRejected, /// The proposal description is too long From 0e2db17f2c0efc2fe0ce7a2e76c9e42c8ab0954a Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 10:27:41 +0900 Subject: [PATCH 36/56] update ink version --- pop-api/examples/dao/Cargo.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index dce2d8a5d..568b5a3b2 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -5,7 +5,7 @@ name = "dao" version = "0.1.0" [dependencies] -ink = { version = "=5.0.0", default-features = false, features = ["ink-debug"] } +ink = { version = "=5.1.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } From 21b13c59222aab5f949e0bf419c300717ebb9508 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 10:41:10 +0900 Subject: [PATCH 37/56] cargo clippy --fix --- pop-api/examples/dao/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 1838afcc0..71fbddd13 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,7 +1,7 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] use ink::{ - prelude::{string::String, vec::Vec}, + prelude::vec::Vec, storage::Mapping, }; use pop_api::{ From 34ba35c567762b29a281f84458827bedc40aa0c0 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 10:58:54 +0900 Subject: [PATCH 38/56] Some clean up --- pop-api/examples/dao/src/lib.rs | 5 +---- pop-api/examples/dao/src/tests.rs | 8 +------- 2 files changed, 2 insertions(+), 11 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 71fbddd13..098dd6528 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,9 +1,6 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] -use ink::{ - prelude::vec::Vec, - storage::Mapping, -}; +use ink::{prelude::vec::Vec, storage::Mapping}; use pop_api::{ primitives::TokenId, v0::fungibles::{ diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 5ee3c4989..1195da988 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -36,10 +36,8 @@ pub struct Pop { impl Default for Pop { fn default() -> Self { // Initialising genesis state, providing accounts with an initial balance. - let balances: Vec<(AccountId, u128)> = vec![(ALICE, INIT_AMOUNT), (BOB, INIT_AMOUNT), (CHARLIE, INIT_AMOUNT)]; - // ink::env::test::advance_block::(); let ext = BlockBuilder::::new_ext(balances); Self { ext } } @@ -51,11 +49,7 @@ drink::impl_sandbox!(Pop, Runtime, ALICE); // Deployment and constructor method tests. fn deploy_with_default(session: &mut Session) -> Result { - deploy( - session, - "new", - vec![TOKEN.to_string(), VOTING_PERIOD.to_string(), MIN_BALANCE.to_string()], - ) + deploy(session, "new", vec![TOKEN.to_string(), MIN_BALANCE.to_string()]) } #[drink::test(sandbox = Pop)] From bd11e860dcf41229f8e3a90a429e6fad073dac78 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 15:21:13 +0900 Subject: [PATCH 39/56] All tests pass --- pop-api/examples/dao/src/lib.rs | 30 +++++----------- pop-api/examples/dao/src/tests.rs | 58 +++++++++++++------------------ 2 files changed, 34 insertions(+), 54 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 098dd6528..3c2cc078d 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -211,13 +211,15 @@ mod dao { return Err(Error::AlreadyVoted); } - let votes = match approve { - true => proposal.yes_votes, - false => proposal.no_votes, + match approve { + true => { + proposal.yes_votes = proposal.yes_votes.saturating_add(member.voting_power); + }, + false => { + proposal.no_votes = proposal.no_votes.saturating_add(member.voting_power); + }, }; - let _ = votes.saturating_add(member.voting_power); - self.proposals.insert(proposal_id, &proposal); self.members.insert( @@ -318,13 +320,8 @@ mod dao { } #[ink(message)] - pub fn positive_votes(&mut self, proposal_id: u32) -> Option { - self.proposals.get(proposal_id).as_ref().map(|x| x.yes_votes) - } - - #[ink(message)] - pub fn negative_votes(&mut self, proposal_id: u32) -> Option { - self.proposals.get(proposal_id).as_ref().map(|x| x.no_votes) + pub fn get_proposal(&mut self, proposal_id: u32) -> Option { + self.proposals.get(proposal_id) } } @@ -367,13 +364,4 @@ mod dao { Error::Psp22(error) } } - - // impl From for Psp22Error { - // fn from(error: Error) -> Self { - // match error { - // Error::Psp22(psp22_error) => psp22_error, - // _ => Psp22Error::Custom(String::from("Unknown error")), - // } - // } - // } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 1195da988..52a19cdec 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -11,7 +11,7 @@ use pop_api::{ }; use super::*; -use crate::dao::{Error, Member, Voted}; +use crate::dao::{Error, Member, Proposal, Voted}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -49,7 +49,11 @@ drink::impl_sandbox!(Pop, Runtime, ALICE); // Deployment and constructor method tests. fn deploy_with_default(session: &mut Session) -> Result { - deploy(session, "new", vec![TOKEN.to_string(), MIN_BALANCE.to_string()]) + deploy( + session, + "new", + vec![TOKEN.to_string(), VOTING_PERIOD.to_string(), MIN_BALANCE.to_string()], + ) } #[drink::test(sandbox = Pop)] @@ -142,13 +146,15 @@ fn members_vote_system_works(mut session: Session) { session.set_actor(CHARLIE); // Charlie vote - let now = block(&mut session).unwrap(); + let now = session.sandbox().block_number(); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); + let prop = proposal(&mut session, 1).unwrap(); + let yes = prop.yes_votes; } #[drink::test(sandbox = Pop)] @@ -225,7 +231,7 @@ fn proposal_enactment_works(mut session: Session) { session.set_actor(ALICE); assert_ok!(create_proposal(&mut session, BOB, amount, description)); - let now = block(&mut session).unwrap(); + let now = session.sandbox().block_number(); session.set_actor(CHARLIE); // Charlie vote assert_ok!(vote(&mut session, 1, true)); @@ -276,7 +282,7 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { session.set_actor(ALICE); assert_ok!(create_proposal(&mut session, BOB, amount, description)); - let now = block(&mut session).unwrap(); + let now = session.sandbox().block_number(); session.set_actor(CHARLIE); // Charlie vote assert_ok!(vote(&mut session, 1, true)); @@ -303,17 +309,17 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { assert_ok!(execute_proposal(&mut session, 1)); // Successfully emit event. - assert_last_contract_event!( - &session, - Approval { - owner: account_id_from_slice(&contract), - spender: account_id_from_slice(&contract), - value: MIN_BALANCE, - } - ); - session.sandbox().build_block(); + // assert_last_contract_event!( + // &session, + // Approval { + // owner: account_id_from_slice(&contract), + // spender: account_id_from_slice(&contract), + // value: MIN_BALANCE, + // } + // ); + // session.sandbox().build_block(); // Second consecutive claim for same proposal fails - assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalExecuted)); + // assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalExecuted)); } #[drink::test(sandbox = Pop)] @@ -330,7 +336,7 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { session.set_actor(ALICE); assert_ok!(create_proposal(&mut session, BOB, amount, description)); - let now = block(&mut session).unwrap(); + let now = session.sandbox().block_number(); session.set_actor(CHARLIE); // Charlie vote assert_ok!(vote(&mut session, 1, false)); @@ -383,10 +389,6 @@ fn members(session: &mut Session, account: AccountId) -> Result(session, "get_member", vec![account.to_string()], None) } -fn block(session: &mut Session) -> Option { - call::, Error>(session, "get_block_number", vec![], None).unwrap() -} - fn create_proposal( session: &mut Session, beneficiary: AccountId, @@ -419,20 +421,10 @@ fn execute_proposal(session: &mut Session, proposal_id: u32) -> Result<(), call::(session, "execute_proposal", vec![proposal_id.to_string()], None) } -fn yes_votes(session: &mut Session, proposal_id: u32) -> Option { - call::, Error>( - session, - "positive_votes", - vec![proposal_id.to_string()], - None, - ) - .unwrap() -} - -fn no_votes(session: &mut Session, proposal_id: u32) -> Option { - call::, Error>( +fn proposal(session: &mut Session, proposal_id: u32) -> Option { + call::, Error>( session, - "negative_votes", + "get_proposal", vec![proposal_id.to_string()], None, ) From 905705c9b187453c5e2a58c3694041a9fc08d6bd Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 15:43:39 +0900 Subject: [PATCH 40/56] cargo fmt --- pop-api/examples/dao/src/tests.rs | 21 +++++++++++---------- 1 file changed, 11 insertions(+), 10 deletions(-) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 52a19cdec..6e0044be3 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -155,6 +155,7 @@ fn members_vote_system_works(mut session: Session) { ); let prop = proposal(&mut session, 1).unwrap(); let yes = prop.yes_votes; + assert_eq!(yes > 0, true); } #[drink::test(sandbox = Pop)] @@ -309,17 +310,17 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { assert_ok!(execute_proposal(&mut session, 1)); // Successfully emit event. - // assert_last_contract_event!( - // &session, - // Approval { - // owner: account_id_from_slice(&contract), - // spender: account_id_from_slice(&contract), - // value: MIN_BALANCE, - // } - // ); - // session.sandbox().build_block(); + assert_last_contract_event!( + &session, + Approval { + owner: account_id_from_slice(&contract), + spender: account_id_from_slice(&contract), + value: MIN_BALANCE, + } + ); + session.sandbox().build_block(); // Second consecutive claim for same proposal fails - // assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalExecuted)); + assert_eq!(execute_proposal(&mut session, 1), Err(Error::ProposalExecuted)); } #[drink::test(sandbox = Pop)] From 0841c3289f484a7fd961a0e65e6f7053d9d1be4a Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 21:48:10 +0900 Subject: [PATCH 41/56] Refactored code, implemented Default trait for Proposal --- pop-api/examples/dao/src/lib.rs | 146 ++++++++++++++++++++---------- pop-api/examples/dao/src/tests.rs | 19 +++- 2 files changed, 114 insertions(+), 51 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 3c2cc078d..88f22c3ab 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -35,29 +35,39 @@ mod dao { // Description of the proposal pub description: Vec, - // Beginnning of the voting period for this proposal - pub vote_start: BlockNumber, - - // End of the voting period for this proposal - pub vote_end: BlockNumber, - - // Balance representing the total votes for this proposal - pub yes_votes: Balance, - - // Balance representing the total votes against this proposal - pub no_votes: Balance, - // Flag that indicates if the proposal was Executed pub status: ProposalStatus, - // The recipient of the proposal - pub beneficiary: AccountId, - - // Amount of tokens to be awarded to the beneficiary - pub amount: Balance, - // Identifier of the proposal pub proposal_id: u32, + + // Information relative to voting + pub votes_infos: Option, + + // Information relative to proposal execution if approved + pub transaction_infos: Option, + } + + impl Default for Proposal { + fn default() -> Self { + let fetch_dao = ink::env::get_contract_storage::(&0u32) + .expect("The dao should have been created already"); + + // The dao is supposed to exist at this point + let dao = fetch_dao.unwrap_or_default(); + let voting_period = dao.voting_period; + let current_block = ink::env::block_number::(); + let vote_end = current_block.saturating_add(voting_period); + let votes_infos = + Some(Votes { vote_start: current_block, vote_end, yes_votes: 0, no_votes: 0 }); + Proposal { + description: Vec::new(), + status: ProposalStatus::Submitted, + proposal_id: 0, + votes_infos, + transaction_infos: None, + } + } } /// Representation of a member in the voting system @@ -72,9 +82,37 @@ mod dao { pub last_vote: BlockNumber, } + #[derive(Debug, Clone)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + pub struct Votes { + // Beginnning of the voting period for this proposal + pub vote_start: BlockNumber, + + // End of the voting period for this proposal + pub vote_end: BlockNumber, + + // Balance representing the total votes for this proposal + pub yes_votes: Balance, + + // Balance representing the total votes against this proposal + pub no_votes: Balance, + } + + #[derive(Debug, Clone)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + pub struct Transaction { + // The recipient of the proposal + pub beneficiary: AccountId, + // Amount of tokens to be awarded to the beneficiary + pub amount: Balance, + } + /// Structure of a DAO (Decentralized Autonomous Organization) /// that uses Psp22 to manage the Dao treasury and funds projects /// selected by the members through governance + #[derive(Default)] #[ink(storage)] pub struct Dao { // Funding proposals @@ -151,32 +189,28 @@ mod dao { &mut self, beneficiary: AccountId, amount: Balance, - description: Vec, + mut description: Vec, ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let current_block = self.env().block_number(); - let proposal_id: u32 = self.proposals_created.saturating_add(1); - let vote_end = current_block.saturating_add(self.voting_period); + self.proposals_created = self.proposals_created.saturating_add(1); + if description.len() >= u8::MAX.into() { return Err(Error::ExceedeMaxDescriptionLength); } - let proposal = Proposal { - description, - vote_start: current_block, - vote_end, - yes_votes: 0, - no_votes: 0, - status: ProposalStatus::Submitted, - beneficiary, - amount, - proposal_id, - }; - self.proposals.insert(proposal_id, &proposal); + let mut proposal = Proposal { proposal_id: self.proposals_created, ..Default::default() }; + proposal.description.append(&mut description); + let transaction_infos = Transaction { beneficiary, amount }; + proposal.transaction_infos = Some(transaction_infos); - self.env() - .emit_event(Created { id: proposal_id, creator: caller, admin: contract }); + self.proposals.insert(proposal.proposal_id, &proposal); + + self.env().emit_event(Created { + id: proposal.proposal_id, + creator: caller, + admin: contract, + }); Ok(()) } @@ -192,10 +226,11 @@ mod dao { let caller = self.env().caller(); let current_block = self.env().block_number(); let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; + let mut votes_infos = proposal.votes_infos.ok_or(Error::WrongContract)?; - if current_block > proposal.vote_end { + if current_block > votes_infos.vote_end { if proposal.status == ProposalStatus::Submitted { - if proposal.yes_votes > proposal.no_votes { + if votes_infos.yes_votes > votes_infos.no_votes { proposal.status = ProposalStatus::Approved; } else { proposal.status = ProposalStatus::Rejected; @@ -207,18 +242,20 @@ mod dao { let member = self.members.get(caller).ok_or(Error::MemberNotFound)?; - if member.last_vote >= proposal.vote_start { + if member.last_vote >= votes_infos.vote_start { return Err(Error::AlreadyVoted); } match approve { true => { - proposal.yes_votes = proposal.yes_votes.saturating_add(member.voting_power); + votes_infos.yes_votes = + votes_infos.yes_votes.saturating_add(member.voting_power); }, false => { - proposal.no_votes = proposal.no_votes.saturating_add(member.voting_power); + votes_infos.no_votes = votes_infos.no_votes.saturating_add(member.voting_power); }, }; + proposal.votes_infos = Some(votes_infos); self.proposals.insert(proposal_id, &proposal); @@ -240,9 +277,13 @@ mod dao { #[ink(message)] pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; + let votes_infos = proposal.votes_infos.clone().ok_or(Error::WrongContract)?; + + let transaction_infos = + proposal.transaction_infos.clone().ok_or(Error::WrongContract)?; // Check the voting period - if self.env().block_number() <= proposal.vote_end { + if self.env().block_number() <= votes_infos.vote_end { return Err(Error::VotingPeriodNotEnded); } @@ -250,28 +291,32 @@ mod dao { return Err(Error::ProposalExecuted); } - if proposal.yes_votes > proposal.no_votes { + if votes_infos.yes_votes > votes_infos.no_votes { let contract = self.env().account_id(); // Execute the proposal let _treasury_balance = match api::balance_of(self.token_id, contract) { - Ok(val) if val > proposal.amount => val, + Ok(val) if val > transaction_infos.amount => val, _ => { return Err(Error::NotEnoughFundsAvailable); }, }; - api::transfer(self.token_id, proposal.beneficiary, proposal.amount) - .map_err(Psp22Error::from)?; + api::transfer( + self.token_id, + transaction_infos.beneficiary, + transaction_infos.amount, + ) + .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { from: Some(contract), - to: Some(proposal.beneficiary), - value: proposal.amount, + to: Some(transaction_infos.beneficiary), + value: transaction_infos.amount, }); self.env().emit_event(Approval { owner: contract, spender: contract, - value: proposal.amount, + value: transaction_infos.amount, }); proposal.status = ProposalStatus::Executed; @@ -355,6 +400,9 @@ mod dao { /// There are not enough funds in the Dao treasury NotEnoughFundsAvailable, + /// The contract creation failed, a new contract is needed + WrongContract, + /// PSP22 specific error Psp22(Psp22Error), } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 6e0044be3..2b32e66f1 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -128,6 +128,20 @@ fn member_create_proposal_works(mut session: Session) { admin: account_id_from_slice(&contract), } ); + + // Another proposal created by Bob + let description_bis = "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); + session.set_actor(BOB); + assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis)); + assert_last_contract_event!( + &session, + Created { + id: 2, + creator: account_id_from_slice(&BOB), + admin: account_id_from_slice(&contract), + } + ); + } #[drink::test(sandbox = Pop)] @@ -154,8 +168,9 @@ fn members_vote_system_works(mut session: Session) { Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); let prop = proposal(&mut session, 1).unwrap(); - let yes = prop.yes_votes; - assert_eq!(yes > 0, true); + let infos = prop.votes_infos.unwrap(); + assert_eq!(infos.yes_votes > 0, true); + assert_eq!(infos.no_votes == 0, true); } #[drink::test(sandbox = Pop)] From 562b17feadfc865b07e00709a7e822da160a4cc7 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 26 Dec 2024 21:49:51 +0900 Subject: [PATCH 42/56] cargo fmt --- pop-api/examples/dao/src/lib.rs | 3 ++- pop-api/examples/dao/src/tests.rs | 4 ++-- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 88f22c3ab..0fe6b5513 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -199,7 +199,8 @@ mod dao { return Err(Error::ExceedeMaxDescriptionLength); } - let mut proposal = Proposal { proposal_id: self.proposals_created, ..Default::default() }; + let mut proposal = + Proposal { proposal_id: self.proposals_created, ..Default::default() }; proposal.description.append(&mut description); let transaction_infos = Transaction { beneficiary, amount }; proposal.transaction_infos = Some(transaction_infos); diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 2b32e66f1..2a627208e 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -130,7 +130,8 @@ fn member_create_proposal_works(mut session: Session) { ); // Another proposal created by Bob - let description_bis = "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); + let description_bis = + "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); session.set_actor(BOB); assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis)); assert_last_contract_event!( @@ -141,7 +142,6 @@ fn member_create_proposal_works(mut session: Session) { admin: account_id_from_slice(&contract), } ); - } #[drink::test(sandbox = Pop)] From eaae705e4a3b9df46efac5fd427f4bbb882efbdc Mon Sep 17 00:00:00 2001 From: ndkazu Date: Fri, 27 Dec 2024 23:26:06 +0900 Subject: [PATCH 43/56] Preparations for use of RuntimeCall --- pop-api/examples/dao/Cargo.toml | 4 ++++ pop-api/examples/dao/src/lib.rs | 37 ++++++++++++++++++++++++++++++++- runtime/devnet/src/lib.rs | 1 + 3 files changed, 41 insertions(+), 1 deletion(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 568b5a3b2..8f65bcfff 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -9,6 +9,8 @@ ink = { version = "=5.1.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } +sp-io = { version = "23.0.0", default-features = false, features = ["disable_allocator", "disable_oom", "disable_panic_handler"] } +sp-runtime = { version = "24.0.0", default-features = false } [dev-dependencies] drink = { package = "pop-drink", git = "https://github.com/r0gue-io/pop-drink" } @@ -31,4 +33,6 @@ ink-as-dependency = [] std = [ "ink/std", "pop-api/std", + "sp-io/std", + "sp-runtime/std", ] diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 0fe6b5513..c07794ba3 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,6 +1,6 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] -use ink::{prelude::vec::Vec, storage::Mapping}; +use ink::{env::Error as EnvError, prelude::vec::Vec, storage::Mapping}; use pop_api::{ primitives::TokenId, v0::fungibles::{ @@ -27,6 +27,19 @@ mod dao { Executed, } + #[ink::scale_derive(Encode)] + pub enum RuntimeCall { + /// We can add additional pallets we might want to use here + #[codec(index = 150)] + Fungibles(FungiblesCall), + } + + #[ink::scale_derive(Encode)] + pub enum FungiblesCall { + #[codec(index = 3)] + Transfer { token: TokenId, to: AccountId, value: Balance }, + } + /// Structure of the proposal used by the Dao governance sysytem #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] @@ -303,6 +316,15 @@ mod dao { }, }; + // first attempt at RuntimeCall use fails + // self.env() + // .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::Transfer { + // token: self.token_id, + // to: transaction_infos.beneficiary, + // value: transaction_infos.amount, + // })) + // .map_err(EnvError::from)?; + api::transfer( self.token_id, transaction_infos.beneficiary, @@ -404,6 +426,9 @@ mod dao { /// The contract creation failed, a new contract is needed WrongContract, + /// The Runtime Call failed + CallRuntimeFailed, + /// PSP22 specific error Psp22(Psp22Error), } @@ -413,4 +438,14 @@ mod dao { Error::Psp22(error) } } + + impl From for Error { + fn from(e: EnvError) -> Self { + use ink::env::ReturnErrorCode; + match e { + EnvError::ReturnError(ReturnErrorCode::CallRuntimeFailed) => e.into(), + _ => panic!("Unexpected error from `pallet-contracts`."), + } + } + } } diff --git a/runtime/devnet/src/lib.rs b/runtime/devnet/src/lib.rs index f539cbdee..d8784de55 100644 --- a/runtime/devnet/src/lib.rs +++ b/runtime/devnet/src/lib.rs @@ -41,6 +41,7 @@ use frame_system::{ }; use pallet_api::fungibles; use pallet_balances::Call as BalancesCall; +use pallet_fungibles::Call as FungiblesCall; use pallet_ismp::mmr::{Leaf, Proof, ProofKeys}; use pallet_xcm::{EnsureXcm, IsVoiceOfBody}; use parachains_common::message_queue::{NarrowOriginToSibling, ParaIdToSibling}; From adb72f9933552e1550e79132e4cdb1b070df3a32 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Fri, 27 Dec 2024 23:50:31 +0900 Subject: [PATCH 44/56] Use transfer_from instead of transfer for runtime_call --- pop-api/examples/dao/src/lib.rs | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index c07794ba3..62924ccb8 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -36,8 +36,8 @@ mod dao { #[ink::scale_derive(Encode)] pub enum FungiblesCall { - #[codec(index = 3)] - Transfer { token: TokenId, to: AccountId, value: Balance }, + #[codec(index = 4)] + TransferFrom { token: TokenId, from: AccountId, to: AccountId, value: Balance }, } /// Structure of the proposal used by the Dao governance sysytem @@ -316,10 +316,11 @@ mod dao { }, }; - // first attempt at RuntimeCall use fails + // First attempt at RuntimeCall transfer, you must comment api::transfer() below // self.env() - // .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::Transfer { + // .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { // token: self.token_id, + // from: contract, // to: transaction_infos.beneficiary, // value: transaction_infos.amount, // })) @@ -331,6 +332,7 @@ mod dao { transaction_infos.amount, ) .map_err(Psp22Error::from)?; + self.env().emit_event(Transfer { from: Some(contract), to: Some(transaction_infos.beneficiary), @@ -443,7 +445,8 @@ mod dao { fn from(e: EnvError) -> Self { use ink::env::ReturnErrorCode; match e { - EnvError::ReturnError(ReturnErrorCode::CallRuntimeFailed) => e.into(), + EnvError::ReturnError(ReturnErrorCode::CallRuntimeFailed) => + Error::CallRuntimeFailed, _ => panic!("Unexpected error from `pallet-contracts`."), } } From 290d0ec91c2c55c35dd276ad4fb062830136f269 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 29 Dec 2024 17:04:26 +0900 Subject: [PATCH 45/56] Corrected test mistake, using transfer_from instead of transfer --- pop-api/examples/dao/src/lib.rs | 3 ++- pop-api/examples/dao/src/tests.rs | 9 +++++++++ 2 files changed, 11 insertions(+), 1 deletion(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 62924ccb8..d4afe0da5 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -326,8 +326,9 @@ mod dao { // })) // .map_err(EnvError::from)?; - api::transfer( + api::transfer_from( self.token_id, + contract, transaction_infos.beneficiary, transaction_infos.amount, ) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 2a627208e..956656c04 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -238,6 +238,9 @@ fn proposal_enactment_works(mut session: Session) { let _ = env_logger::try_init(); // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); + // Mint tokens. + assert_ok!(session.sandbox().mint_into(&TOKEN, &contract.clone(), AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &contract.clone(), &contract.clone(), AMOUNT)); // Prepare voters accounts let _ = prepare_dao(&mut session, contract.clone()); @@ -289,6 +292,9 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { let _ = env_logger::try_init(); // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); + // Mint tokens. + assert_ok!(session.sandbox().mint_into(&TOKEN, &contract.clone(), AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &contract.clone(), &contract.clone(), AMOUNT)); // Prepare voters accounts let _ = prepare_dao(&mut session, contract.clone()); @@ -343,6 +349,9 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { let _ = env_logger::try_init(); // Deploy a new contract. let contract = deploy_with_default(&mut session).unwrap(); + // Mint tokens. + assert_ok!(session.sandbox().mint_into(&TOKEN, &contract.clone(), AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &contract.clone(), &contract.clone(), AMOUNT)); // Prepare voters accounts let _ = prepare_dao(&mut session, contract.clone()); From bad112689b98c6ec65be193ef35a648c49a1d4b6 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Sun, 29 Dec 2024 17:13:28 +0900 Subject: [PATCH 46/56] RuntimeCall working --- pop-api/examples/dao/src/lib.rs | 34 ++++++++++++++++----------------- 1 file changed, 17 insertions(+), 17 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index d4afe0da5..ec1a9c961 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -316,23 +316,23 @@ mod dao { }, }; - // First attempt at RuntimeCall transfer, you must comment api::transfer() below - // self.env() - // .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - // token: self.token_id, - // from: contract, - // to: transaction_infos.beneficiary, - // value: transaction_infos.amount, - // })) - // .map_err(EnvError::from)?; - - api::transfer_from( - self.token_id, - contract, - transaction_infos.beneficiary, - transaction_infos.amount, - ) - .map_err(Psp22Error::from)?; + // RuntimeCall transfer, you must comment api::transfer_from() below + let _ = self.env() + .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: self.token_id, + from: contract, + to: transaction_infos.beneficiary, + value: transaction_infos.amount, + })) + .map_err(EnvError::from); + + // api::transfer_from( + // self.token_id, + // contract, + // transaction_infos.beneficiary, + // transaction_infos.amount, + // ) + // .map_err(Psp22Error::from)?; self.env().emit_event(Transfer { from: Some(contract), From 32ecca50d653122042d0464dffef73365aeaebcf Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 31 Dec 2024 11:42:55 +0900 Subject: [PATCH 47/56] Corrections --- pop-api/examples/dao/src/lib.rs | 95 ++++++++++++++----------------- pop-api/examples/dao/src/tests.rs | 24 ++++---- 2 files changed, 56 insertions(+), 63 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index ec1a9c961..98c34f0a2 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -55,10 +55,10 @@ mod dao { pub proposal_id: u32, // Information relative to voting - pub votes_infos: Option, + pub round: Option, // Information relative to proposal execution if approved - pub transaction_infos: Option, + pub transaction: Option, } impl Default for Proposal { @@ -70,15 +70,15 @@ mod dao { let dao = fetch_dao.unwrap_or_default(); let voting_period = dao.voting_period; let current_block = ink::env::block_number::(); - let vote_end = current_block.saturating_add(voting_period); - let votes_infos = - Some(Votes { vote_start: current_block, vote_end, yes_votes: 0, no_votes: 0 }); + let end = current_block.saturating_add(voting_period); + let round = + Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); Proposal { description: Vec::new(), status: ProposalStatus::Submitted, proposal_id: 0, - votes_infos, - transaction_infos: None, + round, + transaction: None, } } } @@ -98,12 +98,12 @@ mod dao { #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] - pub struct Votes { + pub struct VoteRound { // Beginnning of the voting period for this proposal - pub vote_start: BlockNumber, + pub start: BlockNumber, // End of the voting period for this proposal - pub vote_end: BlockNumber, + pub end: BlockNumber, // Balance representing the total votes for this proposal pub yes_votes: Balance, @@ -144,14 +144,14 @@ mod dao { token_id: TokenId, // Proposals created in the history of the Dao - proposals_created: u32, + proposal_count: u32, } /// Defines an event that is emitted /// every time a member voted. #[derive(Debug)] #[ink(event)] - pub struct Voted { + pub struct Vote { pub who: Option, pub when: Option, } @@ -177,7 +177,7 @@ mod dao { last_votes: Mapping::default(), voting_period, token_id, - proposals_created: 0, + proposal_count: 0, }; let contract_id = instance.env().account_id(); api::create(token_id, contract_id, min_balance).map_err(Psp22Error::from)?; @@ -206,17 +206,18 @@ mod dao { ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - self.proposals_created = self.proposals_created.saturating_add(1); + if description.len() >= u8::MAX.into() { - return Err(Error::ExceedeMaxDescriptionLength); + return Err(Error::MaxDescriptionLengthReached); } + self.proposal_count = self.proposal_count.saturating_add(1); let mut proposal = - Proposal { proposal_id: self.proposals_created, ..Default::default() }; + Proposal { proposal_id: self.proposal_count, ..Default::default() }; proposal.description.append(&mut description); - let transaction_infos = Transaction { beneficiary, amount }; - proposal.transaction_infos = Some(transaction_infos); + let transaction = Transaction { beneficiary, amount }; + proposal.transaction = Some(transaction); self.proposals.insert(proposal.proposal_id, &proposal); @@ -240,11 +241,11 @@ mod dao { let caller = self.env().caller(); let current_block = self.env().block_number(); let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - let mut votes_infos = proposal.votes_infos.ok_or(Error::WrongContract)?; + let mut round = proposal.round.ok_or(Error::ProblemWithTheContract)?; - if current_block > votes_infos.vote_end { + if current_block > round.end { if proposal.status == ProposalStatus::Submitted { - if votes_infos.yes_votes > votes_infos.no_votes { + if round.yes_votes > round.no_votes { proposal.status = ProposalStatus::Approved; } else { proposal.status = ProposalStatus::Rejected; @@ -256,20 +257,20 @@ mod dao { let member = self.members.get(caller).ok_or(Error::MemberNotFound)?; - if member.last_vote >= votes_infos.vote_start { + if member.last_vote >= round.start { return Err(Error::AlreadyVoted); } match approve { true => { - votes_infos.yes_votes = - votes_infos.yes_votes.saturating_add(member.voting_power); + round.yes_votes = + round.yes_votes.saturating_add(member.voting_power); }, false => { - votes_infos.no_votes = votes_infos.no_votes.saturating_add(member.voting_power); + round.no_votes = round.no_votes.saturating_add(member.voting_power); }, }; - proposal.votes_infos = Some(votes_infos); + proposal.round = Some(round); self.proposals.insert(proposal_id, &proposal); @@ -279,7 +280,7 @@ mod dao { ); self.last_votes.insert(caller, ¤t_block); - self.env().emit_event(Voted { who: Some(caller), when: Some(current_block) }); + self.env().emit_event(Vote { who: Some(caller), when: Some(current_block) }); Ok(()) } @@ -291,13 +292,13 @@ mod dao { #[ink(message)] pub fn execute_proposal(&mut self, proposal_id: u32) -> Result<(), Error> { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - let votes_infos = proposal.votes_infos.clone().ok_or(Error::WrongContract)?; + let round = proposal.round.clone().ok_or(Error::ProblemWithTheContract)?; - let transaction_infos = - proposal.transaction_infos.clone().ok_or(Error::WrongContract)?; + let transaction = + proposal.transaction.clone().ok_or(Error::ProblemWithTheContract)?; // Check the voting period - if self.env().block_number() <= votes_infos.vote_end { + if self.env().block_number() <= round.end { return Err(Error::VotingPeriodNotEnded); } @@ -305,44 +306,36 @@ mod dao { return Err(Error::ProposalExecuted); } - if votes_infos.yes_votes > votes_infos.no_votes { + if round.yes_votes > round.no_votes { let contract = self.env().account_id(); // Execute the proposal let _treasury_balance = match api::balance_of(self.token_id, contract) { - Ok(val) if val > transaction_infos.amount => val, + Ok(val) if val > transaction.amount => val, _ => { return Err(Error::NotEnoughFundsAvailable); }, }; - // RuntimeCall transfer, you must comment api::transfer_from() below + // RuntimeCall. let _ = self.env() .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: self.token_id, from: contract, - to: transaction_infos.beneficiary, - value: transaction_infos.amount, + to: transaction.beneficiary, + value: transaction.amount, })) .map_err(EnvError::from); - // api::transfer_from( - // self.token_id, - // contract, - // transaction_infos.beneficiary, - // transaction_infos.amount, - // ) - // .map_err(Psp22Error::from)?; - self.env().emit_event(Transfer { from: Some(contract), - to: Some(transaction_infos.beneficiary), - value: transaction_infos.amount, + to: Some(transaction.beneficiary), + value: transaction.amount, }); self.env().emit_event(Approval { owner: contract, spender: contract, - value: transaction_infos.amount, + value: transaction.amount, }); proposal.status = ProposalStatus::Executed; @@ -421,16 +414,16 @@ mod dao { ProposalRejected, /// The proposal description is too long - ExceedeMaxDescriptionLength, + MaxDescriptionLengthReached, /// There are not enough funds in the Dao treasury NotEnoughFundsAvailable, /// The contract creation failed, a new contract is needed - WrongContract, + ProblemWithTheContract, /// The Runtime Call failed - CallRuntimeFailed, + ProposalExecutionFailed, /// PSP22 specific error Psp22(Psp22Error), @@ -447,7 +440,7 @@ mod dao { use ink::env::ReturnErrorCode; match e { EnvError::ReturnError(ReturnErrorCode::CallRuntimeFailed) => - Error::CallRuntimeFailed, + Error::ProposalExecutionFailed, _ => panic!("Unexpected error from `pallet-contracts`."), } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 956656c04..babbc5d79 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -11,7 +11,7 @@ use pop_api::{ }; use super::*; -use crate::dao::{Error, Member, Proposal, Voted}; +use crate::dao::{Error, Member, Proposal, Vote}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -165,10 +165,10 @@ fn members_vote_system_works(mut session: Session) { assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); let prop = proposal(&mut session, 1).unwrap(); - let infos = prop.votes_infos.unwrap(); + let infos = prop.round.unwrap(); assert_eq!(infos.yes_votes > 0, true); assert_eq!(infos.no_votes == 0, true); } @@ -256,21 +256,21 @@ fn proposal_enactment_works(mut session: Session) { assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); // Alice vote session.set_actor(ALICE); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } ); // BOB vote session.set_actor(BOB); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&BOB)), when: Some(now) } ); session.sandbox().build_blocks(VOTING_PERIOD + 1); @@ -310,21 +310,21 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); // Alice vote session.set_actor(ALICE); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } ); // BOB vote session.set_actor(BOB); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&BOB)), when: Some(now) } ); session.sandbox().build_blocks(VOTING_PERIOD + 1); @@ -367,21 +367,21 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { assert_ok!(vote(&mut session, 1, false)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); // Alice vote session.set_actor(ALICE); assert_ok!(vote(&mut session, 1, false)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } ); // BOB vote session.set_actor(BOB); assert_ok!(vote(&mut session, 1, false)); assert_last_contract_event!( &session, - Voted { who: Some(account_id_from_slice(&BOB)), when: Some(now) } + Vote { who: Some(account_id_from_slice(&BOB)), when: Some(now) } ); session.sandbox().build_blocks(VOTING_PERIOD + 1); From c357a0755706a9bbe01adc3522854370f366cf74 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Tue, 31 Dec 2024 12:12:07 +0900 Subject: [PATCH 48/56] Code re-factoring --- pop-api/examples/dao/src/lib.rs | 50 ++++++++++++++++++++------------- 1 file changed, 31 insertions(+), 19 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 98c34f0a2..88d2d7844 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -84,7 +84,7 @@ mod dao { } /// Representation of a member in the voting system - #[derive(Debug)] + #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Member { @@ -112,6 +112,31 @@ mod dao { pub no_votes: Balance, } + impl VoteRound { + fn get_status(&self, mut proposal: Proposal) -> Proposal { + if proposal.status == ProposalStatus::Submitted { + if self.yes_votes > self.no_votes { + proposal.status = ProposalStatus::Approved; + } else { + proposal.status = ProposalStatus::Rejected; + } + }; + proposal + } + + fn update_votes(&mut self, approved: bool, member: Member) { + match approved { + true => { + self.yes_votes = + self.yes_votes.saturating_add(member.voting_power); + }, + false => { + self.no_votes = self.no_votes.saturating_add(member.voting_power); + }, + }; + } + } + #[derive(Debug, Clone)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] @@ -241,17 +266,12 @@ mod dao { let caller = self.env().caller(); let current_block = self.env().block_number(); let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; - let mut round = proposal.round.ok_or(Error::ProblemWithTheContract)?; + let mut round = proposal.round.clone().ok_or(Error::ProblemWithTheContract)?; if current_block > round.end { - if proposal.status == ProposalStatus::Submitted { - if round.yes_votes > round.no_votes { - proposal.status = ProposalStatus::Approved; - } else { - proposal.status = ProposalStatus::Rejected; - } - }; - + // Update the Proposal status if needed + proposal = round.get_status(proposal); + self.proposals.insert(proposal.proposal_id, &proposal); return Err(Error::VotingPeriodEnded); } @@ -261,15 +281,7 @@ mod dao { return Err(Error::AlreadyVoted); } - match approve { - true => { - round.yes_votes = - round.yes_votes.saturating_add(member.voting_power); - }, - false => { - round.no_votes = round.no_votes.saturating_add(member.voting_power); - }, - }; + round.update_votes(approve, member.clone()); proposal.round = Some(round); self.proposals.insert(proposal_id, &proposal); From c0f9f9d90370c566254a0f1f1362129410584acd Mon Sep 17 00:00:00 2001 From: ndkazu Date: Wed, 8 Jan 2025 21:13:09 +0900 Subject: [PATCH 49/56] RuntimeCall conversion problem --- pop-api/examples/dao/Cargo.toml | 1 + pop-api/examples/dao/src/lib.rs | 28 ++++++++--- pop-api/examples/dao/src/tests.rs | 79 +++++++++++++++++++++++++++---- 3 files changed, 92 insertions(+), 16 deletions(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 8f65bcfff..9b5f790a5 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -9,6 +9,7 @@ ink = { version = "=5.1.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } +serde = { version = "1.0.114", default-features = false } sp-io = { version = "23.0.0", default-features = false, features = ["disable_allocator", "disable_oom", "disable_panic_handler"] } sp-runtime = { version = "24.0.0", default-features = false } diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 88d2d7844..f880bac76 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -9,6 +9,7 @@ use pop_api::{ Psp22Error, }, }; +use sp_runtime::serde::Serialize; #[cfg(test)] mod tests; @@ -27,14 +28,18 @@ mod dao { Executed, } - #[ink::scale_derive(Encode)] + #[derive(Debug, Clone, PartialEq, Serialize)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub enum RuntimeCall { /// We can add additional pallets we might want to use here #[codec(index = 150)] Fungibles(FungiblesCall), } - #[ink::scale_derive(Encode)] + #[derive(Debug, Clone, PartialEq, Serialize)] + #[ink::scale_derive(Encode, Decode, TypeInfo)] + #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub enum FungiblesCall { #[codec(index = 4)] TransferFrom { token: TokenId, from: AccountId, to: AccountId, value: Balance }, @@ -59,6 +64,9 @@ mod dao { // Information relative to proposal execution if approved pub transaction: Option, + + // Call top be executed + pub call: Option } impl Default for Proposal { @@ -71,6 +79,7 @@ mod dao { let voting_period = dao.voting_period; let current_block = ink::env::block_number::(); let end = current_block.saturating_add(voting_period); + let call = None; let round = Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); Proposal { @@ -79,6 +88,7 @@ mod dao { proposal_id: 0, round, transaction: None, + call, } } } @@ -228,10 +238,11 @@ mod dao { beneficiary: AccountId, amount: Balance, mut description: Vec, + call: RuntimeCall, ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - + let wrapped_call = Some(call); if description.len() >= u8::MAX.into() { return Err(Error::MaxDescriptionLengthReached); @@ -239,7 +250,7 @@ mod dao { self.proposal_count = self.proposal_count.saturating_add(1); let mut proposal = - Proposal { proposal_id: self.proposal_count, ..Default::default() }; + Proposal { proposal_id: self.proposal_count, call: wrapped_call, ..Default::default() }; proposal.description.append(&mut description); let transaction = Transaction { beneficiary, amount }; proposal.transaction = Some(transaction); @@ -330,14 +341,16 @@ mod dao { }; // RuntimeCall. - let _ = self.env() + let call = proposal.call.clone().expect("There should be a call"); + let _ = self.env().call_runtime(&call).map_err(EnvError::from); + /*let _ = self.env() .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: self.token_id, from: contract, to: transaction.beneficiary, value: transaction.amount, })) - .map_err(EnvError::from); + .map_err(EnvError::from);*/ self.env().emit_event(Transfer { from: Some(contract), @@ -437,6 +450,9 @@ mod dao { /// The Runtime Call failed ProposalExecutionFailed, + /// The Runtime Call is missing or invalid + NoValidRuntimeCall, + /// PSP22 specific error Psp22(Psp22Error), } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index babbc5d79..6ab096b0b 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -5,13 +5,14 @@ use drink::{ session::Session, AssetsAPI, TestExternalities, NO_SALT, }; +use ink::scale::Encode; use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, }; use super::*; -use crate::dao::{Error, Member, Proposal, Vote}; +use crate::dao::{Error, Member, Proposal, Vote, FungiblesCall, RuntimeCall}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -118,7 +119,13 @@ fn member_create_proposal_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); assert_last_contract_event!( &session, @@ -133,7 +140,13 @@ fn member_create_proposal_works(mut session: Session) { let description_bis = "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); session.set_actor(BOB); - assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&ALICE), + value: amount, + }); + assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis, call0)); assert_last_contract_event!( &session, Created { @@ -156,7 +169,14 @@ fn members_vote_system_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); session.set_actor(CHARLIE); // Charlie vote @@ -185,7 +205,13 @@ fn double_vote_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); session.set_actor(CHARLIE); // Charlie tries to vote twice for the same proposal @@ -205,7 +231,13 @@ fn vote_fails_if_voting_period_ended(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); // Moving to blocks beyond voting period session.sandbox().build_blocks(VOTING_PERIOD + 1); @@ -227,7 +259,13 @@ fn vote_fails_if_not_a_member(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); session.set_actor(NON_MEMBER); assert_eq!(vote(&mut session, 1, true), Err(Error::MemberNotFound)); @@ -248,7 +286,13 @@ fn proposal_enactment_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -302,7 +346,13 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -359,7 +409,13 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - assert_ok!(create_proposal(&mut session, BOB, amount, description)); + let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: TOKEN, + from: account_id_from_slice(&contract), + to: account_id_from_slice(&BOB), + value: amount, + }); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -419,8 +475,10 @@ fn create_proposal( beneficiary: AccountId, amount: Balance, description: Vec, + call: RuntimeCall, ) -> Result<(), Error> { let desc: &[u8] = &description; + let call0: Vec = serde_json::to_vec(&call).unwrap(); call::( session, "create_proposal", @@ -428,6 +486,7 @@ fn create_proposal( beneficiary.to_string(), amount.to_string(), serde_json::to_string::<[u8]>(desc).unwrap(), + serde_json::to_string::<[u8]>(&call0).unwrap(), ], None, ) From 2526c0438b4afc411add3595446833ad674cc74f Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 9 Jan 2025 11:50:30 +0900 Subject: [PATCH 50/56] customised RuntimeCalls works --- pop-api/examples/dao/src/lib.rs | 46 +++++++++++----------- pop-api/examples/dao/src/tests.rs | 65 +++++++++++++++++++------------ 2 files changed, 63 insertions(+), 48 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index f880bac76..0bffc8d44 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,6 +1,10 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] -use ink::{env::Error as EnvError, prelude::vec::Vec, storage::Mapping}; +use ink::{ + env::Error as EnvError, + prelude::vec::Vec, + storage::{traits::Storable, Mapping}, +}; use pop_api::{ primitives::TokenId, v0::fungibles::{ @@ -9,7 +13,6 @@ use pop_api::{ Psp22Error, }, }; -use sp_runtime::serde::Serialize; #[cfg(test)] mod tests; @@ -28,7 +31,7 @@ mod dao { Executed, } - #[derive(Debug, Clone, PartialEq, Serialize)] + #[derive(Debug, Clone, PartialEq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub enum RuntimeCall { @@ -37,7 +40,7 @@ mod dao { Fungibles(FungiblesCall), } - #[derive(Debug, Clone, PartialEq, Serialize)] + #[derive(Debug, Clone, PartialEq)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub enum FungiblesCall { @@ -66,7 +69,7 @@ mod dao { pub transaction: Option, // Call top be executed - pub call: Option + pub call: Option>, } impl Default for Proposal { @@ -80,8 +83,7 @@ mod dao { let current_block = ink::env::block_number::(); let end = current_block.saturating_add(voting_period); let call = None; - let round = - Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); + let round = Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); Proposal { description: Vec::new(), status: ProposalStatus::Submitted, @@ -137,8 +139,7 @@ mod dao { fn update_votes(&mut self, approved: bool, member: Member) { match approved { true => { - self.yes_votes = - self.yes_votes.saturating_add(member.voting_power); + self.yes_votes = self.yes_votes.saturating_add(member.voting_power); }, false => { self.no_votes = self.no_votes.saturating_add(member.voting_power); @@ -232,13 +233,14 @@ mod dao { /// if the proposal is accepted. /// - `amount` - Amount requested for this proposal /// - `description` - Description of the proposal + /// - `call` - Proposal call to be executed #[ink(message)] pub fn create_proposal( &mut self, beneficiary: AccountId, amount: Balance, mut description: Vec, - call: RuntimeCall, + call: Vec, ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); @@ -249,8 +251,11 @@ mod dao { } self.proposal_count = self.proposal_count.saturating_add(1); - let mut proposal = - Proposal { proposal_id: self.proposal_count, call: wrapped_call, ..Default::default() }; + let mut proposal = Proposal { + proposal_id: self.proposal_count, + call: wrapped_call, + ..Default::default() + }; proposal.description.append(&mut description); let transaction = Transaction { beneficiary, amount }; proposal.transaction = Some(transaction); @@ -317,8 +322,7 @@ mod dao { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; let round = proposal.round.clone().ok_or(Error::ProblemWithTheContract)?; - let transaction = - proposal.transaction.clone().ok_or(Error::ProblemWithTheContract)?; + let transaction = proposal.transaction.clone().ok_or(Error::ProblemWithTheContract)?; // Check the voting period if self.env().block_number() <= round.end { @@ -341,16 +345,10 @@ mod dao { }; // RuntimeCall. - let call = proposal.call.clone().expect("There should be a call"); - let _ = self.env().call_runtime(&call).map_err(EnvError::from); - /*let _ = self.env() - .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: self.token_id, - from: contract, - to: transaction.beneficiary, - value: transaction.amount, - })) - .map_err(EnvError::from);*/ + let call = &proposal.call.clone().expect("There should be a call"); + if let Ok(call0) = RuntimeCall::decode(&mut &call[..]) { + let _ = self.env().call_runtime(&call0).map_err(EnvError::from); + } self.env().emit_event(Transfer { from: Some(contract), diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 6ab096b0b..3aaa880c7 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -5,14 +5,14 @@ use drink::{ session::Session, AssetsAPI, TestExternalities, NO_SALT, }; -use ink::scale::Encode; +use ink::scale::Decode; use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, }; use super::*; -use crate::dao::{Error, Member, Proposal, Vote, FungiblesCall, RuntimeCall}; +use crate::dao::{Error, FungiblesCall, Member, Proposal, RuntimeCall, Vote}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -119,13 +119,15 @@ fn member_create_proposal_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); assert_last_contract_event!( &session, @@ -140,13 +142,15 @@ fn member_create_proposal_works(mut session: Session) { let description_bis = "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); session.set_actor(BOB); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&ALICE), value: amount, - }); - assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis, call_f)); assert_last_contract_event!( &session, Created { @@ -169,14 +173,15 @@ fn members_vote_system_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); session.set_actor(CHARLIE); // Charlie vote @@ -205,13 +210,15 @@ fn double_vote_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); session.set_actor(CHARLIE); // Charlie tries to vote twice for the same proposal @@ -231,13 +238,15 @@ fn vote_fails_if_voting_period_ended(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); // Moving to blocks beyond voting period session.sandbox().build_blocks(VOTING_PERIOD + 1); @@ -259,13 +268,15 @@ fn vote_fails_if_not_a_member(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); session.set_actor(NON_MEMBER); assert_eq!(vote(&mut session, 1, true), Err(Error::MemberNotFound)); @@ -286,13 +297,15 @@ fn proposal_enactment_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -346,13 +359,15 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -409,13 +424,15 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); + let mut call_f = Vec::new(); let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { token: TOKEN, from: account_id_from_slice(&contract), to: account_id_from_slice(&BOB), value: amount, - }); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call0)); + }) + .encode(&mut call_f); + assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -475,10 +492,10 @@ fn create_proposal( beneficiary: AccountId, amount: Balance, description: Vec, - call: RuntimeCall, + runtimecall: Vec, ) -> Result<(), Error> { let desc: &[u8] = &description; - let call0: Vec = serde_json::to_vec(&call).unwrap(); + let call0: &[u8] = &runtimecall; call::( session, "create_proposal", @@ -486,7 +503,7 @@ fn create_proposal( beneficiary.to_string(), amount.to_string(), serde_json::to_string::<[u8]>(desc).unwrap(), - serde_json::to_string::<[u8]>(&call0).unwrap(), + serde_json::to_string::<[u8]>(call0).unwrap(), ], None, ) From a7e3084277bcb5ecd8fc77768c8832da271b3dcc Mon Sep 17 00:00:00 2001 From: Kazunobu Ndong <33208377+ndkazu@users.noreply.github.com> Date: Thu, 9 Jan 2025 18:04:19 +0900 Subject: [PATCH 51/56] Option_1 Option 1 could for example turn into one of the following examples: - Community Fund Management: Manage a treasury where members can propose and vote on funding community projects. - Grant Distribution: Allocate funds to developers or projects based on proposals and voting. --- pop-api/examples/dao/src/lib.rs | 54 ++++++++++++--------------------- 1 file changed, 20 insertions(+), 34 deletions(-) diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 0bffc8d44..88d2d7844 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -1,10 +1,6 @@ #![cfg_attr(not(feature = "std"), no_std, no_main)] -use ink::{ - env::Error as EnvError, - prelude::vec::Vec, - storage::{traits::Storable, Mapping}, -}; +use ink::{env::Error as EnvError, prelude::vec::Vec, storage::Mapping}; use pop_api::{ primitives::TokenId, v0::fungibles::{ @@ -31,18 +27,14 @@ mod dao { Executed, } - #[derive(Debug, Clone, PartialEq)] - #[ink::scale_derive(Encode, Decode, TypeInfo)] - #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + #[ink::scale_derive(Encode)] pub enum RuntimeCall { /// We can add additional pallets we might want to use here #[codec(index = 150)] Fungibles(FungiblesCall), } - #[derive(Debug, Clone, PartialEq)] - #[ink::scale_derive(Encode, Decode, TypeInfo)] - #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] + #[ink::scale_derive(Encode)] pub enum FungiblesCall { #[codec(index = 4)] TransferFrom { token: TokenId, from: AccountId, to: AccountId, value: Balance }, @@ -67,9 +59,6 @@ mod dao { // Information relative to proposal execution if approved pub transaction: Option, - - // Call top be executed - pub call: Option>, } impl Default for Proposal { @@ -82,15 +71,14 @@ mod dao { let voting_period = dao.voting_period; let current_block = ink::env::block_number::(); let end = current_block.saturating_add(voting_period); - let call = None; - let round = Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); + let round = + Some(VoteRound { start: current_block, end, yes_votes: 0, no_votes: 0 }); Proposal { description: Vec::new(), status: ProposalStatus::Submitted, proposal_id: 0, round, transaction: None, - call, } } } @@ -139,7 +127,8 @@ mod dao { fn update_votes(&mut self, approved: bool, member: Member) { match approved { true => { - self.yes_votes = self.yes_votes.saturating_add(member.voting_power); + self.yes_votes = + self.yes_votes.saturating_add(member.voting_power); }, false => { self.no_votes = self.no_votes.saturating_add(member.voting_power); @@ -233,29 +222,24 @@ mod dao { /// if the proposal is accepted. /// - `amount` - Amount requested for this proposal /// - `description` - Description of the proposal - /// - `call` - Proposal call to be executed #[ink(message)] pub fn create_proposal( &mut self, beneficiary: AccountId, amount: Balance, mut description: Vec, - call: Vec, ) -> Result<(), Error> { let caller = self.env().caller(); let contract = self.env().account_id(); - let wrapped_call = Some(call); + if description.len() >= u8::MAX.into() { return Err(Error::MaxDescriptionLengthReached); } self.proposal_count = self.proposal_count.saturating_add(1); - let mut proposal = Proposal { - proposal_id: self.proposal_count, - call: wrapped_call, - ..Default::default() - }; + let mut proposal = + Proposal { proposal_id: self.proposal_count, ..Default::default() }; proposal.description.append(&mut description); let transaction = Transaction { beneficiary, amount }; proposal.transaction = Some(transaction); @@ -322,7 +306,8 @@ mod dao { let mut proposal = self.proposals.get(proposal_id).ok_or(Error::ProposalNotFound)?; let round = proposal.round.clone().ok_or(Error::ProblemWithTheContract)?; - let transaction = proposal.transaction.clone().ok_or(Error::ProblemWithTheContract)?; + let transaction = + proposal.transaction.clone().ok_or(Error::ProblemWithTheContract)?; // Check the voting period if self.env().block_number() <= round.end { @@ -345,10 +330,14 @@ mod dao { }; // RuntimeCall. - let call = &proposal.call.clone().expect("There should be a call"); - if let Ok(call0) = RuntimeCall::decode(&mut &call[..]) { - let _ = self.env().call_runtime(&call0).map_err(EnvError::from); - } + let _ = self.env() + .call_runtime(&RuntimeCall::Fungibles(FungiblesCall::TransferFrom { + token: self.token_id, + from: contract, + to: transaction.beneficiary, + value: transaction.amount, + })) + .map_err(EnvError::from); self.env().emit_event(Transfer { from: Some(contract), @@ -448,9 +437,6 @@ mod dao { /// The Runtime Call failed ProposalExecutionFailed, - /// The Runtime Call is missing or invalid - NoValidRuntimeCall, - /// PSP22 specific error Psp22(Psp22Error), } From 2f717a291b51e6d10eae698b9c9b45110f9f73f7 Mon Sep 17 00:00:00 2001 From: Kazunobu Ndong <33208377+ndkazu@users.noreply.github.com> Date: Thu, 9 Jan 2025 18:05:26 +0900 Subject: [PATCH 52/56] Update tests.rs --- pop-api/examples/dao/src/tests.rs | 96 ++++--------------------------- 1 file changed, 10 insertions(+), 86 deletions(-) diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index 3aaa880c7..babbc5d79 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -5,14 +5,13 @@ use drink::{ session::Session, AssetsAPI, TestExternalities, NO_SALT, }; -use ink::scale::Decode; use pop_api::{ primitives::TokenId, v0::fungibles::events::{Approval, Created, Transfer}, }; use super::*; -use crate::dao::{Error, FungiblesCall, Member, Proposal, RuntimeCall, Vote}; +use crate::dao::{Error, Member, Proposal, Vote}; const UNIT: Balance = 10_000_000_000; const INIT_AMOUNT: Balance = 100_000_000 * UNIT; @@ -119,15 +118,7 @@ fn member_create_proposal_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); assert_last_contract_event!( &session, @@ -142,15 +133,7 @@ fn member_create_proposal_works(mut session: Session) { let description_bis = "Funds for creation of another Dao contract".to_string().as_bytes().to_vec(); session.set_actor(BOB); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&ALICE), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis, call_f)); + assert_ok!(create_proposal(&mut session, ALICE, amount, description_bis)); assert_last_contract_event!( &session, Created { @@ -173,15 +156,7 @@ fn members_vote_system_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); session.set_actor(CHARLIE); // Charlie vote @@ -210,15 +185,7 @@ fn double_vote_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); session.set_actor(CHARLIE); // Charlie tries to vote twice for the same proposal @@ -238,15 +205,7 @@ fn vote_fails_if_voting_period_ended(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); // Moving to blocks beyond voting period session.sandbox().build_blocks(VOTING_PERIOD + 1); @@ -268,15 +227,7 @@ fn vote_fails_if_not_a_member(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = AMOUNT * 3; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); session.set_actor(NON_MEMBER); assert_eq!(vote(&mut session, 1, true), Err(Error::MemberNotFound)); @@ -297,15 +248,7 @@ fn proposal_enactment_works(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -359,15 +302,7 @@ fn same_proposal_consecutive_claim_fails(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -424,15 +359,7 @@ fn proposal_enactment_fails_if_proposal_is_rejected(mut session: Session) { let description = "Funds for creation of a Dao contract".to_string().as_bytes().to_vec(); let amount = MIN_BALANCE; session.set_actor(ALICE); - let mut call_f = Vec::new(); - let call0 = RuntimeCall::Fungibles(FungiblesCall::TransferFrom { - token: TOKEN, - from: account_id_from_slice(&contract), - to: account_id_from_slice(&BOB), - value: amount, - }) - .encode(&mut call_f); - assert_ok!(create_proposal(&mut session, BOB, amount, description, call_f)); + assert_ok!(create_proposal(&mut session, BOB, amount, description)); let now = session.sandbox().block_number(); session.set_actor(CHARLIE); @@ -492,10 +419,8 @@ fn create_proposal( beneficiary: AccountId, amount: Balance, description: Vec, - runtimecall: Vec, ) -> Result<(), Error> { let desc: &[u8] = &description; - let call0: &[u8] = &runtimecall; call::( session, "create_proposal", @@ -503,7 +428,6 @@ fn create_proposal( beneficiary.to_string(), amount.to_string(), serde_json::to_string::<[u8]>(desc).unwrap(), - serde_json::to_string::<[u8]>(call0).unwrap(), ], None, ) From 08a61a005e55fb93ae0e18eaa570acc82d4d5862 Mon Sep 17 00:00:00 2001 From: Kazunobu Ndong <33208377+ndkazu@users.noreply.github.com> Date: Thu, 9 Jan 2025 18:06:12 +0900 Subject: [PATCH 53/56] Update Cargo.toml --- pop-api/examples/dao/Cargo.toml | 1 - 1 file changed, 1 deletion(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 9b5f790a5..8f65bcfff 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -9,7 +9,6 @@ ink = { version = "=5.1.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } -serde = { version = "1.0.114", default-features = false } sp-io = { version = "23.0.0", default-features = false, features = ["disable_allocator", "disable_oom", "disable_panic_handler"] } sp-runtime = { version = "24.0.0", default-features = false } From 95ccd0f74eeb832c00f3191d630a232f723bc49c Mon Sep 17 00:00:00 2001 From: Kazunobu Ndong <33208377+ndkazu@users.noreply.github.com> Date: Thu, 9 Jan 2025 18:16:19 +0900 Subject: [PATCH 54/56] Update README.md --- pop-api/examples/dao/README.md | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/pop-api/examples/dao/README.md b/pop-api/examples/dao/README.md index 609d2ca90..ed3c6f222 100644 --- a/pop-api/examples/dao/README.md +++ b/pop-api/examples/dao/README.md @@ -2,6 +2,9 @@ ## Description This contract implements a Decentralized Autonomous Organization using Psp22. +The objectivbe of this DAO is to allow its members to vote for funding of projects they would like to see developped. +To become a member, a user needs to pay some amount of the Dao_token to the Dao Treasury(Dao_tokens obtained through airdrops or token purchase...) ⇒ The subscription payment becomes the Member voting_power. +Approved projects are funded by the Dao Treasury. The key functionalities include: - **Membership Management**: It maintains a registry of DAO members. - **Proposal Lifecycle**: The contract manages the creation, voting, and execution of proposals. Each proposal includes details like description, voting period, vote tallies, execution status, beneficiary, and amount to be awarded. @@ -50,4 +53,4 @@ flowchart LR style J fill:#f99,stroke:#333,stroke-width:2px style K fill:#bfb,stroke:#333,stroke-width:2px style L fill:#bfb,stroke:#333,stroke-width:2px -``` \ No newline at end of file +``` From 2e194e61add0b415b004646ba0f4eae2a9e17020 Mon Sep 17 00:00:00 2001 From: Kazunobu Ndong <33208377+ndkazu@users.noreply.github.com> Date: Thu, 9 Jan 2025 18:16:44 +0900 Subject: [PATCH 55/56] Update README.md --- pop-api/examples/dao/README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pop-api/examples/dao/README.md b/pop-api/examples/dao/README.md index ed3c6f222..79f3ae41e 100644 --- a/pop-api/examples/dao/README.md +++ b/pop-api/examples/dao/README.md @@ -1,4 +1,4 @@ -# PSP22 DAO contract example +# PSP22 DAO contract example_01 ## Description This contract implements a Decentralized Autonomous Organization using Psp22. From d4ba764de1c057fc97fad65ab735ce2eb241e221 Mon Sep 17 00:00:00 2001 From: ndkazu Date: Thu, 23 Jan 2025 17:11:39 +0900 Subject: [PATCH 56/56] Applied corrections --- pop-api/examples/dao/Cargo.toml | 12 +----------- pop-api/examples/dao/README.md | 6 +++--- pop-api/examples/dao/src/lib.rs | 8 ++++---- pop-api/examples/dao/src/tests.rs | 27 ++++++++++++--------------- 4 files changed, 20 insertions(+), 33 deletions(-) diff --git a/pop-api/examples/dao/Cargo.toml b/pop-api/examples/dao/Cargo.toml index 8f65bcfff..5e04b7312 100644 --- a/pop-api/examples/dao/Cargo.toml +++ b/pop-api/examples/dao/Cargo.toml @@ -5,24 +5,16 @@ name = "dao" version = "0.1.0" [dependencies] -ink = { version = "=5.1.0", default-features = false, features = ["ink-debug"] } +ink = { version = "5.1.0", default-features = false, features = ["ink-debug"] } pop-api = { path = "../../../pop-api", default-features = false, features = [ "fungibles", ] } -sp-io = { version = "23.0.0", default-features = false, features = ["disable_allocator", "disable_oom", "disable_panic_handler"] } -sp-runtime = { version = "24.0.0", default-features = false } [dev-dependencies] drink = { package = "pop-drink", git = "https://github.com/r0gue-io/pop-drink" } env_logger = { version = "0.11.3" } serde_json = "1.0.114" -# TODO: due to compilation issues caused by `sp-runtime`, `frame-support-procedural` and `staging-xcm` this dependency -# (with specific version) has to be added. Will be tackled by #348, please ignore for now. -frame-support-procedural = { version = "=30.0.1", default-features = false } -sp-runtime = { version = "=38.0.0", default-features = false } -staging-xcm = { version = "=14.1.0", default-features = false } - [lib] path = "src/lib.rs" @@ -33,6 +25,4 @@ ink-as-dependency = [] std = [ "ink/std", "pop-api/std", - "sp-io/std", - "sp-runtime/std", ] diff --git a/pop-api/examples/dao/README.md b/pop-api/examples/dao/README.md index 79f3ae41e..e77fa71e0 100644 --- a/pop-api/examples/dao/README.md +++ b/pop-api/examples/dao/README.md @@ -1,5 +1,5 @@ -# PSP22 DAO contract example_01 - +# Simple PSP22 DAO contract example +This contract implements a Decentralized Autonomous Organization with token-based voting system built with Pop API. Members can submit a funding proposal, vote for or against it, and enact funds transfer. ## Description This contract implements a Decentralized Autonomous Organization using Psp22. The objectivbe of this DAO is to allow its members to vote for funding of projects they would like to see developped. @@ -9,7 +9,7 @@ The key functionalities include: - **Membership Management**: It maintains a registry of DAO members. - **Proposal Lifecycle**: The contract manages the creation, voting, and execution of proposals. Each proposal includes details like description, voting period, vote tallies, execution status, beneficiary, and amount to be awarded. - **Voting Mechanism**: It implements a voting system where members can vote with their balance on proposals. The contract tracks voting periods and maintains vote counts for each proposal. -- **Token Integration**: The DAO is associated with a specific Psp22 token_id. +- **Token Integration**: The Pop API is used to create and manage the DAO token. - **Governance Parameters**: governance parameters such as voting periods are customizable. - **Vote Tracking**: The contract keeps track of when members last voted. - **Proposal Execution**: Once a proposal's voting period ends and passes, the contract handles its execution: transferring funds to the chosen beneficiary. diff --git a/pop-api/examples/dao/src/lib.rs b/pop-api/examples/dao/src/lib.rs index 88d2d7844..e466ad220 100644 --- a/pop-api/examples/dao/src/lib.rs +++ b/pop-api/examples/dao/src/lib.rs @@ -84,7 +84,7 @@ mod dao { } /// Representation of a member in the voting system - #[derive(Debug, Clone)] + #[derive(Debug, Clone, Default)] #[ink::scale_derive(Encode, Decode, TypeInfo)] #[cfg_attr(feature = "std", derive(ink::storage::traits::StorageLayout))] pub struct Member { @@ -375,7 +375,7 @@ mod dao { api::transfer_from(self.token_id, caller, contract, amount) .map_err(Psp22Error::from)?; let member = - self.members.get(caller).unwrap_or(Member { voting_power: 0, last_vote: 0 }); + self.members.get(caller).unwrap_or_default(); let voting_power = member.voting_power.saturating_add(amount); self.members @@ -392,7 +392,7 @@ mod dao { #[ink(message)] pub fn get_member(&mut self, account: AccountId) -> Member { - self.members.get(account).unwrap_or(Member { voting_power: 0, last_vote: 0 }) + self.members.get(account).unwrap_or_default() } #[ink(message)] @@ -443,7 +443,7 @@ mod dao { impl From for Error { fn from(error: Psp22Error) -> Self { - Error::Psp22(error) + Self::Psp22(error) } } diff --git a/pop-api/examples/dao/src/tests.rs b/pop-api/examples/dao/src/tests.rs index babbc5d79..864a943f9 100644 --- a/pop-api/examples/dao/src/tests.rs +++ b/pop-api/examples/dao/src/tests.rs @@ -252,20 +252,20 @@ fn proposal_enactment_works(mut session: Session) { let now = session.sandbox().block_number(); session.set_actor(CHARLIE); - // Charlie vote + // Charlie votes assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, Vote { who: Some(account_id_from_slice(&CHARLIE)), when: Some(now) } ); - // Alice vote + // Alice votes session.set_actor(ALICE); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( &session, Vote { who: Some(account_id_from_slice(&ALICE)), when: Some(now) } ); - // BOB vote + // BOB votes session.set_actor(BOB); assert_ok!(vote(&mut session, 1, true)); assert_last_contract_event!( @@ -457,17 +457,14 @@ fn proposal(session: &mut Session, proposal_id: u32) -> Option { } fn prepare_dao(session: &mut Session, contract: AccountId) -> Result<(), Error> { - assert_ok!(session.sandbox().mint_into(&TOKEN, &ALICE, AMOUNT)); - assert_ok!(session.sandbox().approve(&TOKEN, &ALICE, &contract.clone(), AMOUNT)); - assert_ok!(session.sandbox().mint_into(&TOKEN, &BOB, AMOUNT)); - assert_ok!(session.sandbox().approve(&TOKEN, &BOB, &contract.clone(), AMOUNT)); - assert_ok!(session.sandbox().mint_into(&TOKEN, &CHARLIE, AMOUNT)); - assert_ok!(session.sandbox().approve(&TOKEN, &CHARLIE, &contract.clone(), AMOUNT)); - session.set_actor(ALICE); - assert_ok!(join(session, AMOUNT / 2)); - session.set_actor(BOB); - assert_ok!(join(session, AMOUNT / 4)); - session.set_actor(CHARLIE); - assert_ok!(join(session, AMOUNT / 3)); + let users: Vec = vec![ALICE, CHARLIE, BOB]; + let mut coeff = 2; + for user in users { + assert_ok!(session.sandbox().mint_into(&TOKEN, &user, AMOUNT)); + assert_ok!(session.sandbox().approve(&TOKEN, &user, &contract.clone(), AMOUNT)); + session.set_actor(user); + assert_ok!(join(session, AMOUNT / coeff)); + coeff += 1; + } Ok(()) }