mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-28 08:37:56 +00:00
e48f0e3b1d
* WIP API changes * debug impls * Get main crate compiling with first round of changes * Some tidy up * Add WithExtrinsicParams, and have SubstrateConfig + PolkadotConfig, not DefaultConfig * move transaction into extrinsic folder * Add runtime updates back to OnlineClient * rework to be 'client first' to fit better with storage + events * add support for events to Client * tidy dupe trait bound * Wire storage into client, but need to remove static reliance * various tidy up and start stripping codegen to remove bits we dont need now * First pass updating calls and constants codegen * WIP storage client updates * First pass migrated runtime storage over to new format * pass over codegen to generate StorageAddresses and throw other stuff out * don't need a Call trait any more * shuffle things around a bit * Various proc_macro fixes to get 'cargo check' working * organise what's exposed from subxt * Get first example working; balance_transfer_with_params * get balance_transfer example compiling * get concurrent_storage_requests.rs example compiling * get fetch_all_accounts example compiling * get a bunch more of the examples compiling * almost get final example working; type mismatch to look into * wee tweaks * move StorageAddress to separate file * pass Defaultable/Iterable info to StorageAddress in codegen * fix storage validation ne, and partial run through example code * Remove static iteration and strip a generic param from everything * fix doc tests in subxt crate * update test utils and start fixing frame tests * fix frame staking tests * fix the rest of the test compile issues, Borrow on storage values * cargo fmt * remove extra logging during tests * Appease clippy and no more need for into_iter on events * cargo fmt * fix dryRun tests by waiting for blocks * wait for blocks instead of sleeping or other test hacks * cargo fmt * Fix doc links * Traitify StorageAddress * remove out-of-date doc comments * optimise decoding storage a little * cleanup tx stuff, trait for TxPayload, remove Err type param and decode at runtime * clippy fixes * fix doc links * fix doc example * constant address trait for consistency * fix a typo and remove EncodeWithMetadata stuff * Put EventDetails behind a proper interface and allow decoding into top level event, too * fix docs * tweak StorageAddress docs * re-export StorageAddress at root for consistency * fix clippy things * Add support for dynamic values * fix double encoding of storage map key after refactor * clippy fix * Fixes and add a dynamic usage example (needs new scale_value release) * bump scale_value version * cargo fmt * Tweak event bits * cargo fmt * Add a test and bump scale-value to 0.4.0 to support this * remove unnecessary vec from dynamic example * Various typo/grammar fixes Co-authored-by: Alexandru Vasile <60601340+lexnv@users.noreply.github.com> * Address PR nits * Undo accidental rename in changelog * Small PR nits/tidyups * fix tests; codegen change against latest substrate * tweak storage address util names * move error decoding to DecodeError and expose * impl some basic traits on the extrinsic param builder Co-authored-by: Alexandru Vasile <60601340+lexnv@users.noreply.github.com>
221 lines
6.9 KiB
Rust
221 lines
6.9 KiB
Rust
// Copyright 2019-2022 Parity Technologies (UK) Ltd.
|
|
// This file is dual-licensed as Apache-2.0 or GPL-3.0.
|
|
// see LICENSE for license details.
|
|
|
|
use crate::{
|
|
node_runtime::{
|
|
self,
|
|
balances,
|
|
system,
|
|
},
|
|
pair_signer,
|
|
test_context,
|
|
utils::wait_for_blocks,
|
|
};
|
|
use futures::StreamExt;
|
|
use sp_keyring::AccountKeyring;
|
|
|
|
// Check that we can subscribe to non-finalized block events.
|
|
#[tokio::test]
|
|
async fn non_finalized_block_subscription() -> Result<(), subxt::Error> {
|
|
let ctx = test_context().await;
|
|
let api = ctx.client();
|
|
|
|
let mut event_sub = api.events().subscribe().await?;
|
|
|
|
// Wait for the next set of events, and check that the
|
|
// associated block hash is not finalized yet.
|
|
let events = event_sub.next().await.unwrap()?;
|
|
let event_block_hash = events.block_hash();
|
|
let current_block_hash = api.rpc().block_hash(None).await?.unwrap();
|
|
|
|
assert_eq!(event_block_hash, current_block_hash);
|
|
Ok(())
|
|
}
|
|
|
|
// Check that we can subscribe to finalized block events.
|
|
#[tokio::test]
|
|
async fn finalized_block_subscription() -> Result<(), subxt::Error> {
|
|
let ctx = test_context().await;
|
|
let api = ctx.client();
|
|
|
|
let mut event_sub = api.events().subscribe_finalized().await?;
|
|
|
|
// Wait for the next set of events, and check that the
|
|
// associated block hash is the one we just finalized.
|
|
// (this can be a bit slow as we have to wait for finalization)
|
|
let events = event_sub.next().await.unwrap()?;
|
|
let event_block_hash = events.block_hash();
|
|
let finalized_hash = api.rpc().finalized_head().await?;
|
|
|
|
assert_eq!(event_block_hash, finalized_hash);
|
|
Ok(())
|
|
}
|
|
|
|
// Check that our subscription actually keeps producing events for
|
|
// a few blocks.
|
|
#[tokio::test]
|
|
async fn subscription_produces_events_each_block() -> Result<(), subxt::Error> {
|
|
let ctx = test_context().await;
|
|
let api = ctx.client();
|
|
|
|
wait_for_blocks(&api).await;
|
|
|
|
let mut event_sub = api.events().subscribe().await?;
|
|
|
|
for i in 0..3 {
|
|
let events = event_sub
|
|
.next()
|
|
.await
|
|
.expect("events expected each block")?;
|
|
|
|
let success_event = events
|
|
.find_first::<system::events::ExtrinsicSuccess>()
|
|
.expect("decode error");
|
|
|
|
if success_event.is_none() {
|
|
let n = events.len();
|
|
panic!("Expected an extrinsic success event on iteration {i} (saw {n} other events)")
|
|
}
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Check that our subscription receives events, and we can filter them based on
|
|
// it's Stream impl, and ultimately see the event we expect.
|
|
#[tokio::test]
|
|
async fn balance_transfer_subscription() -> Result<(), subxt::Error> {
|
|
let ctx = test_context().await;
|
|
let api = ctx.client();
|
|
|
|
// Subscribe to balance transfer events, ignoring all else.
|
|
let event_sub = api
|
|
.events()
|
|
.subscribe()
|
|
.await?
|
|
.filter_events::<(balances::events::Transfer,)>();
|
|
|
|
// Calling `.next()` on the above borrows it, and the `filter_map`
|
|
// means it's no longer `Unpin`, so we pin it on the stack:
|
|
futures::pin_mut!(event_sub);
|
|
|
|
// Make a transfer:
|
|
let alice = pair_signer(AccountKeyring::Alice.pair());
|
|
let bob = AccountKeyring::Bob.to_account_id();
|
|
let transfer_tx = node_runtime::tx()
|
|
.balances()
|
|
.transfer(bob.clone().into(), 10_000);
|
|
|
|
api.tx()
|
|
.sign_and_submit_then_watch_default(&transfer_tx, &alice)
|
|
.await?;
|
|
|
|
// Wait for the next balance transfer event in our subscription stream
|
|
// and check that it lines up:
|
|
let event = event_sub.next().await.unwrap().unwrap().event;
|
|
assert_eq!(
|
|
event,
|
|
balances::events::Transfer {
|
|
from: alice.account_id().clone(),
|
|
to: bob.clone(),
|
|
amount: 10_000
|
|
}
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn missing_block_headers_will_be_filled_in() -> Result<(), subxt::Error> {
|
|
let ctx = test_context().await;
|
|
let api = ctx.client();
|
|
|
|
// This function is not publically available to use, but contains
|
|
// the key logic for filling in missing blocks, so we want to test it.
|
|
// This is used in `subscribe_finalized` to ensure no block headers are
|
|
// missed.
|
|
use subxt::events::subscribe_to_block_headers_filling_in_gaps;
|
|
|
|
// Manually subscribe to the next 6 finalized block headers, but deliberately
|
|
// filter out some in the middle so we get back b _ _ b _ b. This guarantees
|
|
// that there will be some gaps, even if there aren't any from the subscription.
|
|
let some_finalized_blocks = api
|
|
.rpc()
|
|
.subscribe_finalized_blocks()
|
|
.await?
|
|
.enumerate()
|
|
.take(6)
|
|
.filter(|(n, _)| {
|
|
let n = *n;
|
|
async move { n == 0 || n == 3 || n == 5 }
|
|
})
|
|
.map(|(_, h)| h);
|
|
|
|
// This should spot any gaps in the middle and fill them back in.
|
|
let all_finalized_blocks = subscribe_to_block_headers_filling_in_gaps(
|
|
ctx.client(),
|
|
None,
|
|
some_finalized_blocks,
|
|
);
|
|
futures::pin_mut!(all_finalized_blocks);
|
|
|
|
// Iterate the block headers, making sure we get them all in order.
|
|
let mut last_block_number = None;
|
|
while let Some(header) = all_finalized_blocks.next().await {
|
|
let header = header?;
|
|
|
|
use sp_runtime::traits::Header;
|
|
let block_number: u128 = (*header.number()).into();
|
|
|
|
if let Some(last) = last_block_number {
|
|
assert_eq!(last + 1, block_number);
|
|
}
|
|
last_block_number = Some(block_number);
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// This is just a compile-time check that we can subscribe to events in
|
|
// a context that requires the event subscription/filtering to be Send-able.
|
|
// We test a typical use of EventSubscription and FilterEvents. We don't need
|
|
// to run this code; just check that it compiles.
|
|
#[allow(unused)]
|
|
async fn check_events_are_sendable() {
|
|
// check that EventSubscription can be used across await points.
|
|
tokio::task::spawn(async {
|
|
let ctx = test_context().await;
|
|
|
|
let mut event_sub = ctx.client().events().subscribe().await?;
|
|
|
|
while let Some(ev) = event_sub.next().await {
|
|
// if `event_sub` doesn't implement Send, we can't hold
|
|
// it across an await point inside of a tokio::spawn, which
|
|
// requires Send. This will lead to a compile error.
|
|
}
|
|
|
|
Ok::<_, subxt::Error>(())
|
|
});
|
|
|
|
// Check that FilterEvents can be used across await points.
|
|
tokio::task::spawn(async {
|
|
let ctx = test_context().await;
|
|
|
|
let mut event_sub = ctx
|
|
.client()
|
|
.events()
|
|
.subscribe()
|
|
.await?
|
|
.filter_events::<(balances::events::Transfer,)>();
|
|
|
|
while let Some(ev) = event_sub.next().await {
|
|
// if `event_sub` doesn't implement Send, we can't hold
|
|
// it across an await point inside of a tokio::spawn, which
|
|
// requires Send; This will lead to a compile error.
|
|
}
|
|
|
|
Ok::<_, subxt::Error>(())
|
|
});
|
|
}
|