mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-07 02:38:02 +00:00
fd0fe57002
* Asset Id in Config trait * add example configuring the config * fmt * fix Default trait bound * merge examples, fix default again * adjust config in examples * Update subxt/src/config/mod.rs Co-authored-by: James Wilson <james@jsdw.me> --------- Co-authored-by: James Wilson <james@jsdw.me>
279 lines
9.8 KiB
Rust
279 lines
9.8 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.
|
|
|
|
//! Substrate specific configuration
|
|
|
|
use super::{Config, DefaultExtrinsicParams, DefaultExtrinsicParamsBuilder, Hasher, Header};
|
|
use codec::{Decode, Encode};
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
pub use crate::utils::{AccountId32, MultiAddress, MultiSignature};
|
|
pub use primitive_types::{H256, U256};
|
|
|
|
/// Default set of commonly used types by Substrate runtimes.
|
|
// Note: We only use this at the type level, so it should be impossible to
|
|
// create an instance of it.
|
|
pub enum SubstrateConfig {}
|
|
|
|
impl Config for SubstrateConfig {
|
|
type Hash = H256;
|
|
type AccountId = AccountId32;
|
|
type Address = MultiAddress<Self::AccountId, u32>;
|
|
type Signature = MultiSignature;
|
|
type Hasher = BlakeTwo256;
|
|
type Header = SubstrateHeader<u32, BlakeTwo256>;
|
|
type ExtrinsicParams = SubstrateExtrinsicParams<Self>;
|
|
type AssetId = u32;
|
|
}
|
|
|
|
/// A struct representing the signed extra and additional parameters required
|
|
/// to construct a transaction for the default substrate node.
|
|
pub type SubstrateExtrinsicParams<T> = DefaultExtrinsicParams<T>;
|
|
|
|
/// A builder which leads to [`SubstrateExtrinsicParams`] being constructed.
|
|
/// This is what you provide to methods like `sign_and_submit()`.
|
|
pub type SubstrateExtrinsicParamsBuilder<T> = DefaultExtrinsicParamsBuilder<T>;
|
|
|
|
/// A type that can hash values using the blaks2_256 algorithm.
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Encode)]
|
|
pub struct BlakeTwo256;
|
|
|
|
impl Hasher for BlakeTwo256 {
|
|
type Output = H256;
|
|
fn hash(s: &[u8]) -> Self::Output {
|
|
sp_core_hashing::blake2_256(s).into()
|
|
}
|
|
}
|
|
|
|
/// A generic Substrate header type, adapted from `sp_runtime::generic::Header`.
|
|
/// The block number and hasher can be configured to adapt this for other nodes.
|
|
#[derive(Encode, Decode, Debug, PartialEq, Eq, Clone, Serialize, Deserialize)]
|
|
#[serde(rename_all = "camelCase")]
|
|
pub struct SubstrateHeader<N: Copy + Into<U256> + TryFrom<U256>, H: Hasher> {
|
|
/// The parent hash.
|
|
pub parent_hash: H::Output,
|
|
/// The block number.
|
|
#[serde(
|
|
serialize_with = "serialize_number",
|
|
deserialize_with = "deserialize_number"
|
|
)]
|
|
#[codec(compact)]
|
|
pub number: N,
|
|
/// The state trie merkle root
|
|
pub state_root: H::Output,
|
|
/// The merkle root of the extrinsics.
|
|
pub extrinsics_root: H::Output,
|
|
/// A chain-specific digest of data useful for light clients or referencing auxiliary data.
|
|
pub digest: Digest,
|
|
}
|
|
|
|
impl<N, H> Header for SubstrateHeader<N, H>
|
|
where
|
|
N: Copy + Into<u64> + Into<U256> + TryFrom<U256> + Encode,
|
|
H: Hasher + Encode,
|
|
SubstrateHeader<N, H>: Encode + Decode,
|
|
{
|
|
type Number = N;
|
|
type Hasher = H;
|
|
fn number(&self) -> Self::Number {
|
|
self.number
|
|
}
|
|
}
|
|
|
|
/// Generic header digest. From `sp_runtime::generic::digest`.
|
|
#[derive(Encode, Decode, Debug, PartialEq, Eq, Clone, Serialize, Deserialize, Default)]
|
|
pub struct Digest {
|
|
/// A list of digest items.
|
|
pub logs: Vec<DigestItem>,
|
|
}
|
|
|
|
/// Digest item that is able to encode/decode 'system' digest items and
|
|
/// provide opaque access to other items. From `sp_runtime::generic::digest`.
|
|
#[derive(Debug, PartialEq, Eq, Clone)]
|
|
pub enum DigestItem {
|
|
/// A pre-runtime digest.
|
|
///
|
|
/// These are messages from the consensus engine to the runtime, although
|
|
/// the consensus engine can (and should) read them itself to avoid
|
|
/// code and state duplication. It is erroneous for a runtime to produce
|
|
/// these, but this is not (yet) checked.
|
|
///
|
|
/// NOTE: the runtime is not allowed to panic or fail in an `on_initialize`
|
|
/// call if an expected `PreRuntime` digest is not present. It is the
|
|
/// responsibility of a external block verifier to check this. Runtime API calls
|
|
/// will initialize the block without pre-runtime digests, so initialization
|
|
/// cannot fail when they are missing.
|
|
PreRuntime(ConsensusEngineId, Vec<u8>),
|
|
|
|
/// A message from the runtime to the consensus engine. This should *never*
|
|
/// be generated by the native code of any consensus engine, but this is not
|
|
/// checked (yet).
|
|
Consensus(ConsensusEngineId, Vec<u8>),
|
|
|
|
/// Put a Seal on it. This is only used by native code, and is never seen
|
|
/// by runtimes.
|
|
Seal(ConsensusEngineId, Vec<u8>),
|
|
|
|
/// Some other thing. Unsupported and experimental.
|
|
Other(Vec<u8>),
|
|
|
|
/// An indication for the light clients that the runtime execution
|
|
/// environment is updated.
|
|
///
|
|
/// Currently this is triggered when:
|
|
/// 1. Runtime code blob is changed or
|
|
/// 2. `heap_pages` value is changed.
|
|
RuntimeEnvironmentUpdated,
|
|
}
|
|
|
|
// From sp_runtime::generic, DigestItem enum indexes are encoded using this:
|
|
#[repr(u32)]
|
|
#[derive(Encode, Decode)]
|
|
enum DigestItemType {
|
|
Other = 0u32,
|
|
Consensus = 4u32,
|
|
Seal = 5u32,
|
|
PreRuntime = 6u32,
|
|
RuntimeEnvironmentUpdated = 8u32,
|
|
}
|
|
impl Encode for DigestItem {
|
|
fn encode(&self) -> Vec<u8> {
|
|
let mut v = Vec::new();
|
|
|
|
match self {
|
|
Self::Consensus(val, data) => {
|
|
DigestItemType::Consensus.encode_to(&mut v);
|
|
(val, data).encode_to(&mut v);
|
|
}
|
|
Self::Seal(val, sig) => {
|
|
DigestItemType::Seal.encode_to(&mut v);
|
|
(val, sig).encode_to(&mut v);
|
|
}
|
|
Self::PreRuntime(val, data) => {
|
|
DigestItemType::PreRuntime.encode_to(&mut v);
|
|
(val, data).encode_to(&mut v);
|
|
}
|
|
Self::Other(val) => {
|
|
DigestItemType::Other.encode_to(&mut v);
|
|
val.encode_to(&mut v);
|
|
}
|
|
Self::RuntimeEnvironmentUpdated => {
|
|
DigestItemType::RuntimeEnvironmentUpdated.encode_to(&mut v);
|
|
}
|
|
}
|
|
|
|
v
|
|
}
|
|
}
|
|
impl Decode for DigestItem {
|
|
fn decode<I: codec::Input>(input: &mut I) -> Result<Self, codec::Error> {
|
|
let item_type: DigestItemType = Decode::decode(input)?;
|
|
match item_type {
|
|
DigestItemType::PreRuntime => {
|
|
let vals: (ConsensusEngineId, Vec<u8>) = Decode::decode(input)?;
|
|
Ok(Self::PreRuntime(vals.0, vals.1))
|
|
}
|
|
DigestItemType::Consensus => {
|
|
let vals: (ConsensusEngineId, Vec<u8>) = Decode::decode(input)?;
|
|
Ok(Self::Consensus(vals.0, vals.1))
|
|
}
|
|
DigestItemType::Seal => {
|
|
let vals: (ConsensusEngineId, Vec<u8>) = Decode::decode(input)?;
|
|
Ok(Self::Seal(vals.0, vals.1))
|
|
}
|
|
DigestItemType::Other => Ok(Self::Other(Decode::decode(input)?)),
|
|
DigestItemType::RuntimeEnvironmentUpdated => Ok(Self::RuntimeEnvironmentUpdated),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Consensus engine unique ID. From `sp_runtime::ConsensusEngineId`.
|
|
pub type ConsensusEngineId = [u8; 4];
|
|
|
|
impl serde::Serialize for DigestItem {
|
|
fn serialize<S>(&self, seq: S) -> Result<S::Ok, S::Error>
|
|
where
|
|
S: serde::Serializer,
|
|
{
|
|
self.using_encoded(|bytes| impl_serde::serialize::serialize(bytes, seq))
|
|
}
|
|
}
|
|
|
|
impl<'a> serde::Deserialize<'a> for DigestItem {
|
|
fn deserialize<D>(de: D) -> Result<Self, D::Error>
|
|
where
|
|
D: serde::Deserializer<'a>,
|
|
{
|
|
let r = impl_serde::serialize::deserialize(de)?;
|
|
Decode::decode(&mut &r[..])
|
|
.map_err(|e| serde::de::Error::custom(format!("Decode error: {e}")))
|
|
}
|
|
}
|
|
|
|
fn serialize_number<S, T: Copy + Into<U256>>(val: &T, s: S) -> Result<S::Ok, S::Error>
|
|
where
|
|
S: serde::Serializer,
|
|
{
|
|
let u256: U256 = (*val).into();
|
|
serde::Serialize::serialize(&u256, s)
|
|
}
|
|
|
|
fn deserialize_number<'a, D, T: TryFrom<U256>>(d: D) -> Result<T, D::Error>
|
|
where
|
|
D: serde::Deserializer<'a>,
|
|
{
|
|
// At the time of writing, Smoldot gives back block numbers in numeric rather
|
|
// than hex format. So let's support deserializing from both here:
|
|
use crate::backend::legacy::rpc_methods::NumberOrHex;
|
|
let number_or_hex = NumberOrHex::deserialize(d)?;
|
|
let u256 = number_or_hex.into_u256();
|
|
TryFrom::try_from(u256).map_err(|_| serde::de::Error::custom("Try from failed"))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod test {
|
|
use super::*;
|
|
|
|
// Smoldot returns numeric block numbers in the header at the time of writing;
|
|
// ensure we can deserialize them properly.
|
|
#[test]
|
|
fn can_deserialize_numeric_block_number() {
|
|
let numeric_block_number_json = r#"
|
|
{
|
|
"digest": {
|
|
"logs": []
|
|
},
|
|
"extrinsicsRoot": "0x0000000000000000000000000000000000000000000000000000000000000000",
|
|
"number": 4,
|
|
"parentHash": "0xcb2690b2c85ceab55be03fc7f7f5f3857e7efeb7a020600ebd4331e10be2f7a5",
|
|
"stateRoot": "0x0000000000000000000000000000000000000000000000000000000000000000"
|
|
}
|
|
"#;
|
|
|
|
let header: SubstrateHeader<u32, BlakeTwo256> =
|
|
serde_json::from_str(numeric_block_number_json).expect("valid block header");
|
|
assert_eq!(header.number(), 4);
|
|
}
|
|
|
|
// Substrate returns hex block numbers; ensure we can also deserialize those OK.
|
|
#[test]
|
|
fn can_deserialize_hex_block_number() {
|
|
let numeric_block_number_json = r#"
|
|
{
|
|
"digest": {
|
|
"logs": []
|
|
},
|
|
"extrinsicsRoot": "0x0000000000000000000000000000000000000000000000000000000000000000",
|
|
"number": "0x04",
|
|
"parentHash": "0xcb2690b2c85ceab55be03fc7f7f5f3857e7efeb7a020600ebd4331e10be2f7a5",
|
|
"stateRoot": "0x0000000000000000000000000000000000000000000000000000000000000000"
|
|
}
|
|
"#;
|
|
|
|
let header: SubstrateHeader<u32, BlakeTwo256> =
|
|
serde_json::from_str(numeric_block_number_json).expect("valid block header");
|
|
assert_eq!(header.number(), 4);
|
|
}
|
|
}
|