mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-22 21:58:00 +00:00
643795919f
* WIP DispatchError generic param * main crate now compiling with new E generic param for DispatchError * Remove E param from RpcClient since it doesn't really need it * Point to generated DispatchError in codegen so no need for additional param there * More error hunting; cargo check --all-targets passes now * Use our own RuntimeVersion struct (for now) to avoid error decoding into sp_version::RuntimeVersion * cargo fmt * fix docs and expose private documented thing that I think should be pub * move error info to compile time so that we can make DispatchError a little nicer to work with * cargo fmt * clippy * Rework error handling to remove <E> param in most cases * fix Error doc ambiguity (hopefully) * doc tweaks * docs: remove dismbiguation thing that isn't needed now * One more Error<E> that can be a BasicError * rewrite pallet errors thing into normal loops to tidy * tidy errors codegen a little * tidy examples/custom_type_derives.rs a little * cargo fmt * silcnce clippy in example
266 lines
7.2 KiB
Rust
266 lines
7.2 KiB
Rust
// Copyright 2019-2022 Parity Technologies (UK) Ltd.
|
|
// This file is part of subxt.
|
|
//
|
|
// subxt is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// subxt is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with subxt. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
use crate::{
|
|
node_runtime::{
|
|
runtime_types::pallet_staking::{
|
|
RewardDestination,
|
|
ValidatorPrefs,
|
|
},
|
|
staking,
|
|
DispatchError,
|
|
},
|
|
pair_signer,
|
|
test_context,
|
|
};
|
|
use assert_matches::assert_matches;
|
|
use sp_core::{
|
|
sr25519,
|
|
Pair,
|
|
};
|
|
use sp_keyring::AccountKeyring;
|
|
use subxt::{
|
|
Error,
|
|
Signer,
|
|
};
|
|
|
|
/// Helper function to generate a crypto pair from seed
|
|
fn get_from_seed(seed: &str) -> sr25519::Pair {
|
|
sr25519::Pair::from_string(&format!("//{}", seed), None)
|
|
.expect("static values are valid; qed")
|
|
}
|
|
|
|
fn default_validator_prefs() -> ValidatorPrefs {
|
|
ValidatorPrefs {
|
|
commission: sp_runtime::Perbill::default(),
|
|
blocked: false,
|
|
}
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn validate_with_controller_account() {
|
|
let alice = pair_signer(AccountKeyring::Alice.pair());
|
|
let cxt = test_context().await;
|
|
cxt.api
|
|
.tx()
|
|
.staking()
|
|
.validate(default_validator_prefs())
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await
|
|
.unwrap()
|
|
.wait_for_finalized_success()
|
|
.await
|
|
.expect("should be successful");
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn validate_not_possible_for_stash_account() -> Result<(), Error<DispatchError>> {
|
|
let alice_stash = pair_signer(get_from_seed("Alice//stash"));
|
|
let cxt = test_context().await;
|
|
let announce_validator = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.validate(default_validator_prefs())
|
|
.sign_and_submit_then_watch(&alice_stash)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await;
|
|
assert_matches!(announce_validator, Err(Error::Runtime(err)) => {
|
|
let details = err.inner().details().unwrap();
|
|
assert_eq!(details.pallet, "Staking");
|
|
assert_eq!(details.error, "NotController");
|
|
});
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn nominate_with_controller_account() {
|
|
let alice = pair_signer(AccountKeyring::Alice.pair());
|
|
let bob = pair_signer(AccountKeyring::Bob.pair());
|
|
let cxt = test_context().await;
|
|
|
|
cxt.api
|
|
.tx()
|
|
.staking()
|
|
.nominate(vec![bob.account_id().clone().into()])
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await
|
|
.unwrap()
|
|
.wait_for_finalized_success()
|
|
.await
|
|
.expect("should be successful");
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn nominate_not_possible_for_stash_account() -> Result<(), Error<DispatchError>> {
|
|
let alice_stash = pair_signer(get_from_seed("Alice//stash"));
|
|
let bob = pair_signer(AccountKeyring::Bob.pair());
|
|
let cxt = test_context().await;
|
|
|
|
let nomination = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.nominate(vec![bob.account_id().clone().into()])
|
|
.sign_and_submit_then_watch(&alice_stash)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await;
|
|
|
|
assert_matches!(nomination, Err(Error::Runtime(err)) => {
|
|
let details = err.inner().details().unwrap();
|
|
assert_eq!(details.pallet, "Staking");
|
|
assert_eq!(details.error, "NotController");
|
|
});
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn chill_works_for_controller_only() -> Result<(), Error<DispatchError>> {
|
|
let alice_stash = pair_signer(get_from_seed("Alice//stash"));
|
|
let bob_stash = pair_signer(get_from_seed("Bob//stash"));
|
|
let alice = pair_signer(AccountKeyring::Alice.pair());
|
|
let cxt = test_context().await;
|
|
|
|
// this will fail the second time, which is why this is one test, not two
|
|
cxt.api
|
|
.tx()
|
|
.staking()
|
|
.nominate(vec![bob_stash.account_id().clone().into()])
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await?;
|
|
|
|
let ledger = cxt
|
|
.api
|
|
.storage()
|
|
.staking()
|
|
.ledger(alice.account_id().clone(), None)
|
|
.await?
|
|
.unwrap();
|
|
assert_eq!(alice_stash.account_id(), &ledger.stash);
|
|
|
|
let chill = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.chill()
|
|
.sign_and_submit_then_watch(&alice_stash)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await;
|
|
|
|
assert_matches!(chill, Err(Error::Runtime(err)) => {
|
|
let details = err.inner().details().unwrap();
|
|
assert_eq!(details.pallet, "Staking");
|
|
assert_eq!(details.error, "NotController");
|
|
});
|
|
|
|
let is_chilled = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.chill()
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await?
|
|
.has_event::<staking::events::Chilled>()?;
|
|
assert!(is_chilled);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn tx_bond() -> Result<(), Error<DispatchError>> {
|
|
let alice = pair_signer(AccountKeyring::Alice.pair());
|
|
let cxt = test_context().await;
|
|
|
|
let bond = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.bond(
|
|
AccountKeyring::Bob.to_account_id().into(),
|
|
100_000_000_000_000,
|
|
RewardDestination::Stash,
|
|
)
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await;
|
|
|
|
assert!(bond.is_ok());
|
|
|
|
let bond_again = cxt
|
|
.api
|
|
.tx()
|
|
.staking()
|
|
.bond(
|
|
AccountKeyring::Bob.to_account_id().into(),
|
|
100_000_000_000_000,
|
|
RewardDestination::Stash,
|
|
)
|
|
.sign_and_submit_then_watch(&alice)
|
|
.await?
|
|
.wait_for_finalized_success()
|
|
.await;
|
|
|
|
assert_matches!(bond_again, Err(Error::Runtime(err)) => {
|
|
let details = err.inner().details().unwrap();
|
|
assert_eq!(details.pallet, "Staking");
|
|
assert_eq!(details.error, "AlreadyBonded");
|
|
});
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn storage_history_depth() -> Result<(), Error<DispatchError>> {
|
|
let cxt = test_context().await;
|
|
let history_depth = cxt.api.storage().staking().history_depth(None).await?;
|
|
assert_eq!(history_depth, 84);
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn storage_current_era() -> Result<(), Error<DispatchError>> {
|
|
let cxt = test_context().await;
|
|
let _current_era = cxt
|
|
.api
|
|
.storage()
|
|
.staking()
|
|
.current_era(None)
|
|
.await?
|
|
.expect("current era always exists");
|
|
Ok(())
|
|
}
|
|
|
|
#[async_std::test]
|
|
async fn storage_era_reward_points() -> Result<(), Error<DispatchError>> {
|
|
let cxt = test_context().await;
|
|
let current_era_result = cxt
|
|
.api
|
|
.storage()
|
|
.staking()
|
|
.eras_reward_points(0, None)
|
|
.await;
|
|
assert!(current_era_result.is_ok());
|
|
|
|
Ok(())
|
|
}
|