1#![deny(missing_docs)]
25#![cfg_attr(not(feature = "std"), no_std)]
26
27extern crate alloc;
28
29mod methods;
30mod utils;
31
32pub mod extrinsics {
33 pub use crate::methods::extrinsic_decoder::{
39 Extrinsic, ExtrinsicDecodeError, ExtrinsicExtensions, ExtrinsicOwned, ExtrinsicSignature,
40 ExtrinsicType, NamedArg, decode_extrinsic,
41 };
42 pub use crate::methods::extrinsic_type_info::{
43 ExtrinsicCallInfo, ExtrinsicExtensionInfo, ExtrinsicInfoArg, ExtrinsicInfoError,
44 ExtrinsicSignatureInfo, ExtrinsicTypeInfo,
45 };
46}
47
48pub mod storage {
49 pub use crate::methods::storage_decoder::{
60 StorageKey, StorageKeyDecodeError, StorageKeyPart, StorageKeyPartValue,
61 StorageKeyValueDecodeError, StorageValueDecodeError,
62 decode_default_storage_value_with_info, decode_storage_key, decode_storage_key_values,
63 decode_storage_key_with_info, decode_storage_value, decode_storage_value_with_info,
64 };
65 pub use crate::methods::storage_encoder::{
66 StorageKeyEncodeError, encode_storage_key, encode_storage_key_prefix,
67 encode_storage_key_suffix, encode_storage_key_suffix_to,
68 encode_storage_key_suffix_with_info_to, encode_storage_key_to,
69 encode_storage_key_with_info, encode_storage_key_with_info_to,
70 };
71 pub use crate::methods::storage_type_info::{
72 StorageEntry, StorageEntryInfo, StorageHasher, StorageInfo, StorageInfoError,
73 StorageKeyInfo, StorageTypeInfo,
74 };
75 pub use crate::utils::{
76 DecodableValues, EncodableValues, IntoDecodableValues, IntoEncodableValues,
77 };
78}
79
80pub mod constants {
81 pub use crate::methods::constant_decoder::{
90 ConstantDecodeError, decode_constant, decode_constant_with_info,
91 };
92 pub use crate::methods::constant_type_info::{
93 ConstantEntry, ConstantEntryInfo, ConstantInfo, ConstantInfoError, ConstantTypeInfo,
94 };
95}
96
97pub mod runtime_apis {
98 pub use crate::methods::runtime_api_decoder::{
108 RuntimeApiDecodeError, decode_runtime_api_response, decode_runtime_api_response_with_info,
109 };
110 pub use crate::methods::runtime_api_encoder::{
111 RuntimeApiInputsEncodeError, encode_runtime_api_inputs, encode_runtime_api_inputs_to,
112 encode_runtime_api_inputs_with_info_to, encode_runtime_api_name,
113 };
114 pub use crate::methods::runtime_api_type_info::{
115 RuntimeApiEntry, RuntimeApiEntryInfo, RuntimeApiInfo, RuntimeApiInfoError, RuntimeApiInput,
116 RuntimeApiTypeInfo,
117 };
118 pub use crate::utils::{EncodableValues, IntoEncodableValues};
119}
120
121pub mod view_functions {
122 pub use crate::methods::view_function_decoder::{
132 ViewFunctionDecodeError, decode_view_function_response,
133 decode_view_function_response_with_info,
134 };
135 pub use crate::methods::view_function_encoder::{
136 RUNTIME_API_NAME, ViewFunctionInputsEncodeError, encode_view_function_inputs,
137 encode_view_function_inputs_to, encode_view_function_inputs_with_info_to,
138 };
139 pub use crate::methods::view_function_type_info::{
140 ViewFunctionEntry, ViewFunctionEntryInfo, ViewFunctionInfo, ViewFunctionInfoError,
141 ViewFunctionInput, ViewFunctionTypeInfo,
142 };
143 pub use crate::utils::{EncodableValues, IntoEncodableValues};
144}
145
146pub mod custom_values {
147 pub use crate::methods::custom_value_decoder::{
156 CustomValueDecodeError, decode_custom_value, decode_custom_value_with_info,
157 };
158 pub use crate::methods::custom_value_type_info::{
159 CustomValue, CustomValueEntryInfo, CustomValueInfo, CustomValueInfoError,
160 CustomValueTypeInfo,
161 };
162}
163
164#[cfg(feature = "legacy-types")]
165pub mod legacy_types {
166 pub mod polkadot {
169 pub fn relay_chain() -> scale_info_legacy::ChainTypeRegistry {
173 let bytes = include_bytes!("../types/polkadot_relay_types.yaml");
176 serde_yaml::from_slice(bytes).expect("Polkadot RC types are valid YAML")
177 }
178 }
179
180 #[doc(hidden)]
182 pub mod kusama {
183 pub fn relay_chain() -> scale_info_legacy::ChainTypeRegistry {
187 let bytes = include_bytes!("../types/kusama_relay_types.yaml");
190 serde_yaml::from_slice(bytes).expect("Kusama RC types are valid YAML")
191 }
192
193 pub fn asset_hub() -> scale_info_legacy::ChainTypeRegistry {
195 let bytes = include_bytes!("../types/kusama_assethub_types.yaml");
198 serde_yaml::from_slice(bytes).expect("Kusama AssetHub types are valid YAML")
199 }
200 }
201}
202
203pub mod helpers {
204 pub use crate::methods::Entry;
213
214 pub use crate::utils::{
215 DecodableValues, DecodeErrorTrace, EncodableValues, IntoDecodableValues,
216 IntoEncodableValues, decode_with_error_tracing,
217 };
218 #[cfg(feature = "legacy")]
219 pub use crate::utils::{type_registry_from_metadata, type_registry_from_metadata_any};
220
221 pub use scale_decode::visitor::decode_with_visitor;
225
226 pub use scale_decode;
228}
229
230#[cfg(test)]
231mod test {
232 use crate::methods::extrinsic_type_info::ExtrinsicTypeInfo;
233 use crate::methods::runtime_api_type_info::RuntimeApiTypeInfo;
234 use crate::methods::storage_type_info::StorageTypeInfo;
235 use crate::methods::view_function_type_info::ViewFunctionTypeInfo;
236 use crate::utils::ToTypeRegistry;
237 use scale_info_legacy::type_registry::TypeRegistryResolveError;
238 use scale_info_legacy::{ChainTypeRegistry, LookupName};
239 use scale_type_resolver::Field;
240
241 #[test]
243 fn test_deserializing_legacy_types() {
244 let _ = crate::legacy_types::polkadot::relay_chain();
245 let _ = crate::legacy_types::kusama::relay_chain();
246 let _ = crate::legacy_types::kusama::asset_hub();
247 }
248
249 fn legacy_types() -> [(&'static str, ChainTypeRegistry); 3] {
250 [
251 ("Polkadot RC", crate::legacy_types::polkadot::relay_chain()),
252 ("Kusama RC", crate::legacy_types::kusama::relay_chain()),
253 ("Kusama AH", crate::legacy_types::kusama::asset_hub()),
254 ]
255 }
256
257 fn all_type_registry_sets(
258 registry: &scale_info_legacy::ChainTypeRegistry,
259 ) -> impl Iterator<Item = scale_info_legacy::TypeRegistrySet<'_>> {
260 let all_spec_versions = core::iter::once(u64::MAX)
261 .chain(registry.spec_version_ranges().map(|(low, _high)| low));
262 all_spec_versions.map(|version| registry.for_spec_version(version))
263 }
264
265 #[test]
266 fn test_legacy_types_have_sane_type_names() {
267 let builtins = &[
269 "u8", "u16", "u32", "u64", "u128", "u256", "i8", "i16", "i32", "i64", "i128", "i256",
270 "char", "bool", "str",
271 ];
272
273 for (chain, types) in legacy_types() {
274 for types in all_type_registry_sets(&types) {
275 for ty in types.keys() {
276 if let Some(path_and_name) = ty.name() {
277 let name = path_and_name.split("::").last().unwrap();
278
279 if builtins.contains(&name) {
280 continue;
281 }
282
283 if name.starts_with(|c: char| !c.is_uppercase() || !c.is_ascii_alphabetic())
284 {
285 panic!("{chain}: {ty} does not begin with an uppercase letter");
286 }
287 if name.contains(|c: char| !c.is_ascii_alphanumeric()) {
288 panic!("{chain}: {ty} contains a non-ascii-alphanumeric character");
289 }
290 }
291 }
292 }
293 }
294 }
295
296 #[test]
297 fn test_legacy_types_have_sane_field_names() {
298 fn assert_sane<'a>(
299 chain: &str,
300 ty: &LookupName,
301 fields: impl Iterator<Item = Field<'a, LookupName>>,
302 ) {
303 let field_names: Vec<_> = fields.map(|f| f.name).collect();
304
305 let all_fields_named = field_names.iter().all(|n| n.is_some());
306 let all_fields_unnamed = field_names.iter().all(|n| n.is_none());
307
308 if !(all_fields_named || all_fields_unnamed) {
309 panic!("{chain}: All fields must be named or unnamed, but aren't in '{ty}'");
310 }
311 if all_fields_named {
312 for name in field_names.into_iter().map(|n| n.unwrap()) {
313 let Some(fst) = name.chars().next() else {
314 panic!("{chain}: {ty} has a present but empty field name");
315 };
316 if !fst.is_ascii_alphabetic() {
317 panic!(
318 "{chain}: {ty} field name '{name}' is invalid (does not start with ascii letter)"
319 );
320 }
321 if !name.chars().all(|c| c.is_ascii_alphanumeric() || c == '_') {
322 panic!(
323 "{chain}: {ty} field name '{name}' is invalid (non-ascii-letter-or-number-or-underscore present in the name"
324 );
325 }
326 if name.contains(|c: char| c.is_uppercase()) {
327 panic!(
328 "{chain}: {ty} field name '{name}' contains uppercase. Field names should be lowercase only"
329 );
330 }
331 }
332 }
333 }
334
335 for (chain, types) in legacy_types() {
336 for types in all_type_registry_sets(&types) {
337 for ty in types.keys() {
338 let visitor = scale_type_resolver::visitor::new((), |_, _| ())
339 .visit_variant(|_ctx, _path, vars| {
340 for variant in vars {
341 assert_sane(chain, &ty, variant.fields);
342 }
343 })
344 .visit_composite(|_ctx, _path, fields| {
345 assert_sane(chain, &ty, fields);
346 });
347
348 if let Err(e) = types.resolve_type(ty.clone(), visitor) {
349 match e {
350 TypeRegistryResolveError::UnexpectedBitOrderType
351 | TypeRegistryResolveError::UnexpectedBitStoreType => {
352 }
354 e => panic!("{chain}: Cannot resolve type '{ty}': {e}"),
355 }
356 }
357 }
358 }
359 }
360 }
361
362 #[test]
363 fn test_legacy_types_have_sane_variant_names() {
364 fn assert_sane(chain: &str, ty: &LookupName, variant_name: &str) {
365 let Some(fst) = variant_name.chars().next() else {
366 panic!("{chain}: Enum {ty} has an empty variant");
367 };
368
369 if !fst.is_uppercase() {
370 panic!(
371 "{chain}: Enum {ty} variant name '{variant_name}' should start with an uppercase letter"
372 );
373 }
374 if !fst.is_ascii_alphabetic() {
375 panic!(
376 "{chain}: Enum {ty} variant name '{variant_name}' should start with an ASCII letter"
377 );
378 }
379 if !variant_name.chars().all(|c| c.is_ascii_alphanumeric()) {
380 panic!(
381 "{chain}: Enum {ty} variant name '{variant_name}' is invalid (non-ascii-letter-or-number present in the name"
382 );
383 }
384 }
385
386 for (chain, types) in legacy_types() {
387 for types in all_type_registry_sets(&types) {
388 for ty in types.keys() {
389 let visitor = scale_type_resolver::visitor::new((), |_, _| ()).visit_variant(
390 |_ctx, _path, vars| {
391 for variant in vars {
392 assert_sane(chain, &ty, variant.name);
393 }
394 },
395 );
396
397 if let Err(e) = types.resolve_type(ty.clone(), visitor) {
398 match e {
399 TypeRegistryResolveError::UnexpectedBitOrderType
400 | TypeRegistryResolveError::UnexpectedBitStoreType => {
401 }
403 e => panic!("{chain}: Cannot resolve type '{ty}': {e}"),
404 }
405 }
406 }
407 }
408 }
409 }
410
411 macro_rules! impls_trait {
412 ($type:ty, $trait:path) => {
413 const _: () = {
414 const fn assert_impl<T: $trait>() {}
415 assert_impl::<$type>();
416 };
417 };
418 }
419
420 #[rustfmt::skip]
424 const _: () = {
425 impls_trait!(frame_metadata::v8::RuntimeMetadataV8, ExtrinsicTypeInfo);
426 impls_trait!(frame_metadata::v9::RuntimeMetadataV9, ExtrinsicTypeInfo);
427 impls_trait!(frame_metadata::v10::RuntimeMetadataV10, ExtrinsicTypeInfo);
428 impls_trait!(frame_metadata::v11::RuntimeMetadataV11, ExtrinsicTypeInfo);
429 impls_trait!(frame_metadata::v12::RuntimeMetadataV12, ExtrinsicTypeInfo);
430 impls_trait!(frame_metadata::v13::RuntimeMetadataV13, ExtrinsicTypeInfo);
431 impls_trait!(frame_metadata::v14::RuntimeMetadataV14, ExtrinsicTypeInfo);
432 impls_trait!(frame_metadata::v15::RuntimeMetadataV15, ExtrinsicTypeInfo);
433 impls_trait!(frame_metadata::v16::RuntimeMetadataV16, ExtrinsicTypeInfo);
434
435 impls_trait!(frame_metadata::v8::RuntimeMetadataV8, StorageTypeInfo);
436 impls_trait!(frame_metadata::v9::RuntimeMetadataV9, StorageTypeInfo);
437 impls_trait!(frame_metadata::v10::RuntimeMetadataV10, StorageTypeInfo);
438 impls_trait!(frame_metadata::v11::RuntimeMetadataV11, StorageTypeInfo);
439 impls_trait!(frame_metadata::v12::RuntimeMetadataV12, StorageTypeInfo);
440 impls_trait!(frame_metadata::v13::RuntimeMetadataV13, StorageTypeInfo);
441 impls_trait!(frame_metadata::v14::RuntimeMetadataV14, StorageTypeInfo);
442 impls_trait!(frame_metadata::v15::RuntimeMetadataV15, StorageTypeInfo);
443 impls_trait!(frame_metadata::v16::RuntimeMetadataV16, StorageTypeInfo);
444
445 impls_trait!(frame_metadata::v16::RuntimeMetadataV16, ViewFunctionTypeInfo);
449
450 impls_trait!(scale_info_legacy::TypeRegistry, RuntimeApiTypeInfo);
453 impls_trait!(scale_info_legacy::TypeRegistrySet, RuntimeApiTypeInfo);
454 impls_trait!(frame_metadata::v15::RuntimeMetadataV15, RuntimeApiTypeInfo);
455 impls_trait!(frame_metadata::v16::RuntimeMetadataV16, RuntimeApiTypeInfo);
456
457 impls_trait!(frame_metadata::v8::RuntimeMetadataV8, ToTypeRegistry);
459 impls_trait!(frame_metadata::v9::RuntimeMetadataV9, ToTypeRegistry);
460 impls_trait!(frame_metadata::v10::RuntimeMetadataV10, ToTypeRegistry);
461 impls_trait!(frame_metadata::v11::RuntimeMetadataV11, ToTypeRegistry);
462 impls_trait!(frame_metadata::v12::RuntimeMetadataV12, ToTypeRegistry);
463 impls_trait!(frame_metadata::v13::RuntimeMetadataV13, ToTypeRegistry);
464 };
465}