mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-12 18:11:10 +00:00
a2ee750365
* first iteration on storage multi keys * decoding values from concat style hashers * move util functions and remove comments * change codegen for storage keys and fix examples * trait bounds don't match scale value... * fix trait bounds and examples * reconstruct storage keys in iterations * build(deps): bump js-sys from 0.3.67 to 0.3.68 (#1428) Bumps [js-sys](https://github.com/rustwasm/wasm-bindgen) from 0.3.67 to 0.3.68. - [Release notes](https://github.com/rustwasm/wasm-bindgen/releases) - [Changelog](https://github.com/rustwasm/wasm-bindgen/blob/main/CHANGELOG.md) - [Commits](https://github.com/rustwasm/wasm-bindgen/commits) --- updated-dependencies: - dependency-name: js-sys dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump clap from 4.4.18 to 4.5.0 (#1427) Bumps [clap](https://github.com/clap-rs/clap) from 4.4.18 to 4.5.0. - [Release notes](https://github.com/clap-rs/clap/releases) - [Changelog](https://github.com/clap-rs/clap/blob/master/CHANGELOG.md) - [Commits](https://github.com/clap-rs/clap/compare/v4.4.18...clap_complete-v4.5.0) --- updated-dependencies: - dependency-name: clap dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump either from 1.9.0 to 1.10.0 (#1425) Bumps [either](https://github.com/rayon-rs/either) from 1.9.0 to 1.10.0. - [Commits](https://github.com/rayon-rs/either/compare/1.9.0...1.10.0) --- updated-dependencies: - dependency-name: either dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump thiserror from 1.0.56 to 1.0.57 (#1424) Bumps [thiserror](https://github.com/dtolnay/thiserror) from 1.0.56 to 1.0.57. - [Release notes](https://github.com/dtolnay/thiserror/releases) - [Commits](https://github.com/dtolnay/thiserror/compare/1.0.56...1.0.57) --- updated-dependencies: - dependency-name: thiserror dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * build(deps): bump jsonrpsee from 0.21.0 to 0.22.0 (#1426) * build(deps): bump jsonrpsee from 0.21.0 to 0.22.0 Bumps [jsonrpsee](https://github.com/paritytech/jsonrpsee) from 0.21.0 to 0.22.0. - [Release notes](https://github.com/paritytech/jsonrpsee/releases) - [Changelog](https://github.com/paritytech/jsonrpsee/blob/master/CHANGELOG.md) - [Commits](https://github.com/paritytech/jsonrpsee/compare/v0.21.0...v0.22.0) --- updated-dependencies: - dependency-name: jsonrpsee dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * Update Cargo.lock --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: James Wilson <james@jsdw.me> Co-authored-by: Niklas Adolfsson <niklasadolfsson1@gmail.com> * subxt: Derive `std::cmp` traits for subxt payloads and addresses (#1429) * subxt/tx: Derive std::cmp traits Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt/runtime_api: Derive std::cmp traits Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt/constants: Derive std::cmp traits Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt/custom_values: Derive std::cmp traits Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt/storage: Derive std::cmp traits Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt: Fix non_canonical_partial_ord_impl clippy introduced in 1.73 Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * subxt: Add comment wrt derivative issue that triggers clippy warning Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> * Update subxt/src/backend/mod.rs * Update subxt/src/constants/constant_address.rs --------- Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> Co-authored-by: Niklas Adolfsson <niklasadolfsson1@gmail.com> * fix clippy * add integration tests * fix doc tests * change hashing logic for hashers=1 * refactor * clippy and fmt * regenerate polkadot file which got changed by the automatic PR * nested design for storage keys * refactor codegen * codegen adjustments * fix storage hasher codegen test * Suggestions for storage value decoding (#1457) * Storage decode tweaks * doc tweak * more precise error when leftover or not enough bytes * integrate nits from PR * add fuzztest for storage keys, fix decoding bug * clippy and fmt * clippy * Niklas Suggestions * lifetime issues and iterator impls * fmt and clippy * regenerate polkadot.rs * fix storage key encoding for empty keys * rename trait methods for storage keys * fix hasher bug... * impl nits, add iterator struct seperate from `StorageHashers` * clippy fix * remove println --------- Signed-off-by: dependabot[bot] <support@github.com> Signed-off-by: Alexandru Vasile <alexandru.vasile@parity.io> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: James Wilson <james@jsdw.me> Co-authored-by: Niklas Adolfsson <niklasadolfsson1@gmail.com> Co-authored-by: Alexandru Vasile <60601340+lexnv@users.noreply.github.com>
131 lines
5.7 KiB
Rust
131 lines
5.7 KiB
Rust
// Copyright 2019-2023 Parity Technologies (UK) Ltd.
|
|
// This file is dual-licensed as Apache-2.0 or GPL-3.0.
|
|
// see LICENSE for license details.
|
|
|
|
//! # Storage
|
|
//!
|
|
//! A Substrate based chain can be seen as a key/value database which starts off at some initial
|
|
//! state, and is modified by the extrinsics in each block. This database is referred to as the
|
|
//! node storage. With Subxt, you can query this key/value storage with the following steps:
|
|
//!
|
|
//! 1. [Constructing a storage query](#constructing-a-storage-query).
|
|
//! 2. [Submitting the query to get back the associated values](#submitting-it).
|
|
//!
|
|
//! ## Constructing a storage query
|
|
//!
|
|
//! We can use the statically generated interface to build storage queries:
|
|
//!
|
|
//! ```rust,no_run
|
|
//! use subxt_signer::sr25519::dev;
|
|
//!
|
|
//! #[subxt::subxt(runtime_metadata_path = "../artifacts/polkadot_metadata_small.scale")]
|
|
//! pub mod polkadot {}
|
|
//!
|
|
//! let account = dev::alice().public_key().into();
|
|
//! let storage_query = polkadot::storage().system().account(&account);
|
|
//! ```
|
|
//!
|
|
//! Alternately, we can dynamically construct a storage query. This will not be type checked or
|
|
//! validated until it's submitted:
|
|
//!
|
|
//! ```rust,no_run
|
|
//! use subxt_signer::sr25519::dev;
|
|
//! use subxt::dynamic::Value;
|
|
//!
|
|
//! let account = dev::alice().public_key();
|
|
//! let storage_query = subxt::dynamic::storage("System", "Account", vec![
|
|
//! Value::from_bytes(account)
|
|
//! ]);
|
|
//! ```
|
|
//!
|
|
//! As well as accessing specific entries, some storage locations can also be iterated over (such as
|
|
//! the map of account information). To do this, suffix `_iter` onto the query constructor (this
|
|
//! will only be available on static constructors when iteration is actually possible):
|
|
//!
|
|
//! ```rust,no_run
|
|
//! #[subxt::subxt(runtime_metadata_path = "../artifacts/polkadot_metadata_small.scale")]
|
|
//! pub mod polkadot {}
|
|
//!
|
|
//! // A static query capable of iterating over accounts:
|
|
//! let storage_query = polkadot::storage().system().account_iter();
|
|
//! // A dynamic query to do the same:
|
|
//! let storage_query = subxt::dynamic::storage("System", "Account", ());
|
|
//! ```
|
|
//!
|
|
//! Some storage entries are maps with multiple keys. As an example, we might end up with
|
|
//! an API like `runtime::storage().foo().bar(u8, bool, u16, String)` to fetch some entry "bar".
|
|
//! When this is the case, the codegen will generate multiple iterator query functions alongside
|
|
//! the function to fetch an individual value:
|
|
//!
|
|
//! - `runtime::storage().foo().bar(u8, bool, u16, String)`: fetch a single entry from the "bar" map.
|
|
//! - `runtime::storage().foo().bar_iter()`: iterate over all of the entries in the "bar" map.
|
|
//! - `runtime::storage().foo().bar_iter1(u8)`: iterate over all of the entries in the "bar" map under
|
|
//! a given `u8`.
|
|
//! - `runtime::storage().foo().bar_iter2(u8, bool)`: iterate over all of the entries in the "bar" map under
|
|
//! a given `u8` and `bool` value.
|
|
//! - `runtime::storage().foo().bar_iter3(u8, bool, u16)`: iterate over all of the entries in the "bar" map under
|
|
//! a given `u8`, `bool` and `u16` value.
|
|
//!
|
|
//! All valid storage queries implement [`crate::storage::StorageAddress`]. As well as describing
|
|
//! how to build a valid storage query, this trait also has some associated types that determine the
|
|
//! shape of the result you'll get back, and determine what you can do with it (ie, can you iterate
|
|
//! over storage entries using it).
|
|
//!
|
|
//! Static queries set appropriate values for these associated types, and can therefore only be used
|
|
//! where it makes sense. Dynamic queries don't know any better and can be used in more places, but
|
|
//! may fail at runtime instead if they are invalid in those places.
|
|
//!
|
|
//! ## Submitting it
|
|
//!
|
|
//! Storage queries can be handed to various functions in [`crate::storage::Storage`] in order to
|
|
//! obtain the associated values (also referred to as storage entries) back.
|
|
//!
|
|
//! ### Fetching storage entries
|
|
//!
|
|
//! The simplest way to access storage entries is to construct a query and then call either
|
|
//! [`crate::storage::Storage::fetch()`] or [`crate::storage::Storage::fetch_or_default()`] (the
|
|
//! latter will only work for storage queries that have a default value when empty):
|
|
//!
|
|
//! ```rust,ignore
|
|
#![doc = include_str!("../../../examples/storage_fetch.rs")]
|
|
//! ```
|
|
//!
|
|
//! For completeness, below is an example using a dynamic query instead. The return type from a
|
|
//! dynamic query is a [`crate::dynamic::DecodedValueThunk`], which can be decoded into a
|
|
//! [`crate::dynamic::Value`], or else the raw bytes can be accessed instead.
|
|
//!
|
|
//! ```rust,ignore
|
|
#![doc = include_str!("../../../examples/storage_fetch_dynamic.rs")]
|
|
//! ```
|
|
//!
|
|
//! ### Iterating storage entries
|
|
//!
|
|
//! Many storage entries are maps of values; as well as fetching individual values, it's possible to
|
|
//! iterate over all of the values stored at that location:
|
|
//!
|
|
//! ```rust,ignore
|
|
#![doc = include_str!("../../../examples/storage_iterating.rs")]
|
|
//! ```
|
|
//!
|
|
//! Here's the same logic but using dynamically constructed values instead:
|
|
//!
|
|
//! ```rust,ignore
|
|
#![doc = include_str!("../../../examples/storage_iterating_dynamic.rs")]
|
|
//! ```
|
|
//!
|
|
//! Here is an example of iterating over partial keys. In this example some multi-signature operations
|
|
//! are sent to the node. We can iterate over the pending multisig operations of a single multisig account:
|
|
//!
|
|
//! ```rust,ignore
|
|
#![doc = include_str!("../../../examples/storage_iterating_partial.rs")]
|
|
//! ```
|
|
//!
|
|
//! ### Advanced
|
|
//!
|
|
//! For more advanced use cases, have a look at [`crate::storage::Storage::fetch_raw`] and
|
|
//! [`crate::storage::Storage::fetch_raw_keys`]. Both of these take raw bytes as arguments, which can be
|
|
//! obtained from a [`crate::storage::StorageAddress`] by using
|
|
//! [`crate::storage::StorageClient::address_bytes()`] or
|
|
//! [`crate::storage::StorageClient::address_root_bytes()`].
|
|
//!
|