mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-28 07:27:55 +00:00
b413e5e84e
* work in progress * add custom types access * nit * custom values client * adjust light client * adjust doc comments * adjust book for custom values in code gen * format and check docs * work in progress * add custom types access * nit * custom values client * adjust light client * codegen and validation * adjust docs * use ignore in docs in book * change iter implementation * use validation hash and other codegen changes * add ui test for custom values codegen * allow 'latest' metadata to be returned from the fallback code (#1127) * nits * fix validation check * fix comments * nits --------- Co-authored-by: James Wilson <james@jsdw.me>
162 lines
5.6 KiB
Rust
162 lines
5.6 KiB
Rust
use crate::client::OfflineClientT;
|
|
use crate::custom_values::custom_value_address::CustomValueAddress;
|
|
use crate::error::MetadataError;
|
|
use crate::metadata::DecodeWithMetadata;
|
|
use crate::{Config, Error};
|
|
use derivative::Derivative;
|
|
|
|
/// A client for accessing custom values stored in the metadata.
|
|
#[derive(Derivative)]
|
|
#[derivative(Clone(bound = "Client: Clone"))]
|
|
pub struct CustomValuesClient<T, Client> {
|
|
client: Client,
|
|
_marker: std::marker::PhantomData<T>,
|
|
}
|
|
|
|
impl<T, Client> CustomValuesClient<T, Client> {
|
|
/// Create a new [`CustomValuesClient`].
|
|
pub fn new(client: Client) -> Self {
|
|
Self {
|
|
client,
|
|
_marker: std::marker::PhantomData,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: Config, Client: OfflineClientT<T>> CustomValuesClient<T, Client> {
|
|
/// Access a custom value by the address it is registered under. This can be just a [str] to get back a dynamic value,
|
|
/// or a static address from the generated static interface to get a value of a static type returned.
|
|
pub fn at<Address: CustomValueAddress + ?Sized>(
|
|
&self,
|
|
address: &Address,
|
|
) -> Result<Address::Target, Error> {
|
|
// 1. Validate custom value shape if hash given:
|
|
self.validate(address)?;
|
|
|
|
// 2. Attempt to decode custom value:
|
|
let metadata = self.client.metadata();
|
|
let custom = metadata.custom();
|
|
let custom_value = custom
|
|
.get(address.name())
|
|
.ok_or_else(|| MetadataError::CustomValueNameNotFound(address.name().to_string()))?;
|
|
|
|
let value = <Address::Target as DecodeWithMetadata>::decode_with_metadata(
|
|
&mut custom_value.bytes(),
|
|
custom_value.type_id(),
|
|
&metadata,
|
|
)?;
|
|
Ok(value)
|
|
}
|
|
|
|
/// Run the validation logic against some custom value address you'd like to access. Returns `Ok(())`
|
|
/// if the address is valid (or if it's not possible to check since the address has no validation hash).
|
|
/// Returns an error if the address was not valid (wrong name, type or raw bytes)
|
|
pub fn validate<Address: CustomValueAddress + ?Sized>(
|
|
&self,
|
|
address: &Address,
|
|
) -> Result<(), Error> {
|
|
let metadata = self.client.metadata();
|
|
if let Some(actual_hash) = address.validation_hash() {
|
|
let custom = metadata.custom();
|
|
let custom_value = custom
|
|
.get(address.name())
|
|
.ok_or_else(|| MetadataError::CustomValueNameNotFound(address.name().into()))?;
|
|
let expected_hash = custom_value.hash();
|
|
if actual_hash != expected_hash {
|
|
return Err(MetadataError::IncompatibleCodegen.into());
|
|
}
|
|
}
|
|
if metadata.custom().get(address.name()).is_none() {
|
|
return Err(MetadataError::IncompatibleCodegen.into());
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use crate::backend::RuntimeVersion;
|
|
use crate::custom_values::CustomValuesClient;
|
|
use crate::{Metadata, OfflineClient, SubstrateConfig};
|
|
use codec::Encode;
|
|
use scale_decode::DecodeAsType;
|
|
use scale_info::form::PortableForm;
|
|
use scale_info::TypeInfo;
|
|
use std::collections::BTreeMap;
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq, Encode, TypeInfo, DecodeAsType)]
|
|
pub struct Person {
|
|
age: u16,
|
|
name: String,
|
|
}
|
|
|
|
fn mock_metadata() -> Metadata {
|
|
let person_ty = scale_info::MetaType::new::<Person>();
|
|
let unit = scale_info::MetaType::new::<()>();
|
|
let mut types = scale_info::Registry::new();
|
|
let person_ty_id = types.register_type(&person_ty);
|
|
let unit_id = types.register_type(&unit);
|
|
let types: scale_info::PortableRegistry = types.into();
|
|
|
|
let person = Person {
|
|
age: 42,
|
|
name: "Neo".into(),
|
|
};
|
|
|
|
let person_value_metadata: frame_metadata::v15::CustomValueMetadata<PortableForm> =
|
|
frame_metadata::v15::CustomValueMetadata {
|
|
ty: person_ty_id,
|
|
value: person.encode(),
|
|
};
|
|
|
|
let frame_metadata = frame_metadata::v15::RuntimeMetadataV15 {
|
|
types,
|
|
pallets: vec![],
|
|
extrinsic: frame_metadata::v15::ExtrinsicMetadata {
|
|
version: 0,
|
|
address_ty: unit_id,
|
|
call_ty: unit_id,
|
|
signature_ty: unit_id,
|
|
extra_ty: unit_id,
|
|
signed_extensions: vec![],
|
|
},
|
|
ty: unit_id,
|
|
apis: vec![],
|
|
outer_enums: frame_metadata::v15::OuterEnums {
|
|
call_enum_ty: unit_id,
|
|
event_enum_ty: unit_id,
|
|
error_enum_ty: unit_id,
|
|
},
|
|
custom: frame_metadata::v15::CustomMetadata {
|
|
map: BTreeMap::from_iter([("Person".to_string(), person_value_metadata)]),
|
|
},
|
|
};
|
|
|
|
let metadata: subxt_metadata::Metadata = frame_metadata.try_into().unwrap();
|
|
Metadata::new(metadata)
|
|
}
|
|
|
|
#[test]
|
|
fn test_decoding() {
|
|
let client = OfflineClient::<SubstrateConfig>::new(
|
|
Default::default(),
|
|
RuntimeVersion {
|
|
spec_version: 0,
|
|
transaction_version: 0,
|
|
},
|
|
mock_metadata(),
|
|
);
|
|
let custom_value_client = CustomValuesClient::new(client);
|
|
assert!(custom_value_client.at("No one").is_err());
|
|
let person_decoded_value_thunk = custom_value_client.at("Person").unwrap();
|
|
let person: Person = person_decoded_value_thunk.as_type().unwrap();
|
|
assert_eq!(
|
|
person,
|
|
Person {
|
|
age: 42,
|
|
name: "Neo".into()
|
|
}
|
|
)
|
|
}
|
|
}
|