From 966b3768a040f027f58717b7b50dfdac10ad1448 Mon Sep 17 00:00:00 2001 From: Amiya Behera Date: Sat, 25 May 2024 15:34:39 +0530 Subject: [PATCH] positive-externality --- Cargo.lock | 24 + Cargo.toml | 9 + pallets/positive-externality/Cargo.toml | 56 +++ pallets/positive-externality/README.md | 1 + .../positive-externality-rpc/Cargo.toml | 14 + .../positive-externality-rpc/src/lib.rs | 226 ++++++++++ .../Cargo.toml | 17 + .../src/lib.rs | 18 + .../positive-externality/src/benchmarking.rs | 35 ++ pallets/positive-externality/src/extras.rs | 178 ++++++++ pallets/positive-externality/src/lib.rs | 424 ++++++++++++++++++ pallets/positive-externality/src/mock.rs | 160 +++++++ pallets/positive-externality/src/tests.rs | 208 +++++++++ pallets/positive-externality/src/types.rs | 35 ++ pallets/positive-externality/src/weights.rs | 91 ++++ pallets/project-tips/src/mock.rs | 6 +- 16 files changed, 1498 insertions(+), 4 deletions(-) create mode 100644 pallets/positive-externality/Cargo.toml create mode 100644 pallets/positive-externality/README.md create mode 100644 pallets/positive-externality/positive-externality-rpc/Cargo.toml create mode 100644 pallets/positive-externality/positive-externality-rpc/src/lib.rs create mode 100644 pallets/positive-externality/positive-externality-runtime-api/Cargo.toml create mode 100644 pallets/positive-externality/positive-externality-runtime-api/src/lib.rs create mode 100644 pallets/positive-externality/src/benchmarking.rs create mode 100644 pallets/positive-externality/src/extras.rs create mode 100644 pallets/positive-externality/src/lib.rs create mode 100644 pallets/positive-externality/src/mock.rs create mode 100644 pallets/positive-externality/src/tests.rs create mode 100644 pallets/positive-externality/src/types.rs create mode 100644 pallets/positive-externality/src/weights.rs diff --git a/Cargo.lock b/Cargo.lock index 9878f9e..9a1da3f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -8936,6 +8936,30 @@ dependencies = [ "tp-traits", ] +[[package]] +name = "pallet-positive-externality" +version = "4.0.0-dev" +dependencies = [ + "frame-benchmarking", + "frame-support", + "frame-support-test", + "frame-system", + "pallet-balances", + "pallet-schelling-game-shared", + "pallet-shared-storage", + "pallet-sortition-sum-game", + "pallet-support", + "pallet-timestamp", + "parity-scale-codec", + "scale-info", + "sp-core", + "sp-io", + "sp-runtime", + "sp-std", + "trait-schelling-game-shared", + "trait-shared-storage", +] + [[package]] name = "pallet-preimage" version = "4.0.0-dev" diff --git a/Cargo.toml b/Cargo.toml index 75110f7..d57bdf3 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -77,6 +77,7 @@ pallet-shared-storage = {path = "pallets/shared-storage", default-features = fal pallet-schelling-game-shared = {path = "pallets/schelling-game-shared", default-features = false } pallet-profile-validation = {path = "pallets/profile-validation", default-features = false} pallet-project-tips = {path = "pallets/project-tips", default-features = false} +pallet-positive-externality = { path = "pallets/positive-externality", default-features = false } ## Traits trait-sortition-sum-game = {path = "traits/trait-sortition-sum-game", default-features=false} @@ -86,6 +87,14 @@ trait-schelling-game-shared = {path = "traits/trait-schelling-game-shared", defa ## Api profile-validation-runtime-api = { path = "pallets/profile-validation/profile-validation-runtime-api", default-features = false} project-tips-runtime-api = { path = "pallets/project-tips/project-tips-runtime-api", default-features = false } +positive-externality-runtime-api = { path = "pallets/positive-externality/positive-externality-runtime-api", default-features = false} + + +## Rpc +profile-validation-rpc = { path = "pallets/profile-validation/profile-validation-rpc", default-features = false} +project-tips-rpc = { path = "pallets/project-tips/project-tips-rpc", default-features = false } +positive-externality-rpc = { path = "pallets/positive-externality/positive-externality-rpc", default-features = false} + ## Additional dependancies diff --git a/pallets/positive-externality/Cargo.toml b/pallets/positive-externality/Cargo.toml new file mode 100644 index 0000000..0cef975 --- /dev/null +++ b/pallets/positive-externality/Cargo.toml @@ -0,0 +1,56 @@ +[package] +name = "pallet-positive-externality" +version = "4.0.0-dev" +description = "FRAME pallet template for defining custom runtime logic." +authors = ["Substrate DevHub "] +homepage = "https://substrate.io" +edition = "2021" +license = "MIT-0" +publish = false +repository = "https://github.com/substrate-developer-hub/substrate-node-template/" + +[package.metadata.docs.rs] +targets = ["x86_64-unknown-linux-gnu"] + +[dependencies] +parity-scale-codec = { workspace = true } +scale-info = { workspace = true } +frame-benchmarking = { workspace = true , optional = true} +frame-support = { workspace = true } +frame-system = { workspace = true } +sp-std = { workspace = true} +pallet-timestamp = { workspace = true } +pallet-balances = { workspace = true } +pallet-support = { workspace = true } +pallet-shared-storage = { workspace = true } +trait-shared-storage = { workspace = true } +pallet-schelling-game-shared = { workspace = true } +trait-schelling-game-shared = { workspace = true } +pallet-sortition-sum-game = { workspace = true } + + + +[dev-dependencies] +sp-core = { workspace = true } +sp-io = { workspace = true } +sp-runtime = { workspace = true } +frame-support-test = { workspace = true } + +[features] +default = ["std"] +std = [ + "parity-scale-codec/std", + "frame-benchmarking?/std", + "frame-support/std", + "frame-system/std", + "scale-info/std", + "pallet-timestamp/std", + "pallet-balances/std", + "pallet-support/std", + "pallet-shared-storage/std", + "pallet-schelling-game-shared/std", + "pallet-sortition-sum-game/std", + "frame-support-test/std", +] +runtime-benchmarks = ["frame-benchmarking/runtime-benchmarks"] +try-runtime = ["frame-support/try-runtime"] diff --git a/pallets/positive-externality/README.md b/pallets/positive-externality/README.md new file mode 100644 index 0000000..d0d5953 --- /dev/null +++ b/pallets/positive-externality/README.md @@ -0,0 +1 @@ +License: MIT-0 \ No newline at end of file diff --git a/pallets/positive-externality/positive-externality-rpc/Cargo.toml b/pallets/positive-externality/positive-externality-rpc/Cargo.toml new file mode 100644 index 0000000..b7a0241 --- /dev/null +++ b/pallets/positive-externality/positive-externality-rpc/Cargo.toml @@ -0,0 +1,14 @@ +[package] +name = "positive-externality-rpc" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +jsonrpsee = { workspace = true } +sc-rpc = { workspace = true } +sp-api = { workspace = true } +sp-blockchain = { workspace = true } +sp-runtime = { workspace = true } +positive-externality-runtime-api = { workspace = true} \ No newline at end of file diff --git a/pallets/positive-externality/positive-externality-rpc/src/lib.rs b/pallets/positive-externality/positive-externality-rpc/src/lib.rs new file mode 100644 index 0000000..60d9331 --- /dev/null +++ b/pallets/positive-externality/positive-externality-rpc/src/lib.rs @@ -0,0 +1,226 @@ +use jsonrpsee::{ + core::{Error as JsonRpseeError, RpcResult}, + proc_macros::rpc, + types::error::{CallError, ErrorCode, ErrorObject}, +}; +use positive_externality_runtime_api::PositiveExternalityApi as PositiveExternalityRuntimeApi; +use sp_api::codec::Codec; +use sp_api::ProvideRuntimeApi; +use sp_blockchain::HeaderBackend; +use sp_runtime::traits::Block as BlockT; +use std::sync::Arc; + +#[rpc(client, server)] +pub trait PositiveExternalityApi { + #[method(name = "positiveexternality_evidenceperiodendblock")] + fn get_evidence_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult>; + #[method(name = "positiveexternality_stakingperiodendblock")] + fn get_staking_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult>; + #[method(name = "positiveexternality_drawingperiodend")] + fn get_drawing_period_end( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult<(u64, u64, bool)>; + #[method(name = "positiveexternality_commitendblock")] + fn get_commit_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult>; + #[method(name = "positiveexternality_voteendblock")] + fn get_vote_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult>; + #[method(name = "positiveexternality_selectedjuror")] + fn selected_as_juror( + &self, + user_to_calculate: AccountId, + who: AccountId, + at: Option, + ) -> RpcResult; +} + +/// A struct that implements the `SumStorageApi`. +pub struct PositiveExternality { + // If you have more generics, no need to SumStorage + // just use a tuple like SumStorage + client: Arc, + _marker: std::marker::PhantomData, +} + +impl PositiveExternality { + /// Create new `SumStorage` instance with the given reference to the client. + pub fn new(client: Arc) -> Self { + Self { client, _marker: Default::default() } + } +} + +/// Error type of this RPC api. +pub enum Error { + /// The transaction was not decodable. + DecodeError, + /// The call to runtime failed. + RuntimeError, +} + +impl From for i32 { + fn from(e: Error) -> i32 { + match e { + Error::RuntimeError => 1, + Error::DecodeError => 2, + } + } +} + +impl PositiveExternalityApiServer<::Hash, AccountId> + for PositiveExternality +where + Block: BlockT, + AccountId: Codec, + C: Send + Sync + 'static, + C: ProvideRuntimeApi, + C: HeaderBackend, + C::Api: PositiveExternalityRuntimeApi, +{ + fn get_evidence_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.get_evidence_period_end_block(at, user_to_calculate); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } + fn get_staking_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.get_staking_period_end_block(at, user_to_calculate); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } + fn get_drawing_period_end( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult<(u64, u64, bool)> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.get_drawing_period_end(at, user_to_calculate); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } + + fn get_commit_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.get_commit_period_end_block(at, user_to_calculate); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } + + fn get_vote_period_end_block( + &self, + user_to_calculate: AccountId, + at: Option, + ) -> RpcResult> { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.get_vote_period_end_block(at, user_to_calculate); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } + + fn selected_as_juror( + &self, + user_to_calculate: AccountId, + who: AccountId, + at: Option, + ) -> RpcResult { + let api = self.client.runtime_api(); + let at = at.unwrap_or_else(|| + // If the block hash is not supplied assume the best block. + self.client.info().best_hash); + + let runtime_api_result = api.selected_as_juror(at, user_to_calculate, who); + fn map_err(error: impl ToString, desc: &'static str) -> CallError { + CallError::Custom(ErrorObject::owned( + Error::RuntimeError.into(), + desc, + Some(error.to_string()), + )) + } + let res = runtime_api_result.map_err(|e| map_err(e, "Unable to query dispatch info."))?; + Ok(res) + } +} diff --git a/pallets/positive-externality/positive-externality-runtime-api/Cargo.toml b/pallets/positive-externality/positive-externality-runtime-api/Cargo.toml new file mode 100644 index 0000000..739cf39 --- /dev/null +++ b/pallets/positive-externality/positive-externality-runtime-api/Cargo.toml @@ -0,0 +1,17 @@ +[package] +name = "positive-externality-runtime-api" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +sp-api = { workspace = true } +frame-support = { workspace = true } + +[features] +default = ["std"] +std = [ + "sp-api/std", + "frame-support/std", +] \ No newline at end of file diff --git a/pallets/positive-externality/positive-externality-runtime-api/src/lib.rs b/pallets/positive-externality/positive-externality-runtime-api/src/lib.rs new file mode 100644 index 0000000..b8c2483 --- /dev/null +++ b/pallets/positive-externality/positive-externality-runtime-api/src/lib.rs @@ -0,0 +1,18 @@ +#![cfg_attr(not(feature = "std"), no_std)] + +// use frame_support::sp_std::{vec::Vec}; +// or +use frame_support::sp_std::prelude::*; +use sp_api::codec::Codec; + +sp_api::decl_runtime_apis! { + pub trait PositiveExternalityApi where AccountId: Codec { + + fn get_evidence_period_end_block(user_to_calculate: AccountId) -> Option; + fn get_staking_period_end_block(user_to_calculate: AccountId) -> Option; + fn get_drawing_period_end(user_to_calculate: AccountId) -> (u64, u64, bool); + fn get_commit_period_end_block(user_to_calculate: AccountId) -> Option; + fn get_vote_period_end_block(user_to_calculate: AccountId) -> Option; + fn selected_as_juror(user_to_calculate: AccountId, who: AccountId) -> bool; + } +} diff --git a/pallets/positive-externality/src/benchmarking.rs b/pallets/positive-externality/src/benchmarking.rs new file mode 100644 index 0000000..5a26241 --- /dev/null +++ b/pallets/positive-externality/src/benchmarking.rs @@ -0,0 +1,35 @@ +//! Benchmarking setup for pallet-template +#![cfg(feature = "runtime-benchmarks")] +use super::*; + +#[allow(unused)] +use crate::Pallet as Template; +use frame_benchmarking::v2::*; +use frame_system::RawOrigin; + +#[benchmarks] +mod benchmarks { + use super::*; + + #[benchmark] + fn do_something() { + let value = 100u32.into(); + let caller: T::AccountId = whitelisted_caller(); + #[extrinsic_call] + do_something(RawOrigin::Signed(caller), value); + + assert_eq!(Something::::get(), Some(value)); + } + + #[benchmark] + fn cause_error() { + Something::::put(100u32); + let caller: T::AccountId = whitelisted_caller(); + #[extrinsic_call] + cause_error(RawOrigin::Signed(caller)); + + assert_eq!(Something::::get(), Some(101u32)); + } + + impl_benchmark_test_suite!(Template, crate::mock::new_test_ext(), crate::mock::Test); +} diff --git a/pallets/positive-externality/src/extras.rs b/pallets/positive-externality/src/extras.rs new file mode 100644 index 0000000..710f3f8 --- /dev/null +++ b/pallets/positive-externality/src/extras.rs @@ -0,0 +1,178 @@ +use frame_support::dispatch::DispatchResult; + +use super::*; + +impl Post { + pub fn new(id: PostId, created_by: T::AccountId, content: Content) -> Self { + Post { + id, + created: new_who_and_when::(created_by.clone()), + edited: false, + owner: created_by, + content, + hidden: false, + upvotes_count: 0, + downvotes_count: 0, + } + } + + pub fn ensure_owner(&self, account: &T::AccountId) -> DispatchResult { + ensure!(self.is_owner(account), Error::::NotAPostOwner); + Ok(()) + } + + pub fn is_owner(&self, account: &T::AccountId) -> bool { + self.owner == *account + } +} + +impl Pallet { + pub(super) fn get_phase_data() -> PhaseData { + T::SchellingGameSharedSource::create_phase_data(50, 5, 3, 100, (100, 100)) + } + + pub fn ensure_validation_on_positive_externality(account: T::AccountId) -> DispatchResult { + let bool_data = Validate::::get(account); + ensure!(bool_data == true, Error::::ValidationPositiveExternalityIsOff); + + Ok(()) + } + + pub fn ensure_min_stake_positive_externality(account: T::AccountId) -> DispatchResult { + let stake = StakeBalance::::get(account); + let min_stake = MinimumStake::::get(); + // println!("stake {:?}", stake); + // println!("min stake {:?}", min_stake); + ensure!(stake >= min_stake, Error::::LessThanMinStake); + + Ok(()) + } + + pub(super) fn u64_to_balance_saturated(input: u64) -> BalanceOf { + input.saturated_into::>() + } + + pub(super) fn u64_to_block_saturated(input: u64) -> BlockNumberOf { + input.saturated_into::>() + } + + pub(super) fn get_drawn_jurors(user_to_calculate: T::AccountId) -> Vec<(T::AccountId, u64)> { + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + T::SchellingGameSharedSource::get_drawn_jurors(key) + } + + // Block code start + + pub fn get_evidence_period_end_block(user_to_calculate: T::AccountId) -> Option { + let now = >::block_number(); + + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + let result = T::SchellingGameSharedSource::get_evidence_period_end_block_helper_link( + key, phase_data, now, + ); + result + } + + pub fn get_staking_period_end_block(user_to_calculate: T::AccountId) -> Option { + let now = >::block_number(); + + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + let result = T::SchellingGameSharedSource::get_staking_period_end_block_helper_link( + key, phase_data, now, + ); + result + } + + pub fn get_drawing_period_end(user_to_calculate: T::AccountId) -> (u64, u64, bool) { + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + let phase_data = Self::get_phase_data(); + + let result = + T::SchellingGameSharedSource::get_drawing_period_end_helper_link(key, phase_data); + result + } + + pub fn get_commit_period_end_block(user_to_calculate: T::AccountId) -> Option { + let now = >::block_number(); + + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + let result = T::SchellingGameSharedSource::get_commit_period_end_block_helper_link( + key, phase_data, now, + ); + result + } + + pub fn get_vote_period_end_block(user_to_calculate: T::AccountId) -> Option { + let now = >::block_number(); + + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + let result = T::SchellingGameSharedSource::get_vote_period_end_block_helper_link( + key, phase_data, now, + ); + result + } + + pub fn selected_as_juror(user_to_calculate: T::AccountId, who: T::AccountId) -> bool { + let pe_block_number = + >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let result = T::SchellingGameSharedSource::selected_as_juror_helper_link(key, who); + result + } + + // Block code end +} diff --git a/pallets/positive-externality/src/lib.rs b/pallets/positive-externality/src/lib.rs new file mode 100644 index 0000000..34c3684 --- /dev/null +++ b/pallets/positive-externality/src/lib.rs @@ -0,0 +1,424 @@ +#![cfg_attr(not(feature = "std"), no_std)] + +/// Edit this file to define custom logic or remove it if it is not needed. +/// Learn more about FRAME and the core library of Substrate FRAME pallets: +/// +pub use pallet::*; + +#[cfg(test)] +mod mock; + +#[cfg(test)] +mod tests; + +#[cfg(feature = "runtime-benchmarks")] +mod benchmarking; +pub mod weights; +pub use weights::*; + +mod extras; +pub mod types; +pub use types::{Post, FIRST_POST_ID}; + +use frame_support::sp_runtime::traits::Saturating; +use frame_support::sp_runtime::SaturatedConversion; +use sp_std::prelude::*; +use frame_support::{ + dispatch::{DispatchResult}, + ensure, fail, +}; +use frame_system::pallet_prelude::*; +use frame_support::pallet_prelude::*; +use frame_support::pallet_prelude::DispatchError; +use frame_support::{ + traits::{Currency, ExistenceRequirement, Get, ReservableCurrency, WithdrawReasons}, + PalletId, +}; +use pallet_support::{ + ensure_content_is_valid, new_who_and_when, remove_from_vec, Content, PostId, WhoAndWhen, + WhoAndWhenOf, +}; +use pallet_schelling_game_shared::types::{Period, PhaseData, RangePoint, SchellingGameType}; +use trait_schelling_game_shared::SchellingGameSharedLink; +use trait_shared_storage::SharedStorageLink; +use pallet_sortition_sum_game::types::SumTreeName; +type AccountIdOf = ::AccountId; +type BalanceOf = <::Currency as Currency>>::Balance; +pub type BlockNumberOf = BlockNumberFor; +pub type SumTreeNameType = SumTreeName, BlockNumberOf>; + +#[frame_support::pallet(dev_mode)] +pub mod pallet { + use super::*; + + + + #[pallet::pallet] + #[pallet::without_storage_info] + pub struct Pallet(_); + + /// Configure the pallet by specifying the parameters and types on which it depends. + #[pallet::config] + pub trait Config: + frame_system::Config + pallet_timestamp::Config + pallet_schelling_game_shared::Config + { + /// Because this pallet emits events, it depends on the runtime's definition of an event. + type RuntimeEvent: From> + IsType<::RuntimeEvent>; + /// Type representing the weight of this pallet + type WeightInfo: WeightInfo; + + type SharedStorageSource: SharedStorageLink>; + type SchellingGameSharedSource: SchellingGameSharedLink< + SumTreeName = SumTreeName>, + SchellingGameType = SchellingGameType, + BlockNumber = BlockNumberOf, + AccountId = AccountIdOf, + Balance = BalanceOf, + RangePoint = RangePoint, + Period = Period, + PhaseData = PhaseData, + >; + type Currency: ReservableCurrency; + } + + // The pallet's runtime storage items. + // https://docs.substrate.io/main-docs/build/runtime-storage/ + #[pallet::storage] + #[pallet::getter(fn something)] + // Learn more about declaring storage items: + // https://docs.substrate.io/main-docs/build/runtime-storage/#declaring-storage-items + pub type Something = StorageValue<_, u32>; + + #[pallet::type_value] + pub fn DefaultForNextPostId() -> PostId { + FIRST_POST_ID + } + + /// The next post id. + #[pallet::storage] + #[pallet::getter(fn next_post_id)] + pub type NextPostId = StorageValue<_, PostId, ValueQuery, DefaultForNextPostId>; + + /// Get the details of a post by its' id. + #[pallet::storage] + #[pallet::getter(fn post_by_id)] + pub type PostById = StorageMap<_, Twox64Concat, PostId, Post>; + + #[pallet::storage] + #[pallet::getter(fn evidence)] + pub type Evidence = + StorageMap<_, Blake2_128Concat, T::AccountId, Vec, ValueQuery>; + + #[pallet::type_value] + pub fn MinimumStake() -> BalanceOf { + 10000u128.saturated_into::>() + } + + #[pallet::storage] + #[pallet::getter(fn user_stake)] + pub type StakeBalance = + StorageMap<_, Twox64Concat, T::AccountId, BalanceOf, ValueQuery>; + + #[pallet::type_value] + pub fn DefaultValidate() -> bool { + true + } + + #[pallet::storage] + #[pallet::getter(fn validate)] + pub type Validate = + StorageMap<_, Twox64Concat, T::AccountId, bool, ValueQuery, DefaultValidate>; + + #[pallet::storage] + #[pallet::getter(fn validation_block)] + pub type ValidationBlock = + StorageMap<_, Blake2_128Concat, T::AccountId, BlockNumberOf, ValueQuery>; + + // Pallets use events to inform users when important changes are made. + // https://docs.substrate.io/main-docs/build/events-errors/ + #[pallet::event] + #[pallet::generate_deposit(pub(super) fn deposit_event)] + pub enum Event { + /// Event documentation should end with an array that provides descriptive names for event + /// parameters. [something, who] + SomethingStored { something: u32, who: T::AccountId }, + } + + // Errors inform users that something went wrong. + #[pallet::error] + pub enum Error { + /// Error names should be descriptive. + NoneValue, + /// Errors should have helpful documentation associated with them. + StorageOverflow, + NotAPostOwner, + ValidationPositiveExternalityIsOff, + LessThanMinStake, + CannotStakeNow, + ChoiceOutOfRange, + } + + // Dispatchable functions allows users to interact with the pallet and invoke state changes. + // These functions materialize as "extrinsics", which are often compared to transactions. + // Dispatchable functions must be annotated with a weight and must return a DispatchResult. + #[pallet::call] + impl Pallet { + #[pallet::call_index(0)] + #[pallet::weight(0)] + pub fn create_positive_externality_post( + origin: OriginFor, + content: Content, + ) -> DispatchResult { + let creator = ensure_signed(origin)?; + + ensure_content_is_valid(content.clone())?; + + // Citizen approved To comment out in production, citizen approved in added in profile validation + + // T::SharedStorageSource::check_citizen_is_approved_link(creator.clone())?; + + let new_post_id = Self::next_post_id(); + + let new_post: Post = Post::new(new_post_id, creator.clone(), content.clone()); + + Evidence::::mutate(creator, |ids| ids.push(new_post_id)); + + PostById::insert(new_post_id, new_post); + NextPostId::::mutate(|n| { + *n += 1; + }); + + // emit event + + Ok(()) + } + + #[pallet::call_index(1)] + #[pallet::weight(0)] + pub fn set_validate_positive_externality( + origin: OriginFor, + value: bool, + ) -> DispatchResult { + let who = ensure_signed(origin)?; + // Check user has done kyc + + Validate::::insert(&who, value); + // emit event + Ok(()) + } + + #[pallet::call_index(2)] + #[pallet::weight(0)] + pub fn apply_staking_period( + origin: OriginFor, + user_to_calculate: T::AccountId, + ) -> DispatchResult { + let who = ensure_signed(origin)?; + + Self::ensure_validation_on_positive_externality(user_to_calculate.clone())?; + + let stake = MinimumStake::::get(); + + let _ = ::Currency::withdraw( + &who, + stake, + WithdrawReasons::TRANSFER, + ExistenceRequirement::AllowDeath, + )?; + + StakeBalance::::insert(&user_to_calculate, stake); + + let pe_block_number = >::get(user_to_calculate.clone()); + // println!("{:?}", pe_block_number); + let zero_block_number = Self::u64_to_block_saturated(0); + let now = >::block_number(); + let three_month_number = (3 * 30 * 24 * 60 * 60) / 6; + let three_month_block = Self::u64_to_block_saturated(three_month_number); + let modulus = now % three_month_block; + let storage_main_block = now - modulus; + // println!("{:?}", now); + // println!("{:?}", three_month_number); + // println!("{:?}", storage_main_block); + // println!("{:?}", pe_block_number); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate.clone(), + block_number: storage_main_block.clone(), + }; + + // let game_type = SchellingGameType::PositiveExternality; + // || pe_block_number == zero_block_number + + if storage_main_block > pe_block_number || pe_block_number == zero_block_number { + >::insert(user_to_calculate.clone(), storage_main_block); + // check what if called again + T::SchellingGameSharedSource::set_to_staking_period_pe_link(key.clone(), now)?; + T::SchellingGameSharedSource::create_tree_helper_link(key, 3)?; + + // println!("{:?}", data); + } else { + return Err(Error::::CannotStakeNow.into()); + } + + Ok(()) + } + + #[pallet::call_index(3)] + #[pallet::weight(0)] + pub fn apply_jurors( + origin: OriginFor, + user_to_calculate: T::AccountId, + stake: BalanceOf, + ) -> DispatchResult { + let who = ensure_signed(origin)?; + + Self::ensure_validation_on_positive_externality(user_to_calculate.clone())?; + Self::ensure_min_stake_positive_externality(user_to_calculate.clone())?; + + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + T::SchellingGameSharedSource::apply_jurors_helper_link(key, phase_data, who, stake)?; + + Ok(()) + } + + #[pallet::call_index(4)] + #[pallet::weight(0)] + pub fn pass_period( + origin: OriginFor, + user_to_calculate: T::AccountId, + ) -> DispatchResult { + let _who = ensure_signed(origin)?; + + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let now = >::block_number(); + let phase_data = Self::get_phase_data(); + T::SchellingGameSharedSource::change_period_link(key, phase_data, now)?; + + Ok(()) + } + + #[pallet::call_index(5)] + #[pallet::weight(0)] + pub fn draw_jurors( + origin: OriginFor, + user_to_calculate: T::AccountId, + iterations: u64, + ) -> DispatchResult { + let _who = ensure_signed(origin)?; + + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + + T::SchellingGameSharedSource::draw_jurors_helper_link(key, phase_data, iterations)?; + + Ok(()) + } + + // Unstaking + // Stop drawn juror to unstake ✔️ + #[pallet::call_index(6)] + #[pallet::weight(0)] + pub fn unstaking(origin: OriginFor, user_to_calculate: T::AccountId) -> DispatchResult { + let who = ensure_signed(origin)?; + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + T::SchellingGameSharedSource::unstaking_helper_link(key, who)?; + Ok(()) + } + + #[pallet::call_index(7)] + #[pallet::weight(0)] + pub fn commit_vote( + origin: OriginFor, + user_to_calculate: T::AccountId, + vote_commit: [u8; 32], + ) -> DispatchResult { + let who = ensure_signed(origin)?; + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + T::SchellingGameSharedSource::commit_vote_for_score_helper_link(key, who, vote_commit)?; + Ok(()) + } + + #[pallet::call_index(8)] + #[pallet::weight(0)] + pub fn reveal_vote( + origin: OriginFor, + user_to_calculate: T::AccountId, + choice: i64, + salt: Vec, + ) -> DispatchResult { + let who = ensure_signed(origin)?; + + ensure!(choice <= 5 && choice >= 1, Error::::ChoiceOutOfRange); + + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate, + block_number: pe_block_number.clone(), + }; + + T::SchellingGameSharedSource::reveal_vote_score_helper_link(key, who, choice, salt)?; + Ok(()) + } + + #[pallet::call_index(9)] + #[pallet::weight(0)] + pub fn get_incentives( + origin: OriginFor, + user_to_calculate: T::AccountId, + ) -> DispatchResult { + let _who = ensure_signed(origin)?; + let pe_block_number = >::get(user_to_calculate.clone()); + + let key = SumTreeName::PositiveExternality { + user_address: user_to_calculate.clone(), + block_number: pe_block_number.clone(), + }; + + let phase_data = Self::get_phase_data(); + T::SchellingGameSharedSource::get_incentives_score_schelling_helper_link( + key.clone(), + phase_data, + RangePoint::ZeroToFive, + )?; + + let score = T::SchellingGameSharedSource::get_mean_value_link(key.clone()); + // println!("Score {:?}", score); + T::SharedStorageSource::set_positive_externality_link(user_to_calculate, score)?; + + Ok(()) + } + } +} diff --git a/pallets/positive-externality/src/mock.rs b/pallets/positive-externality/src/mock.rs new file mode 100644 index 0000000..e6c7720 --- /dev/null +++ b/pallets/positive-externality/src/mock.rs @@ -0,0 +1,160 @@ +use crate as pallet_template; +use frame_support::{ + derive_impl, + parameter_types, + traits::{ConstU16, ConstU64}, +}; +use frame_support_test::TestRandomness; +use sp_core::H256; +use sp_runtime::{ + traits::{BlakeTwo256, IdentityLookup}, + BuildStorage, +}; + +type Block = frame_system::mocking::MockBlock; + +// Configure a mock runtime to test the pallet. +frame_support::construct_runtime!( + pub enum Test + { + System: frame_system, + TemplateModule: pallet_template, + Balances: pallet_balances, + Timestamp: pallet_timestamp, + SharedStorage:pallet_shared_storage, + SchellingGameShared: pallet_schelling_game_shared, + SortitionSumGame: pallet_sortition_sum_game, + } +); + +#[derive_impl(frame_system::config_preludes::TestDefaultConfig as frame_system::DefaultConfig)] +impl frame_system::Config for Test { + type BaseCallFilter = frame_support::traits::Everything; + type BlockWeights = (); + type BlockLength = (); + type DbWeight = (); + type RuntimeOrigin = RuntimeOrigin; + type RuntimeCall = RuntimeCall; + type Nonce = u64; + type Hash = H256; + type Hashing = BlakeTwo256; + type AccountId = u64; + type Lookup = IdentityLookup; + type Block = Block; + type RuntimeEvent = RuntimeEvent; + type BlockHashCount = ConstU64<250>; + type Version = (); + type PalletInfo = PalletInfo; + type OnNewAccount = (); + type OnKilledAccount = (); + type SystemWeightInfo = (); + type SS58Prefix = ConstU16<42>; + type OnSetCode = (); + type MaxConsumers = frame_support::traits::ConstU32<16>; + type AccountData = pallet_balances::AccountData; // N +} + +impl pallet_shared_storage::Config for Test { + type RuntimeEvent = RuntimeEvent; + type WeightInfo = (); +} +parameter_types! { + pub const MinimumPeriod: u64 = 5; +} + +impl pallet_timestamp::Config for Test { + type Moment = u64; + type OnTimestampSet = (); + type MinimumPeriod = MinimumPeriod; + type WeightInfo = (); +} + +impl pallet_balances::Config for Test { + type MaxHolds = (); + type MaxLocks = (); + type MaxReserves = (); + type ReserveIdentifier = [u8; 8]; + type Balance = u64; + type RuntimeEvent = RuntimeEvent; + type DustRemoval = (); + type ExistentialDeposit = ConstU64<1>; + type AccountStore = System; + type WeightInfo = (); + type FreezeIdentifier = (); + type MaxFreezes = (); + type RuntimeHoldReason = (); + type RuntimeFreezeReason = (); +} + +impl pallet_schelling_game_shared::Config for Test { + type RuntimeEvent = RuntimeEvent; + type WeightInfo = (); + type Currency = Balances; // New code + type RandomnessSource = TestRandomness; + type Slash = (); + type Reward = (); + type SortitionSumGameSource = SortitionSumGame; +} + +impl pallet_sortition_sum_game::Config for Test { + type RuntimeEvent = RuntimeEvent; + type WeightInfo = (); +} + +impl pallet_template::Config for Test { + type RuntimeEvent = RuntimeEvent; + type WeightInfo = (); + type SharedStorageSource = SharedStorage; + type Currency = Balances; // New code + type SchellingGameSharedSource = SchellingGameShared; +} + +// Build genesis storage according to the mock runtime. +pub fn new_test_ext() -> sp_io::TestExternalities { + let mut t = frame_system::GenesisConfig::::default().build_storage().unwrap(); + pallet_balances::GenesisConfig:: { + balances: vec![ + (1, 100000), + (2, 200000), + (3, 300000), + (4, 300000), + (5, 300000), + (6, 300000), + (7, 300000), + (8, 300000), + (9, 300000), + (10, 300000), + (11, 300000), + (12, 300000), + (13, 300000), + (14, 300000), + (15, 300000), + (16, 300000), + (17, 300000), + (18, 300000), + (19, 300000), + (20, 300000), + (21, 300000), + (22, 300000), + (23, 300000), + (24, 300000), + (25, 300000), + (26, 300000), + (27, 300000), + (28, 300000), + (29, 300000), + (30, 300000), + (31, 300000), + (32, 300000), + (33, 300000), + (34, 300000), + (35, 300000), + ], + } // new code + .assimilate_storage(&mut t) + .unwrap(); + pallet_shared_storage::GenesisConfig:: { approved_citizen_address: vec![1, 2] } + .assimilate_storage(&mut t) + .unwrap(); + t.into() +} diff --git a/pallets/positive-externality/src/tests.rs b/pallets/positive-externality/src/tests.rs new file mode 100644 index 0000000..8421940 --- /dev/null +++ b/pallets/positive-externality/src/tests.rs @@ -0,0 +1,208 @@ +use crate::types::Post; +use crate::{mock::*, Error, Event}; +use frame_support::{assert_noop, assert_ok}; +use pallet_support::{Content, WhoAndWhen}; + +#[test] +fn test_positive_externality_post() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::create_positive_externality_post( + RuntimeOrigin::signed(1), + Content::None + )); + let post = TemplateModule::post_by_id(1); + let post_compare = Some(Post { + id: 1, + created: WhoAndWhen { account: 1, block: 0, time: 0 }, + edited: false, + owner: 1, + content: Content::None, + hidden: false, + upvotes_count: 0, + downvotes_count: 0, + }); + assert_eq!(post, post_compare); + // assert_ok!(TemplateModule::apply_jurors(Origin::signed(1), 2, 60)); + }); +} + +#[test] +fn test_setting_positive_externality_validation() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + let value = TemplateModule::validate(1); + assert_eq!(value, true); + }); +} + +#[test] +fn test_applying_for_staking_period() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + System::set_block_number(1298000 + 1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + }); +} + +#[test] +fn test_appying_jurors() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + // System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(4), 1, 1000)); + }); +} + +#[test] +fn test_change_period() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(4), 1, 1000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(5), 1, 2000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(6), 1, 3000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(7), 1, 4000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(8), 1, 5000)); + System::set_block_number(1298080); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + }) +} + +#[test] +fn test_draw_jurors_period() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(4), 1, 1000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(5), 1, 2000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(6), 1, 3000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(7), 1, 4000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(8), 1, 5000)); + System::set_block_number(1298080); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + assert_ok!(TemplateModule::draw_jurors(RuntimeOrigin::signed(8), 1, 5)); + }) +} + +#[test] +fn test_drawn_jurors() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + let balance = Balances::free_balance(4); + assert_eq!(300000, balance); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(4), 1, 1000)); + let balance = Balances::free_balance(4); + assert_eq!(299000, balance); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(5), 1, 2000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(6), 1, 3000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(7), 1, 4000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(8), 1, 5000)); + System::set_block_number(1298080); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + assert_ok!(TemplateModule::draw_jurors(RuntimeOrigin::signed(8), 1, 5)); + let data = TemplateModule::get_drawn_jurors(1); + assert_eq!(data, [(4, 1000), (5, 2000), (6, 3000), (7, 4000), (8, 5000)]); + // println!("drawn jurors {:?}",data); + }) +} + +#[test] +fn test_commit_and_incentives_vote() { + new_test_ext().execute_with(|| { + assert_ok!(TemplateModule::set_validate_positive_externality( + RuntimeOrigin::signed(1), + true + )); + System::set_block_number(1298000); + assert_ok!(TemplateModule::apply_staking_period(RuntimeOrigin::signed(2), 1)); + let balance = Balances::free_balance(4); + assert_eq!(300000, balance); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(4), 1, 1000)); + let balance = Balances::free_balance(4); + assert_eq!(299000, balance); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(5), 1, 2000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(6), 1, 3000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(7), 1, 4000)); + assert_ok!(TemplateModule::apply_jurors(RuntimeOrigin::signed(8), 1, 5000)); + System::set_block_number(1298080); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + assert_ok!(TemplateModule::draw_jurors(RuntimeOrigin::signed(8), 1, 5)); + + let data = TemplateModule::get_drawn_jurors(1); + assert_eq!(data, [(4, 1000), (5, 2000), (6, 3000), (7, 4000), (8, 5000)]); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + + let hash = sp_io::hashing::keccak_256("1salt".as_bytes()); + assert_ok!(TemplateModule::commit_vote(RuntimeOrigin::signed(4), 1, hash)); + + let hash = sp_io::hashing::keccak_256("1salt2".as_bytes()); + assert_ok!(TemplateModule::commit_vote(RuntimeOrigin::signed(5), 1, hash)); + let hash = sp_io::hashing::keccak_256("5salt3".as_bytes()); + assert_ok!(TemplateModule::commit_vote(RuntimeOrigin::signed(6), 1, hash)); + let hash = sp_io::hashing::keccak_256("1salt4".as_bytes()); + assert_ok!(TemplateModule::commit_vote(RuntimeOrigin::signed(7), 1, hash)); + let hash = sp_io::hashing::keccak_256("5salt5".as_bytes()); + assert_ok!(TemplateModule::commit_vote(RuntimeOrigin::signed(8), 1, hash)); + System::set_block_number(12980160); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + assert_ok!(TemplateModule::reveal_vote( + RuntimeOrigin::signed(4), + 1, + 1, + "salt".as_bytes().to_vec() + )); + assert_ok!(TemplateModule::reveal_vote( + RuntimeOrigin::signed(5), + 1, + 1, + "salt2".as_bytes().to_vec() + )); + assert_ok!(TemplateModule::reveal_vote( + RuntimeOrigin::signed(6), + 1, + 5, + "salt3".as_bytes().to_vec() + )); + assert_ok!(TemplateModule::reveal_vote( + RuntimeOrigin::signed(7), + 1, + 1, + "salt4".as_bytes().to_vec() + )); + assert_ok!(TemplateModule::reveal_vote( + RuntimeOrigin::signed(8), + 1, + 5, + "salt5".as_bytes().to_vec() + )); + System::set_block_number(12980260); + assert_ok!(TemplateModule::pass_period(RuntimeOrigin::signed(4), 1)); + + assert_ok!(TemplateModule::get_incentives(RuntimeOrigin::signed(4), 1)); + }) +} diff --git a/pallets/positive-externality/src/types.rs b/pallets/positive-externality/src/types.rs new file mode 100644 index 0000000..a21e4e5 --- /dev/null +++ b/pallets/positive-externality/src/types.rs @@ -0,0 +1,35 @@ +use parity_scale_codec::{Decode, Encode}; +use scale_info::TypeInfo; + +use frame_support::pallet_prelude::*; + +use super::*; + +pub const FIRST_POST_ID: u64 = 1; + +/// Information about a post's owner, its' related space, content, and visibility. +#[derive(Encode, Decode, Clone, Eq, PartialEq, RuntimeDebug, TypeInfo)] +#[scale_info(skip_type_params(T))] +pub struct Post { + pub id: PostId, + + pub created: WhoAndWhenOf, + + pub edited: bool, + + pub owner: T::AccountId, + + pub content: Content, + + pub hidden: bool, + + pub upvotes_count: u32, + + pub downvotes_count: u32, +} + +#[derive(Encode, Decode, Default, Clone, Eq, PartialEq, RuntimeDebug, TypeInfo)] +pub struct PositiveExternalityPostUpdate { + pub content: Option, + pub hidden: Option, +} diff --git a/pallets/positive-externality/src/weights.rs b/pallets/positive-externality/src/weights.rs new file mode 100644 index 0000000..e8fbc09 --- /dev/null +++ b/pallets/positive-externality/src/weights.rs @@ -0,0 +1,91 @@ + +//! Autogenerated weights for pallet_template +//! +//! THIS FILE WAS AUTO-GENERATED USING THE SUBSTRATE BENCHMARK CLI VERSION 4.0.0-dev +//! DATE: 2023-04-06, STEPS: `50`, REPEAT: `20`, LOW RANGE: `[]`, HIGH RANGE: `[]` +//! WORST CASE MAP SIZE: `1000000` +//! HOSTNAME: `Alexs-MacBook-Pro-2.local`, CPU: `` +//! EXECUTION: Some(Wasm), WASM-EXECUTION: Compiled, CHAIN: Some("dev"), DB CACHE: 1024 + +// Executed Command: +// ../../target/release/node-template +// benchmark +// pallet +// --chain +// dev +// --pallet +// pallet_template +// --extrinsic +// * +// --steps=50 +// --repeat=20 +// --execution=wasm +// --wasm-execution=compiled +// --output +// pallets/template/src/weights.rs +// --template +// ../../.maintain/frame-weight-template.hbs + +#![cfg_attr(rustfmt, rustfmt_skip)] +#![allow(unused_parens)] +#![allow(unused_imports)] + +use frame_support::{traits::Get, weights::{Weight, constants::RocksDbWeight}}; +use core::marker::PhantomData; + +/// Weight functions needed for pallet_template. +pub trait WeightInfo { + fn do_something() -> Weight; + fn cause_error() -> Weight; +} + +/// Weights for pallet_template using the Substrate node and recommended hardware. +pub struct SubstrateWeight(PhantomData); +impl WeightInfo for SubstrateWeight { + /// Storage: TemplateModule Something (r:0 w:1) + /// Proof: TemplateModule Something (max_values: Some(1), max_size: Some(4), added: 499, mode: MaxEncodedLen) + fn do_something() -> Weight { + // Proof Size summary in bytes: + // Measured: `0` + // Estimated: `0` + // Minimum execution time: 8_000_000 picoseconds. + Weight::from_parts(9_000_000, 0) + .saturating_add(T::DbWeight::get().writes(1_u64)) + } + /// Storage: TemplateModule Something (r:1 w:1) + /// Proof: TemplateModule Something (max_values: Some(1), max_size: Some(4), added: 499, mode: MaxEncodedLen) + fn cause_error() -> Weight { + // Proof Size summary in bytes: + // Measured: `32` + // Estimated: `1489` + // Minimum execution time: 6_000_000 picoseconds. + Weight::from_parts(6_000_000, 1489) + .saturating_add(T::DbWeight::get().reads(1_u64)) + .saturating_add(T::DbWeight::get().writes(1_u64)) + } +} + +// For backwards compatibility and tests +impl WeightInfo for () { + /// Storage: TemplateModule Something (r:0 w:1) + /// Proof: TemplateModule Something (max_values: Some(1), max_size: Some(4), added: 499, mode: MaxEncodedLen) + fn do_something() -> Weight { + // Proof Size summary in bytes: + // Measured: `0` + // Estimated: `0` + // Minimum execution time: 8_000_000 picoseconds. + Weight::from_parts(9_000_000, 0) + .saturating_add(RocksDbWeight::get().writes(1_u64)) + } + /// Storage: TemplateModule Something (r:1 w:1) + /// Proof: TemplateModule Something (max_values: Some(1), max_size: Some(4), added: 499, mode: MaxEncodedLen) + fn cause_error() -> Weight { + // Proof Size summary in bytes: + // Measured: `32` + // Estimated: `1489` + // Minimum execution time: 6_000_000 picoseconds. + Weight::from_parts(6_000_000, 1489) + .saturating_add(RocksDbWeight::get().reads(1_u64)) + .saturating_add(RocksDbWeight::get().writes(1_u64)) + } +} diff --git a/pallets/project-tips/src/mock.rs b/pallets/project-tips/src/mock.rs index 0065873..d514043 100644 --- a/pallets/project-tips/src/mock.rs +++ b/pallets/project-tips/src/mock.rs @@ -2,16 +2,14 @@ use crate as pallet_template; use frame_support::{ derive_impl, parameter_types, - traits::{ConstU16, ConstU64, GenesisBuild}, + traits::{ConstU16, ConstU64}, }; use sp_core::H256; use sp_runtime::{ - testing::Header, traits::{BlakeTwo256, IdentityLookup}, BuildStorage, }; -type UncheckedExtrinsic = frame_system::mocking::MockUncheckedExtrinsic; type Block = frame_system::mocking::MockBlock; use frame_support_test::TestRandomness; @@ -113,7 +111,7 @@ impl pallet_sortition_sum_game::Config for Test { // Build genesis storage according to the mock runtime. pub fn new_test_ext() -> sp_io::TestExternalities { - let mut t: sp_runtime::Storage = frame_system::GenesisConfig::::default().build_storage().unwrap(); + let mut t = frame_system::GenesisConfig::::default().build_storage().unwrap(); pallet_balances::GenesisConfig:: { balances: vec![ (1, 100000),