Files
pezkuwi-subxt/cumulus/parachains/pallets/collective-content/src/tests.rs
T
Muharem 9e403629d5 The Ambassador Program (#1308)
The Ambassador Program on Polkadot Collectives Parachain

The Polkadot Ambassador Program has existed for a while; more
information can be found
[here](https://wiki.polkadot.network/docs/ambassadors).
In this PR, the program is being brought on chain.

### On Chain Structure
The on-chain program consists of nine ranks, divided into four
categories ([full
list](https://github.com/paritytech/cumulus/blob/c238fb26b75569a11abb57437fd14acd26e05f18/parachains/runtimes/collectives/collectives-polkadot/src/ambassador/mod.rs#L52)):
- Ambassadors (1-2 tiers)
- Senior Ambassadors (3-4 tiers)
- Head Ambassadors (5-7 tiers)
- Master Ambassadors (8-9 tiers)

Each rank has a corresponding `Origin` (e.g., `HeadAmbassadorsTier5` -
[full
list](https://github.com/paritytech/cumulus/blob/c238fb26b75569a11abb57437fd14acd26e05f18/parachains/runtimes/collectives/collectives-polkadot/src/ambassador/origins.rs#L35)),
which represents the collective voice of members of that rank and above.

### Referendum

The `AmbassadorReferenda` instance of [referenda
pallet](https://docs.rs/pallet-referenda/latest/pallet_referenda/)
consists of [nine
tracks](https://github.com/paritytech/cumulus/blob/c238fb26b75569a11abb57437fd14acd26e05f18/parachains/runtimes/collectives/collectives-polkadot/src/ambassador/tracks.rs#L51),
each corresponding to an `Origin`. A referendum taken on `senior
ambassador tier 4` track invites all members from rank 4 or above to
vote and commands `SeniorAmbassadors` `Origin`. Every member gets one
vote plus an additional vote for every excess rank. The referendum
proposal can be submitted by any member of a senior rank or above.

### Membership Management

Initial members will be brought on chain via migration, with subsequent
member management handled through the `AmbassadorCollective` instance of
[ranked collective
pallet](https://docs.rs/pallet-ranked-collective/latest/pallet_ranked_collective/).
Both `Root` and `FellowshipAdmin` `Origins`, commanded via public
Polkadot referendum, can promote or demote members to and from any rank.
Members themselves also have the power to promote or demote via its
referendum, with a senior member vote by the rank two above the new /
current rank - [full
configuration](https://github.com/paritytech/cumulus/blob/9ab6aa47063d7e8b67ddc10d9c136037f99c03a3/parachains/runtimes/collectives/collectives-polkadot/src/ambassador/mod.rs#L67).

### Content Management

The program's on-chain content is managed via the collectives content
pallet, allowing for setting its charter and making announcements. The
voice of head ambassadors have the authority to set the charter, while
announcements can be made by any senior rank member or through a
referendum among all members.

### Additional Functionality

The `AmbassadorCore` instance of [core fellowship
pallet](https://docs.rs/pallet-core-fellowship/latest/pallet_core_fellowship/)
decorates the ranked collectives pallet with features like salary
determination, activity/passivity registration, and the handling of
promotion and demotion periods. While the usage of this pallet is
optional in the first version, future updates will make it the exclusive
method for induction/promotion.

Periodic salaries in USDt, payable on Asset Hub, are introduced through
the [salary
pallet](https://docs.rs/pallet-salary/latest/pallet_salary/). This
requires induction into the ambassador core pallet.

Please for more information on the pallets' functionality refer to their
documentations.

### Next Steps:
- Migrate to seed the program members
- Mint ambassador NFT badges on Asset Hub when promoting
- Treasury pallet instance for the Ambassador Program

---------

Co-authored-by: Kian Paimani <5588131+kianenigma@users.noreply.github.com>
Co-authored-by: joe petrowski <25483142+joepetrowski@users.noreply.github.com>
2023-09-20 21:55:56 +02:00

205 lines
7.0 KiB
Rust

// Copyright (C) 2023 Parity Technologies (UK) Ltd.
// SPDX-License-Identifier: Apache-2.0
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//! Tests.
use super::{mock::*, *};
use frame_support::{assert_noop, assert_ok, error::BadOrigin, pallet_prelude::Pays};
/// returns CID hash of 68 bytes of given `i`.
fn create_cid(i: u8) -> OpaqueCid {
let cid: OpaqueCid = [i; 68].to_vec().try_into().unwrap();
cid
}
#[test]
fn set_charter_works() {
new_test_ext().execute_with(|| {
// wrong origin.
let origin = RuntimeOrigin::signed(SomeAccount::get());
let cid = create_cid(1);
assert_noop!(CollectiveContent::set_charter(origin, cid), BadOrigin);
// success.
let origin = RuntimeOrigin::signed(CharterManager::get());
let cid = create_cid(2);
assert_ok!(CollectiveContent::set_charter(origin, cid.clone()));
assert_eq!(Charter::<Test, _>::get(), Some(cid.clone()));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::NewCharterSet { cid }));
// reset. success.
let origin = RuntimeOrigin::signed(CharterManager::get());
let cid = create_cid(3);
assert_ok!(CollectiveContent::set_charter(origin, cid.clone()));
assert_eq!(Charter::<Test, _>::get(), Some(cid.clone()));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::NewCharterSet { cid }));
});
}
#[test]
fn announce_works() {
new_test_ext().execute_with(|| {
let now = frame_system::Pallet::<Test>::block_number();
// wrong origin.
let origin = RuntimeOrigin::signed(SomeAccount::get());
let cid = create_cid(1);
assert_noop!(CollectiveContent::announce(origin, cid, None), BadOrigin);
// success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(2);
let maybe_expire_at = None;
assert_ok!(CollectiveContent::announce(origin, cid.clone(), maybe_expire_at));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementAnnounced {
cid,
expire_at: now.saturating_add(AnnouncementLifetime::get()),
}));
// one more. success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(3);
let maybe_expire_at = None;
assert_ok!(CollectiveContent::announce(origin, cid.clone(), maybe_expire_at));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementAnnounced {
cid,
expire_at: now.saturating_add(AnnouncementLifetime::get()),
}));
// one more with expire. success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(4);
let maybe_expire_at = DispatchTime::<_>::After(10);
assert_ok!(CollectiveContent::announce(origin, cid.clone(), Some(maybe_expire_at)));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementAnnounced {
cid,
expire_at: maybe_expire_at.evaluate(now),
}));
// one more with later expire. success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(5);
let maybe_expire_at = DispatchTime::<_>::At(now + 20);
assert_ok!(CollectiveContent::announce(origin, cid.clone(), Some(maybe_expire_at)));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementAnnounced {
cid,
expire_at: maybe_expire_at.evaluate(now),
}));
// one more with earlier expire. success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(6);
let maybe_expire_at = DispatchTime::<_>::At(now + 5);
assert_ok!(CollectiveContent::announce(origin, cid.clone(), Some(maybe_expire_at)));
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementAnnounced {
cid,
expire_at: maybe_expire_at.evaluate(now),
}));
// one more with earlier expire. success.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(7);
let maybe_expire_at = DispatchTime::<_>::At(now + 5);
assert_eq!(<Announcements<Test, _>>::count(), MaxAnnouncements::get());
assert_noop!(
CollectiveContent::announce(origin, cid.clone(), Some(maybe_expire_at)),
Error::<Test>::TooManyAnnouncements
);
});
}
#[test]
fn remove_announcement_works() {
new_test_ext().execute_with(|| {
// wrong origin.
let origin = RuntimeOrigin::signed(CharterManager::get());
let cid = create_cid(8);
assert_noop!(
CollectiveContent::remove_announcement(origin, cid),
Error::<Test>::MissingAnnouncement
);
// missing announcement.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(9);
assert_noop!(
CollectiveContent::remove_announcement(origin, cid),
Error::<Test>::MissingAnnouncement
);
// wrong origin. announcement not yet expired.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(10);
assert_ok!(CollectiveContent::announce(origin.clone(), cid.clone(), None));
assert!(<Announcements<Test>>::contains_key(cid.clone()));
let origin = RuntimeOrigin::signed(SomeAccount::get());
assert_noop!(CollectiveContent::remove_announcement(origin, cid.clone()), BadOrigin);
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
assert_ok!(CollectiveContent::remove_announcement(origin, cid));
// success.
// remove first announcement and assert.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(11);
assert_ok!(CollectiveContent::announce(origin.clone(), cid.clone(), None));
assert!(<Announcements<Test>>::contains_key(cid.clone()));
let info = CollectiveContent::remove_announcement(origin.clone(), cid.clone()).unwrap();
assert_eq!(info.pays_fee, Pays::Yes);
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementRemoved {
cid: cid.clone(),
}));
assert_noop!(
CollectiveContent::remove_announcement(origin, cid.clone()),
Error::<Test>::MissingAnnouncement
);
assert!(!<Announcements<Test>>::contains_key(cid));
// remove expired announcement and assert.
let origin = RuntimeOrigin::signed(AnnouncementManager::get());
let cid = create_cid(12);
assert_ok!(CollectiveContent::announce(
origin.clone(),
cid.clone(),
Some(DispatchTime::<_>::At(10))
));
assert!(<Announcements<Test>>::contains_key(cid.clone()));
System::set_block_number(11);
let origin = RuntimeOrigin::signed(SomeAccount::get());
let info = CollectiveContent::remove_announcement(origin.clone(), cid.clone()).unwrap();
assert_eq!(info.pays_fee, Pays::No);
System::assert_last_event(RuntimeEvent::CollectiveContent(Event::AnnouncementRemoved {
cid: cid.clone(),
}));
assert_noop!(
CollectiveContent::remove_announcement(origin, cid),
Error::<Test>::MissingAnnouncement
);
});
}