mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-22 07:58:02 +00:00
ed25a3ac26
* Remove codec::Encode and codec::Decode from generated APIs by default * clippy fixes * clippy * More fixes, and CompactAs only if Encode/Decode * revert println in example * fix lightclient test * fix docs * Fix another rust doc comment * Fix failing storage test * Remove now-unnecessary test * clippy * clippy * Remove pointless clone
226 lines
7.7 KiB
Rust
226 lines
7.7 KiB
Rust
// Copyright 2019-2025 Parity Technologies (UK) Ltd.
|
|
// This file is dual-licensed as Apache-2.0 or GPL-3.0.
|
|
// see LICENSE for license details.
|
|
|
|
//! # Light Client Initialization and Testing
|
|
//!
|
|
//! The initialization process of the light client can be slow, especially when
|
|
//! it needs to synchronize with a local running node for each individual
|
|
//! #[tokio::test] in subxt. To optimize this process, a subset of tests is
|
|
//! exposed to ensure the light client remains functional over time. Currently,
|
|
//! these tests are placed under an unstable feature flag.
|
|
//!
|
|
//! Ideally, we would place the light client initialization in a shared static
|
|
//! using `OnceCell`. However, during the initialization, tokio::spawn is used
|
|
//! to multiplex between subxt requests and node responses. The #[tokio::test]
|
|
//! macro internally creates a new Runtime for each individual test. This means
|
|
//! that only the first test, which spawns the substrate binary and synchronizes
|
|
//! the light client, would have access to the background task. The cleanup process
|
|
//! would destroy the spawned background task, preventing subsequent tests from
|
|
//! accessing it.
|
|
//!
|
|
//! To address this issue, we can consider creating a slim proc-macro that
|
|
//! transforms the #[tokio::test] into a plain #[test] and runs all the tests
|
|
//! on a shared tokio runtime. This approach would allow multiple tests to share
|
|
//! the same background task, ensuring consistent access to the light client.
|
|
//!
|
|
//! For more context see: https://github.com/tokio-rs/tokio/issues/2374.
|
|
//!
|
|
|
|
use crate::utils::node_runtime;
|
|
use codec::Compact;
|
|
use std::sync::Arc;
|
|
use subxt::backend::chain_head::ChainHeadBackend;
|
|
use subxt::backend::rpc::RpcClient;
|
|
use subxt::{client::OnlineClient, config::PolkadotConfig, lightclient::LightClient};
|
|
use subxt_metadata::Metadata;
|
|
|
|
type Client = OnlineClient<PolkadotConfig>;
|
|
|
|
/// The Polkadot chainspec.
|
|
const POLKADOT_SPEC: &str = include_str!("../../../../artifacts/demo_chain_specs/polkadot.json");
|
|
|
|
// Check that we can subscribe to non-finalized blocks.
|
|
async fn non_finalized_headers_subscription(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
|
|
tracing::trace!("Check non_finalized_headers_subscription");
|
|
let mut sub = api.blocks().subscribe_best().await?;
|
|
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("First block took {:?}", now.elapsed());
|
|
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("Second block took {:?}", now.elapsed());
|
|
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("Third block took {:?}", now.elapsed());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Check that we can subscribe to finalized blocks.
|
|
async fn finalized_headers_subscription(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
|
|
tracing::trace!("Check finalized_headers_subscription");
|
|
|
|
let mut sub = api.blocks().subscribe_finalized().await?;
|
|
let header = sub.next().await.unwrap()?;
|
|
tracing::trace!("First block took {:?}", now.elapsed());
|
|
|
|
let finalized_hash = api
|
|
.backend()
|
|
.latest_finalized_block_ref()
|
|
.await
|
|
.unwrap()
|
|
.hash();
|
|
|
|
tracing::trace!(
|
|
"Finalized hash: {:?} took {:?}",
|
|
finalized_hash,
|
|
now.elapsed()
|
|
);
|
|
|
|
assert_eq!(header.hash(), finalized_hash);
|
|
tracing::trace!("Check progress {:?}", now.elapsed());
|
|
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("Second block took {:?}", now.elapsed());
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("Third block took {:?}", now.elapsed());
|
|
let _block = sub.next().await.unwrap()?;
|
|
tracing::trace!("Fourth block took {:?}\n", now.elapsed());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Check that we can subscribe to non-finalized blocks.
|
|
async fn runtime_api_call(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
tracing::trace!("Check runtime_api_call");
|
|
|
|
let mut sub = api.blocks().subscribe_best().await?;
|
|
|
|
let block = sub.next().await.unwrap()?;
|
|
tracing::trace!("First block took {:?}", now.elapsed());
|
|
let rt = block.runtime_api().await?;
|
|
|
|
// get metadata via state_call. if it decodes ok, it's probably all good.
|
|
let result_bytes = rt.call_raw("Metadata_metadata", None).await?;
|
|
let (_, _meta): (Compact<u32>, Metadata) = codec::Decode::decode(&mut &*result_bytes)?;
|
|
|
|
tracing::trace!("Made runtime API call in {:?}\n", now.elapsed());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Lookup for the `Timestamp::now` plain storage entry.
|
|
async fn storage_plain_lookup(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
tracing::trace!("Check storage_plain_lookup");
|
|
|
|
let addr = node_runtime::storage().timestamp().now();
|
|
let entry = api
|
|
.storage()
|
|
.at_latest()
|
|
.await?
|
|
.fetch_or_default(&addr)
|
|
.await?;
|
|
|
|
tracing::trace!("Storage lookup took {:?}\n", now.elapsed());
|
|
|
|
assert!(entry > 0);
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Make a dynamic constant query for `System::BlockLength`.
|
|
async fn dynamic_constant_query(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
tracing::trace!("Check dynamic_constant_query");
|
|
|
|
let constant_query = subxt::dynamic::constant("System", "BlockLength");
|
|
let _value = api.constants().at(&constant_query)?;
|
|
|
|
tracing::trace!("Dynamic constant query took {:?}\n", now.elapsed());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
// Fetch a few all events from the latest block and decode them dynamically.
|
|
async fn dynamic_events(api: &Client) -> Result<(), subxt::Error> {
|
|
let now = std::time::Instant::now();
|
|
tracing::trace!("Check dynamic_events");
|
|
|
|
let events = api.events().at_latest().await?;
|
|
|
|
for event in events.iter() {
|
|
let _event = event?;
|
|
|
|
tracing::trace!("Event decoding took {:?}", now.elapsed());
|
|
}
|
|
|
|
tracing::trace!("Dynamic events took {:?}\n", now.elapsed());
|
|
|
|
Ok(())
|
|
}
|
|
|
|
async fn run_test(backend: BackendType) -> Result<(), subxt::Error> {
|
|
// Note: This code fetches the chainspec from the Polkadot public RPC node.
|
|
// This is not recommended for production use, as it may be slow and unreliable.
|
|
// However, this can come in handy for testing purposes.
|
|
//
|
|
// let chainspec = subxt::utils::fetch_chainspec_from_rpc_node("wss://rpc.polkadot.io:443")
|
|
// .await
|
|
// .unwrap();
|
|
// let chain_config = chainspec.get();
|
|
|
|
tracing::trace!("Init light client");
|
|
let now = std::time::Instant::now();
|
|
let (_lc, rpc) = LightClient::relay_chain(POLKADOT_SPEC)?;
|
|
|
|
let api = match backend {
|
|
BackendType::Unstable => {
|
|
let backend =
|
|
ChainHeadBackend::builder().build_with_background_driver(RpcClient::new(rpc));
|
|
let api: OnlineClient<PolkadotConfig> =
|
|
OnlineClient::from_backend(Arc::new(backend)).await?;
|
|
api
|
|
}
|
|
|
|
BackendType::Legacy => Client::from_rpc_client(rpc).await?,
|
|
};
|
|
|
|
tracing::trace!("Light client initialization took {:?}", now.elapsed());
|
|
|
|
non_finalized_headers_subscription(&api).await?;
|
|
finalized_headers_subscription(&api).await?;
|
|
runtime_api_call(&api).await?;
|
|
storage_plain_lookup(&api).await?;
|
|
dynamic_constant_query(&api).await?;
|
|
dynamic_events(&api).await?;
|
|
|
|
tracing::trace!("Light complete testing took {:?}", now.elapsed());
|
|
Ok(())
|
|
}
|
|
|
|
/// Backend type for light client testing.
|
|
enum BackendType {
|
|
/// Use the unstable backend (ie chainHead).
|
|
Unstable,
|
|
/// Use the legacy backend.
|
|
Legacy,
|
|
}
|
|
|
|
#[tokio::test]
|
|
async fn light_client_testing() -> Result<(), subxt::Error> {
|
|
tracing_subscriber::fmt::init();
|
|
|
|
// Run light client test with both backends.
|
|
run_test(BackendType::Unstable).await?;
|
|
run_test(BackendType::Legacy).await?;
|
|
|
|
Ok(())
|
|
}
|