mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-12 18:11:10 +00:00
599107b432
* WIP generalising RPC client * WIP: non-object-safe RpcClientT.. aah generics everywhere * WIP object-safe RpcClientT trait and no more extra generics * Get core things compiling again with object-safe RpcClientT trait * Make jsonrpsee optional and get test-runtime working again * cargo fmt * add RpcParams object to enforce correct formatting of rps params * Wee tweaks * clippy fixes * cargo fmt * TWeak a few types * make sure we get jsonrpsee-types, too * Add examples for rpc_params/RpcParams * more doc tweaks * remove a now unneeded dev note * Option<Box<RawValue>> instead to avoid allocations in some cases * update docs * tweak RpcClientT trait docs * Tweak docs around RpcClient and RpcClientT. Don't expose RpcClientT directly * more doc tweaking about RpcParams and undo decision not to expose RpcParamsT * Doc tweak Co-authored-by: Alexandru Vasile <60601340+lexnv@users.noreply.github.com> * more doc tweaks * Fix doc thing * Add an example of injecting a custom RPC client * Fix a typo * Address clippy things in example * Fix a silly typo * another clippy fix * rpc_params to panic instead of returning a result, like serde_json::json, and deref on Rpc<T> * fix docs Co-authored-by: Alexandru Vasile <60601340+lexnv@users.noreply.github.com>
280 lines
8.1 KiB
Rust
280 lines
8.1 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 super::{
|
|
OfflineClient,
|
|
OfflineClientT,
|
|
};
|
|
use crate::{
|
|
constants::ConstantsClient,
|
|
error::Error,
|
|
events::EventsClient,
|
|
rpc::{
|
|
Rpc,
|
|
RpcClientT,
|
|
RuntimeVersion,
|
|
},
|
|
storage::StorageClient,
|
|
tx::TxClient,
|
|
Config,
|
|
Metadata,
|
|
};
|
|
use derivative::Derivative;
|
|
use futures::future;
|
|
use parking_lot::RwLock;
|
|
use std::sync::Arc;
|
|
|
|
/// A trait representing a client that can perform
|
|
/// online actions.
|
|
pub trait OnlineClientT<T: Config>: OfflineClientT<T> {
|
|
/// Return an RPC client that can be used to communicate with a node.
|
|
fn rpc(&self) -> &Rpc<T>;
|
|
}
|
|
|
|
/// A client that can be used to perform API calls (that is, either those
|
|
/// requiriing an [`OfflineClientT`] or those requiring an [`OnlineClientT`]).
|
|
#[derive(Derivative)]
|
|
#[derivative(Clone(bound = ""))]
|
|
pub struct OnlineClient<T: Config> {
|
|
inner: Arc<RwLock<Inner<T>>>,
|
|
rpc: Rpc<T>,
|
|
}
|
|
|
|
#[derive(Derivative)]
|
|
#[derivative(Debug(bound = ""))]
|
|
struct Inner<T: Config> {
|
|
genesis_hash: T::Hash,
|
|
runtime_version: RuntimeVersion,
|
|
metadata: Metadata,
|
|
}
|
|
|
|
impl<T: Config> std::fmt::Debug for OnlineClient<T> {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.debug_struct("Client")
|
|
.field("rpc", &"RpcClient")
|
|
.field("inner", &self.inner)
|
|
.finish()
|
|
}
|
|
}
|
|
|
|
// The default constructors assume Jsonrpsee.
|
|
#[cfg(feature = "jsonrpsee")]
|
|
impl<T: Config> OnlineClient<T> {
|
|
/// Construct a new [`OnlineClient`] using default settings which
|
|
/// point to a locally running node on `ws://127.0.0.1:9944`.
|
|
pub async fn new() -> Result<OnlineClient<T>, Error> {
|
|
let url = "ws://127.0.0.1:9944";
|
|
OnlineClient::from_url(url).await
|
|
}
|
|
|
|
/// Construct a new [`OnlineClient`], providing a URL to connect to.
|
|
pub async fn from_url(url: impl AsRef<str>) -> Result<OnlineClient<T>, Error> {
|
|
let client = jsonrpsee_helpers::ws_client(url.as_ref())
|
|
.await
|
|
.map_err(|e| crate::error::RpcError(e.to_string()))?;
|
|
OnlineClient::from_rpc_client(client).await
|
|
}
|
|
}
|
|
|
|
impl<T: Config> OnlineClient<T> {
|
|
/// Construct a new [`OnlineClient`] by providing an underlying [`RpcClientT`]
|
|
/// implementation to drive the connection.
|
|
pub async fn from_rpc_client<R: RpcClientT>(
|
|
rpc_client: R,
|
|
) -> Result<OnlineClient<T>, Error> {
|
|
let rpc = Rpc::new(rpc_client);
|
|
|
|
let (genesis_hash, runtime_version, metadata) = future::join3(
|
|
rpc.genesis_hash(),
|
|
rpc.runtime_version(None),
|
|
rpc.metadata(),
|
|
)
|
|
.await;
|
|
|
|
Ok(OnlineClient {
|
|
inner: Arc::new(RwLock::new(Inner {
|
|
genesis_hash: genesis_hash?,
|
|
runtime_version: runtime_version?,
|
|
metadata: metadata?,
|
|
})),
|
|
rpc,
|
|
})
|
|
}
|
|
|
|
/// Create an object which can be used to keep the runtime uptodate
|
|
/// in a separate thread.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```no_run
|
|
/// # #[tokio::main]
|
|
/// # async fn main() {
|
|
/// use subxt::{ OnlineClient, PolkadotConfig };
|
|
///
|
|
/// let client = OnlineClient::<PolkadotConfig>::new().await.unwrap();
|
|
///
|
|
/// let update_task = client.subscribe_to_updates();
|
|
/// tokio::spawn(async move {
|
|
/// update_task.perform_runtime_updates().await;
|
|
/// });
|
|
/// # }
|
|
/// ```
|
|
pub fn subscribe_to_updates(&self) -> ClientRuntimeUpdater<T> {
|
|
ClientRuntimeUpdater(self.clone())
|
|
}
|
|
|
|
/// Return the [`Metadata`] used in this client.
|
|
pub fn metadata(&self) -> Metadata {
|
|
let inner = self.inner.read();
|
|
inner.metadata.clone()
|
|
}
|
|
|
|
/// Return the genesis hash.
|
|
pub fn genesis_hash(&self) -> T::Hash {
|
|
let inner = self.inner.read();
|
|
inner.genesis_hash
|
|
}
|
|
|
|
/// Return the runtime version.
|
|
pub fn runtime_version(&self) -> RuntimeVersion {
|
|
let inner = self.inner.read();
|
|
inner.runtime_version.clone()
|
|
}
|
|
|
|
/// Return an RPC client to make raw requests with.
|
|
pub fn rpc(&self) -> &Rpc<T> {
|
|
&self.rpc
|
|
}
|
|
|
|
/// Return an offline client with the same configuration as this.
|
|
pub fn offline(&self) -> OfflineClient<T> {
|
|
let inner = self.inner.read();
|
|
OfflineClient::new(
|
|
inner.genesis_hash,
|
|
inner.runtime_version.clone(),
|
|
inner.metadata.clone(),
|
|
)
|
|
}
|
|
|
|
// Just a copy of the most important trait methods so that people
|
|
// don't need to import the trait for most things:
|
|
|
|
/// Work with transactions.
|
|
pub fn tx(&self) -> TxClient<T, Self> {
|
|
<Self as OfflineClientT<T>>::tx(self)
|
|
}
|
|
|
|
/// Work with events.
|
|
pub fn events(&self) -> EventsClient<T, Self> {
|
|
<Self as OfflineClientT<T>>::events(self)
|
|
}
|
|
|
|
/// Work with storage.
|
|
pub fn storage(&self) -> StorageClient<T, Self> {
|
|
<Self as OfflineClientT<T>>::storage(self)
|
|
}
|
|
|
|
/// Access constants.
|
|
pub fn constants(&self) -> ConstantsClient<T, Self> {
|
|
<Self as OfflineClientT<T>>::constants(self)
|
|
}
|
|
}
|
|
|
|
impl<T: Config> OfflineClientT<T> for OnlineClient<T> {
|
|
fn metadata(&self) -> Metadata {
|
|
self.metadata()
|
|
}
|
|
fn genesis_hash(&self) -> T::Hash {
|
|
self.genesis_hash()
|
|
}
|
|
fn runtime_version(&self) -> RuntimeVersion {
|
|
self.runtime_version()
|
|
}
|
|
}
|
|
|
|
impl<T: Config> OnlineClientT<T> for OnlineClient<T> {
|
|
fn rpc(&self) -> &Rpc<T> {
|
|
&self.rpc
|
|
}
|
|
}
|
|
|
|
/// Client wrapper for performing runtime updates. See [`OnlineClient::subscribe_to_updates()`]
|
|
/// for example usage.
|
|
pub struct ClientRuntimeUpdater<T: Config>(OnlineClient<T>);
|
|
|
|
impl<T: Config> ClientRuntimeUpdater<T> {
|
|
fn is_runtime_version_different(&self, new: &RuntimeVersion) -> bool {
|
|
let curr = self.0.inner.read();
|
|
&curr.runtime_version != new
|
|
}
|
|
|
|
/// Performs runtime updates indefinitely unless encountering an error.
|
|
///
|
|
/// *Note:* This will run indefinitely until it errors, so the typical usage
|
|
/// would be to run it in a separate background task.
|
|
pub async fn perform_runtime_updates(&self) -> Result<(), Error> {
|
|
// Obtain an update subscription to further detect changes in the runtime version of the node.
|
|
let mut update_subscription = self.0.rpc.subscribe_runtime_version().await?;
|
|
|
|
while let Some(new_runtime_version) = update_subscription.next().await {
|
|
// The Runtime Version obtained via subscription.
|
|
let new_runtime_version = new_runtime_version?;
|
|
|
|
// Ignore this update if there is no difference.
|
|
if !self.is_runtime_version_different(&new_runtime_version) {
|
|
continue
|
|
}
|
|
|
|
// Fetch new metadata.
|
|
let new_metadata = self.0.rpc.metadata().await?;
|
|
|
|
// Do the update.
|
|
let mut writable = self.0.inner.write();
|
|
writable.metadata = new_metadata;
|
|
writable.runtime_version = new_runtime_version;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// helpers for a jsonrpsee specific OnlineClient.
|
|
#[cfg(feature = "jsonrpsee")]
|
|
mod jsonrpsee_helpers {
|
|
pub use jsonrpsee::{
|
|
client_transport::ws::{
|
|
InvalidUri,
|
|
Receiver,
|
|
Sender,
|
|
Uri,
|
|
WsTransportClientBuilder,
|
|
},
|
|
core::{
|
|
client::{
|
|
Client,
|
|
ClientBuilder,
|
|
},
|
|
Error,
|
|
},
|
|
};
|
|
|
|
/// Build WS RPC client from URL
|
|
pub async fn ws_client(url: &str) -> Result<Client, Error> {
|
|
let (sender, receiver) = ws_transport(url).await?;
|
|
Ok(ClientBuilder::default()
|
|
.max_notifs_per_subscription(4096)
|
|
.build_with_tokio(sender, receiver))
|
|
}
|
|
|
|
async fn ws_transport(url: &str) -> Result<(Sender, Receiver), Error> {
|
|
let url: Uri = url
|
|
.parse()
|
|
.map_err(|e: InvalidUri| Error::Transport(e.into()))?;
|
|
WsTransportClientBuilder::default()
|
|
.build(url)
|
|
.await
|
|
.map_err(|e| Error::Transport(e.into()))
|
|
}
|
|
}
|