lib.rs 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699
  1. // Ensure we're `no_std` when compiling for Wasm.
  2. #![cfg_attr(not(feature = "std"), no_std)]
  3. // Clippy linter requirement.
  4. // Disable it because of the substrate lib design
  5. // Example: pub PaidMembershipTermsById get(paid_membership_terms_by_id) build(|config: &GenesisConfig<T>| {}
  6. #![allow(clippy::redundant_closure_call)]
  7. pub mod genesis;
  8. pub(crate) mod mock;
  9. mod tests;
  10. use codec::{Codec, Decode, Encode};
  11. use frame_support::traits::Currency;
  12. use frame_support::{decl_event, decl_module, decl_storage, ensure, Parameter};
  13. use frame_system::{ensure_root, ensure_signed};
  14. use sp_arithmetic::traits::{BaseArithmetic, One};
  15. use sp_runtime::traits::{MaybeSerialize, Member};
  16. use sp_std::borrow::ToOwned;
  17. use sp_std::vec;
  18. use sp_std::vec::Vec;
  19. use common::currency::{BalanceOf, GovernanceCurrency};
  20. //TODO: Convert errors to the Substrate decl_error! macro.
  21. /// Result with string error message. This exists for backward compatibility purpose.
  22. pub type DispatchResult = Result<(), &'static str>;
  23. pub trait Trait: frame_system::Trait + GovernanceCurrency + pallet_timestamp::Trait {
  24. type Event: From<Event<Self>> + Into<<Self as frame_system::Trait>::Event>;
  25. type MemberId: Parameter
  26. + Member
  27. + BaseArithmetic
  28. + Codec
  29. + Default
  30. + Copy
  31. + MaybeSerialize
  32. + PartialEq;
  33. type PaidTermId: Parameter
  34. + Member
  35. + BaseArithmetic
  36. + Codec
  37. + Default
  38. + Copy
  39. + MaybeSerialize
  40. + PartialEq;
  41. type SubscriptionId: Parameter
  42. + Member
  43. + BaseArithmetic
  44. + Codec
  45. + Default
  46. + Copy
  47. + MaybeSerialize
  48. + PartialEq;
  49. /// Describes the common type for the working group members (workers).
  50. type ActorId: Parameter
  51. + Member
  52. + BaseArithmetic
  53. + Codec
  54. + Default
  55. + Copy
  56. + MaybeSerialize
  57. + PartialEq
  58. + Ord;
  59. }
  60. const FIRST_PAID_TERMS_ID: u8 = 1;
  61. // Default paid membership terms
  62. pub const DEFAULT_PAID_TERM_ID: u8 = 0;
  63. // Default user info constraints
  64. const DEFAULT_MIN_HANDLE_LENGTH: u32 = 5;
  65. const DEFAULT_MAX_HANDLE_LENGTH: u32 = 40;
  66. const DEFAULT_MAX_AVATAR_URI_LENGTH: u32 = 1024;
  67. const DEFAULT_MAX_ABOUT_TEXT_LENGTH: u32 = 2048;
  68. /// Public membership object alias.
  69. pub type Membership<T> = MembershipObject<
  70. <T as frame_system::Trait>::BlockNumber,
  71. <T as pallet_timestamp::Trait>::Moment,
  72. <T as Trait>::PaidTermId,
  73. <T as Trait>::SubscriptionId,
  74. <T as frame_system::Trait>::AccountId,
  75. >;
  76. #[derive(Encode, Decode, Default)]
  77. /// Stored information about a registered user
  78. pub struct MembershipObject<BlockNumber, Moment, PaidTermId, SubscriptionId, AccountId> {
  79. /// The unique handle chosen by member
  80. pub handle: Vec<u8>,
  81. /// A Url to member's Avatar image
  82. pub avatar_uri: Vec<u8>,
  83. /// Short text chosen by member to share information about themselves
  84. pub about: Vec<u8>,
  85. /// Block number when member was registered
  86. pub registered_at_block: BlockNumber,
  87. /// Timestamp when member was registered
  88. pub registered_at_time: Moment,
  89. /// How the member was registered
  90. pub entry: EntryMethod<PaidTermId, AccountId>,
  91. /// Whether the member is suspended or not.
  92. pub suspended: bool,
  93. /// The type of subscription the member has purchased if any.
  94. pub subscription: Option<SubscriptionId>,
  95. /// Member's root account id. Only the root account is permitted to set a new root account
  96. /// and update the controller account. Other modules may only allow certain actions if
  97. /// signed with root account. It is intended to be an account that can remain offline and
  98. /// potentially hold a member's funds, and be a source for staking roles.
  99. pub root_account: AccountId,
  100. /// Member's controller account id. This account is intended to be used by
  101. /// a member to act under their identity in other modules. It will usually be used more
  102. /// online and will have less funds in its balance.
  103. pub controller_account: AccountId,
  104. }
  105. // Contains valid or default user details
  106. struct ValidatedUserInfo {
  107. handle: Vec<u8>,
  108. avatar_uri: Vec<u8>,
  109. about: Vec<u8>,
  110. }
  111. #[derive(Encode, Decode, Debug, PartialEq)]
  112. pub enum EntryMethod<PaidTermId, AccountId> {
  113. Paid(PaidTermId),
  114. Screening(AccountId),
  115. Genesis,
  116. }
  117. /// Must be default constructible because it indirectly is a value in a storage map.
  118. /// ***SHOULD NEVER ACTUALLY GET CALLED, IS REQUIRED TO DUE BAD STORAGE MODEL IN SUBSTRATE***
  119. impl<PaidTermId, AccountId> Default for EntryMethod<PaidTermId, AccountId> {
  120. fn default() -> Self {
  121. Self::Genesis
  122. }
  123. }
  124. #[derive(Encode, Decode, Eq, PartialEq, Default)]
  125. pub struct PaidMembershipTerms<Balance> {
  126. /// Quantity of native tokens which must be provably burned
  127. pub fee: Balance,
  128. /// String of capped length describing human readable conditions which are being agreed upon
  129. pub text: Vec<u8>,
  130. }
  131. decl_storage! {
  132. trait Store for Module<T: Trait> as Membership {
  133. /// MemberId to assign to next member that is added to the registry, and is also the
  134. /// total number of members created. MemberIds start at Zero.
  135. pub NextMemberId get(fn members_created) : T::MemberId;
  136. /// Mapping of member's id to their membership profile
  137. pub MembershipById get(fn membership) : map hasher(blake2_128_concat)
  138. T::MemberId => Membership<T>;
  139. /// Mapping of a root account id to vector of member ids it controls.
  140. pub(crate) MemberIdsByRootAccountId : map hasher(blake2_128_concat)
  141. T::AccountId => Vec<T::MemberId>;
  142. /// Mapping of a controller account id to vector of member ids it controls
  143. pub(crate) MemberIdsByControllerAccountId : map hasher(blake2_128_concat)
  144. T::AccountId => Vec<T::MemberId>;
  145. /// Registered unique handles and their mapping to their owner
  146. pub MemberIdByHandle get(fn handles) : map hasher(blake2_128_concat)
  147. Vec<u8> => T::MemberId;
  148. /// Next paid membership terms id
  149. pub NextPaidMembershipTermsId get(fn next_paid_membership_terms_id) :
  150. T::PaidTermId = T::PaidTermId::from(FIRST_PAID_TERMS_ID);
  151. /// Paid membership terms record
  152. // Remember to add _genesis_phantom_data: std::marker::PhantomData{} to membership
  153. // genesis config if not providing config() or extra_genesis
  154. pub PaidMembershipTermsById get(fn paid_membership_terms_by_id) build(|config: &GenesisConfig<T>| {
  155. // This method only gets called when initializing storage, and is
  156. // compiled as native code. (Will be called when building `raw` chainspec)
  157. // So it can't be relied upon to initialize storage for runtimes updates.
  158. // Initialization for updated runtime is done in run_migration()
  159. let terms = PaidMembershipTerms {
  160. fee: config.default_paid_membership_fee,
  161. text: Vec::default(),
  162. };
  163. vec![(T::PaidTermId::from(DEFAULT_PAID_TERM_ID), terms)]
  164. }) : map hasher(blake2_128_concat) T::PaidTermId => PaidMembershipTerms<BalanceOf<T>>;
  165. /// Active Paid membership terms
  166. pub ActivePaidMembershipTerms get(fn active_paid_membership_terms) :
  167. Vec<T::PaidTermId> = vec![T::PaidTermId::from(DEFAULT_PAID_TERM_ID)];
  168. /// Is the platform is accepting new members or not
  169. pub NewMembershipsAllowed get(fn new_memberships_allowed) : bool = true;
  170. pub ScreeningAuthority get(fn screening_authority) : T::AccountId;
  171. // User Input Validation parameters - do these really need to be state variables
  172. // I don't see a need to adjust these in future?
  173. pub MinHandleLength get(fn min_handle_length) : u32 = DEFAULT_MIN_HANDLE_LENGTH;
  174. pub MaxHandleLength get(fn max_handle_length) : u32 = DEFAULT_MAX_HANDLE_LENGTH;
  175. pub MaxAvatarUriLength get(fn max_avatar_uri_length) : u32 = DEFAULT_MAX_AVATAR_URI_LENGTH;
  176. pub MaxAboutTextLength get(fn max_about_text_length) : u32 = DEFAULT_MAX_ABOUT_TEXT_LENGTH;
  177. }
  178. add_extra_genesis {
  179. config(default_paid_membership_fee): BalanceOf<T>;
  180. config(members) : Vec<genesis::Member<T::MemberId, T::AccountId, T::Moment>>;
  181. build(|config: &GenesisConfig<T>| {
  182. for member in &config.members {
  183. let checked_user_info = <Module<T>>::check_user_registration_info(
  184. Some(member.handle.clone().into_bytes()),
  185. Some(member.avatar_uri.clone().into_bytes()),
  186. Some(member.about.clone().into_bytes())
  187. ).expect("Importing Member Failed");
  188. let member_id = <Module<T>>::insert_member(
  189. &member.root_account,
  190. &member.controller_account,
  191. &checked_user_info,
  192. EntryMethod::Genesis,
  193. T::BlockNumber::from(1),
  194. member.registered_at_time
  195. ).expect("Importing Member Failed");
  196. // ensure imported member id matches assigned id
  197. assert_eq!(member_id, member.member_id, "Import Member Failed: MemberId Incorrect");
  198. }
  199. });
  200. }
  201. }
  202. decl_event! {
  203. pub enum Event<T> where
  204. <T as frame_system::Trait>::AccountId,
  205. <T as Trait>::MemberId,
  206. {
  207. MemberRegistered(MemberId, AccountId),
  208. MemberUpdatedAboutText(MemberId),
  209. MemberUpdatedAvatar(MemberId),
  210. MemberUpdatedHandle(MemberId),
  211. MemberSetRootAccount(MemberId, AccountId),
  212. MemberSetControllerAccount(MemberId, AccountId),
  213. }
  214. }
  215. decl_module! {
  216. pub struct Module<T: Trait> for enum Call where origin: T::Origin {
  217. fn deposit_event() = default;
  218. /// Non-members can buy membership
  219. #[weight = 10_000_000] // TODO: adjust weight
  220. pub fn buy_membership(
  221. origin,
  222. paid_terms_id: T::PaidTermId,
  223. handle: Option<Vec<u8>>,
  224. avatar_uri: Option<Vec<u8>>,
  225. about: Option<Vec<u8>>
  226. ) {
  227. let who = ensure_signed(origin)?;
  228. // make sure we are accepting new memberships
  229. ensure!(Self::new_memberships_allowed(), "new members not allowed");
  230. // ensure paid_terms_id is active
  231. let terms = Self::ensure_active_terms_id(paid_terms_id)?;
  232. // ensure enough free balance to cover terms fees
  233. ensure!(T::Currency::can_slash(&who, terms.fee), "not enough balance to buy membership");
  234. let user_info = Self::check_user_registration_info(handle, avatar_uri, about)?;
  235. let member_id = Self::insert_member(
  236. &who,
  237. &who,
  238. &user_info,
  239. EntryMethod::Paid(paid_terms_id),
  240. <frame_system::Module<T>>::block_number(),
  241. <pallet_timestamp::Module<T>>::now()
  242. )?;
  243. let _ = T::Currency::slash(&who, terms.fee);
  244. Self::deposit_event(RawEvent::MemberRegistered(member_id, who));
  245. }
  246. /// Change member's about text
  247. #[weight = 10_000_000] // TODO: adjust weight
  248. pub fn change_member_about_text(origin, member_id: T::MemberId, text: Vec<u8>) {
  249. let sender = ensure_signed(origin)?;
  250. let membership = Self::ensure_membership(member_id)?;
  251. ensure!(membership.controller_account == sender, "only controller account can update member about text");
  252. Self::_change_member_about_text(member_id, &text)?;
  253. }
  254. /// Change member's avatar
  255. #[weight = 10_000_000] // TODO: adjust weight
  256. pub fn change_member_avatar(origin, member_id: T::MemberId, uri: Vec<u8>) {
  257. let sender = ensure_signed(origin)?;
  258. let membership = Self::ensure_membership(member_id)?;
  259. ensure!(membership.controller_account == sender, "only controller account can update member avatar");
  260. Self::_change_member_avatar(member_id, &uri)?;
  261. }
  262. /// Change member's handle. Will ensure new handle is unique and old one will be available
  263. /// for other members to use.
  264. #[weight = 10_000_000] // TODO: adjust weight
  265. pub fn change_member_handle(origin, member_id: T::MemberId, handle: Vec<u8>) {
  266. let sender = ensure_signed(origin)?;
  267. let membership = Self::ensure_membership(member_id)?;
  268. ensure!(membership.controller_account == sender, "only controller account can update member handle");
  269. Self::_change_member_handle(member_id, handle)?;
  270. }
  271. /// Update member's all or some of handle, avatar and about text.
  272. #[weight = 10_000_000] // TODO: adjust weight
  273. pub fn update_membership(
  274. origin,
  275. member_id: T::MemberId,
  276. handle: Option<Vec<u8>>,
  277. avatar_uri: Option<Vec<u8>>,
  278. about: Option<Vec<u8>>
  279. ) {
  280. let sender = ensure_signed(origin)?;
  281. let membership = Self::ensure_membership(member_id)?;
  282. ensure!(membership.controller_account == sender, "only controller account can update member info");
  283. if let Some(uri) = avatar_uri {
  284. Self::_change_member_avatar(member_id, &uri)?;
  285. }
  286. if let Some(about) = about {
  287. Self::_change_member_about_text(member_id, &about)?;
  288. }
  289. if let Some(handle) = handle {
  290. Self::_change_member_handle(member_id, handle)?;
  291. }
  292. }
  293. #[weight = 10_000_000] // TODO: adjust weight
  294. pub fn set_controller_account(origin, member_id: T::MemberId, new_controller_account: T::AccountId) {
  295. let sender = ensure_signed(origin)?;
  296. let mut membership = Self::ensure_membership(member_id)?;
  297. ensure!(membership.root_account == sender, "only root account can set new controller account");
  298. // only update if new_controller_account is different than current one
  299. if membership.controller_account != new_controller_account {
  300. <MemberIdsByControllerAccountId<T>>::mutate(&membership.controller_account, |ids| {
  301. ids.retain(|id| *id != member_id);
  302. });
  303. <MemberIdsByControllerAccountId<T>>::mutate(&new_controller_account, |ids| {
  304. ids.push(member_id);
  305. });
  306. membership.controller_account = new_controller_account.clone();
  307. <MembershipById<T>>::insert(member_id, membership);
  308. Self::deposit_event(RawEvent::MemberSetControllerAccount(member_id, new_controller_account));
  309. }
  310. }
  311. #[weight = 10_000_000] // TODO: adjust weight
  312. pub fn set_root_account(origin, member_id: T::MemberId, new_root_account: T::AccountId) {
  313. let sender = ensure_signed(origin)?;
  314. let mut membership = Self::ensure_membership(member_id)?;
  315. ensure!(membership.root_account == sender, "only root account can set new root account");
  316. // only update if new root account is different than current one
  317. if membership.root_account != new_root_account {
  318. <MemberIdsByRootAccountId<T>>::mutate(&membership.root_account, |ids| {
  319. ids.retain(|id| *id != member_id);
  320. });
  321. <MemberIdsByRootAccountId<T>>::mutate(&new_root_account, |ids| {
  322. ids.push(member_id);
  323. });
  324. membership.root_account = new_root_account.clone();
  325. <MembershipById<T>>::insert(member_id, membership);
  326. Self::deposit_event(RawEvent::MemberSetRootAccount(member_id, new_root_account));
  327. }
  328. }
  329. #[weight = 10_000_000] // TODO: adjust weight
  330. pub fn add_screened_member(
  331. origin,
  332. new_member_account: T::AccountId,
  333. handle: Option<Vec<u8>>,
  334. avatar_uri: Option<Vec<u8>>,
  335. about: Option<Vec<u8>>
  336. ) {
  337. // ensure sender is screening authority
  338. let sender = ensure_signed(origin)?;
  339. if <ScreeningAuthority<T>>::exists() {
  340. ensure!(sender == Self::screening_authority(), "not screener");
  341. } else {
  342. // no screening authority defined. Cannot accept this request
  343. return Err("no screening authority defined".into());
  344. }
  345. // make sure we are accepting new memberships
  346. ensure!(Self::new_memberships_allowed(), "new members not allowed");
  347. let user_info = Self::check_user_registration_info(handle, avatar_uri, about)?;
  348. let member_id = Self::insert_member(
  349. &new_member_account,
  350. &new_member_account,
  351. &user_info,
  352. EntryMethod::Screening(sender),
  353. <frame_system::Module<T>>::block_number(),
  354. <pallet_timestamp::Module<T>>::now()
  355. )?;
  356. Self::deposit_event(RawEvent::MemberRegistered(member_id, new_member_account));
  357. }
  358. #[weight = 10_000_000] // TODO: adjust weight
  359. pub fn set_screening_authority(origin, authority: T::AccountId) {
  360. ensure_root(origin)?;
  361. <ScreeningAuthority<T>>::put(authority);
  362. }
  363. }
  364. }
  365. /// Reason why a given member id does not have a given account as the controller account.
  366. pub enum ControllerAccountForMemberCheckFailed {
  367. NotMember,
  368. NotControllerAccount,
  369. }
  370. pub enum MemberControllerAccountDidNotSign {
  371. UnsignedOrigin,
  372. MemberIdInvalid,
  373. SignerControllerAccountMismatch,
  374. }
  375. pub enum MemberControllerAccountMismatch {
  376. MemberIdInvalid,
  377. SignerControllerAccountMismatch,
  378. }
  379. pub enum MemberRootAccountMismatch {
  380. MemberIdInvalid,
  381. SignerRootAccountMismatch,
  382. }
  383. impl<T: Trait> Module<T> {
  384. /// Provided that the member_id exists return its membership. Returns error otherwise.
  385. pub fn ensure_membership(id: T::MemberId) -> Result<Membership<T>, &'static str> {
  386. if <MembershipById<T>>::contains_key(&id) {
  387. Ok(Self::membership(&id))
  388. } else {
  389. Err("member profile not found")
  390. }
  391. }
  392. /// Ensure that given member has given account as the controller account
  393. pub fn ensure_is_controller_account_for_member(
  394. member_id: &T::MemberId,
  395. account: &T::AccountId,
  396. ) -> Result<Membership<T>, ControllerAccountForMemberCheckFailed> {
  397. if MembershipById::<T>::contains_key(member_id) {
  398. let membership = MembershipById::<T>::get(member_id);
  399. if membership.controller_account == *account {
  400. Ok(membership)
  401. } else {
  402. Err(ControllerAccountForMemberCheckFailed::NotControllerAccount)
  403. }
  404. } else {
  405. Err(ControllerAccountForMemberCheckFailed::NotMember)
  406. }
  407. }
  408. /// Returns true if account is either a member's root or controller account
  409. pub fn is_member_account(who: &T::AccountId) -> bool {
  410. <MemberIdsByRootAccountId<T>>::contains_key(who)
  411. || <MemberIdsByControllerAccountId<T>>::contains_key(who)
  412. }
  413. fn ensure_active_terms_id(
  414. terms_id: T::PaidTermId,
  415. ) -> Result<PaidMembershipTerms<BalanceOf<T>>, &'static str> {
  416. let active_terms = Self::active_paid_membership_terms();
  417. ensure!(
  418. active_terms.iter().any(|&id| id == terms_id),
  419. "paid terms id not active"
  420. );
  421. if <PaidMembershipTermsById<T>>::contains_key(terms_id) {
  422. Ok(Self::paid_membership_terms_by_id(terms_id))
  423. } else {
  424. Err("paid membership term id does not exist")
  425. }
  426. }
  427. #[allow(clippy::ptr_arg)] // cannot change to the "&[u8]" suggested by clippy
  428. fn ensure_unique_handle(handle: &Vec<u8>) -> DispatchResult {
  429. ensure!(
  430. !<MemberIdByHandle<T>>::contains_key(handle),
  431. "handle already registered"
  432. );
  433. Ok(())
  434. }
  435. fn validate_handle(handle: &[u8]) -> DispatchResult {
  436. ensure!(
  437. handle.len() >= Self::min_handle_length() as usize,
  438. "handle too short"
  439. );
  440. ensure!(
  441. handle.len() <= Self::max_handle_length() as usize,
  442. "handle too long"
  443. );
  444. Ok(())
  445. }
  446. fn validate_text(text: &[u8]) -> Vec<u8> {
  447. let mut text = text.to_owned();
  448. text.truncate(Self::max_about_text_length() as usize);
  449. text
  450. }
  451. fn validate_avatar(uri: &[u8]) -> DispatchResult {
  452. ensure!(
  453. uri.len() <= Self::max_avatar_uri_length() as usize,
  454. "avatar uri too long"
  455. );
  456. Ok(())
  457. }
  458. /// Basic user input validation
  459. fn check_user_registration_info(
  460. handle: Option<Vec<u8>>,
  461. avatar_uri: Option<Vec<u8>>,
  462. about: Option<Vec<u8>>,
  463. ) -> Result<ValidatedUserInfo, &'static str> {
  464. // Handle is required during registration
  465. let handle = handle.ok_or("handle must be provided during registration")?;
  466. Self::validate_handle(&handle)?;
  467. let about = Self::validate_text(&about.unwrap_or_default());
  468. let avatar_uri = avatar_uri.unwrap_or_default();
  469. Self::validate_avatar(&avatar_uri)?;
  470. Ok(ValidatedUserInfo {
  471. handle,
  472. avatar_uri,
  473. about,
  474. })
  475. }
  476. fn insert_member(
  477. root_account: &T::AccountId,
  478. controller_account: &T::AccountId,
  479. user_info: &ValidatedUserInfo,
  480. entry_method: EntryMethod<T::PaidTermId, T::AccountId>,
  481. registered_at_block: T::BlockNumber,
  482. registered_at_time: T::Moment,
  483. ) -> Result<T::MemberId, &'static str> {
  484. Self::ensure_unique_handle(&user_info.handle)?;
  485. let new_member_id = Self::members_created();
  486. let membership: Membership<T> = MembershipObject {
  487. handle: user_info.handle.clone(),
  488. avatar_uri: user_info.avatar_uri.clone(),
  489. about: user_info.about.clone(),
  490. registered_at_block,
  491. registered_at_time,
  492. entry: entry_method,
  493. suspended: false,
  494. subscription: None,
  495. root_account: root_account.clone(),
  496. controller_account: controller_account.clone(),
  497. };
  498. <MemberIdsByRootAccountId<T>>::mutate(root_account, |ids| {
  499. ids.push(new_member_id);
  500. });
  501. <MemberIdsByControllerAccountId<T>>::mutate(controller_account, |ids| {
  502. ids.push(new_member_id);
  503. });
  504. <MembershipById<T>>::insert(new_member_id, membership);
  505. <MemberIdByHandle<T>>::insert(user_info.handle.clone(), new_member_id);
  506. <NextMemberId<T>>::put(new_member_id + One::one());
  507. Ok(new_member_id)
  508. }
  509. fn _change_member_about_text(id: T::MemberId, text: &[u8]) -> DispatchResult {
  510. let mut membership = Self::ensure_membership(id)?;
  511. let text = Self::validate_text(text);
  512. membership.about = text;
  513. Self::deposit_event(RawEvent::MemberUpdatedAboutText(id));
  514. <MembershipById<T>>::insert(id, membership);
  515. Ok(())
  516. }
  517. fn _change_member_avatar(id: T::MemberId, uri: &[u8]) -> DispatchResult {
  518. let mut membership = Self::ensure_membership(id)?;
  519. Self::validate_avatar(uri)?;
  520. membership.avatar_uri = uri.to_owned();
  521. Self::deposit_event(RawEvent::MemberUpdatedAvatar(id));
  522. <MembershipById<T>>::insert(id, membership);
  523. Ok(())
  524. }
  525. fn _change_member_handle(id: T::MemberId, handle: Vec<u8>) -> DispatchResult {
  526. let mut membership = Self::ensure_membership(id)?;
  527. Self::validate_handle(&handle)?;
  528. Self::ensure_unique_handle(&handle)?;
  529. <MemberIdByHandle<T>>::remove(&membership.handle);
  530. <MemberIdByHandle<T>>::insert(handle.clone(), id);
  531. membership.handle = handle;
  532. Self::deposit_event(RawEvent::MemberUpdatedHandle(id));
  533. <MembershipById<T>>::insert(id, membership);
  534. Ok(())
  535. }
  536. pub fn ensure_member_controller_account_signed(
  537. origin: T::Origin,
  538. member_id: &T::MemberId,
  539. ) -> Result<T::AccountId, MemberControllerAccountDidNotSign> {
  540. // Ensure transaction is signed.
  541. let signer_account =
  542. ensure_signed(origin).map_err(|_| MemberControllerAccountDidNotSign::UnsignedOrigin)?;
  543. // Ensure member exists
  544. let membership = Self::ensure_membership(*member_id)
  545. .map_err(|_| MemberControllerAccountDidNotSign::MemberIdInvalid)?;
  546. ensure!(
  547. membership.controller_account == signer_account,
  548. MemberControllerAccountDidNotSign::SignerControllerAccountMismatch
  549. );
  550. Ok(signer_account)
  551. }
  552. pub fn ensure_member_controller_account(
  553. signer_account: &T::AccountId,
  554. member_id: &T::MemberId,
  555. ) -> Result<(), MemberControllerAccountMismatch> {
  556. // Ensure member exists
  557. let membership = Self::ensure_membership(*member_id)
  558. .map_err(|_| MemberControllerAccountMismatch::MemberIdInvalid)?;
  559. ensure!(
  560. membership.controller_account == *signer_account,
  561. MemberControllerAccountMismatch::SignerControllerAccountMismatch
  562. );
  563. Ok(())
  564. }
  565. pub fn ensure_member_root_account(
  566. signer_account: &T::AccountId,
  567. member_id: &T::MemberId,
  568. ) -> Result<(), MemberRootAccountMismatch> {
  569. // Ensure member exists
  570. let membership = Self::ensure_membership(*member_id)
  571. .map_err(|_| MemberRootAccountMismatch::MemberIdInvalid)?;
  572. ensure!(
  573. membership.root_account == *signer_account,
  574. MemberRootAccountMismatch::SignerRootAccountMismatch
  575. );
  576. Ok(())
  577. }
  578. }