mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-23 03:47:59 +00:00
Obtain DispatchError::Module info dynamically (#453)
* Add error information back into metadata to roll back removal in #394 * Go back to obtaining runtime error info * re-do codegen too to check that it's all gravy * Convert DispatchError module errors into a module variant to make them easier to work with * Fix broken doc link
This commit is contained in:
+21
-147
@@ -15,64 +15,18 @@
|
||||
// along with subxt. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
use frame_metadata::v14::RuntimeMetadataV14;
|
||||
use proc_macro2::{
|
||||
Span as Span2,
|
||||
TokenStream as TokenStream2,
|
||||
};
|
||||
use proc_macro2::TokenStream as TokenStream2;
|
||||
use proc_macro_error::abort_call_site;
|
||||
use quote::quote;
|
||||
use scale_info::TypeDef;
|
||||
|
||||
/// Tokens which allow us to provide static error information in the generated output.
|
||||
pub struct ErrorDetails {
|
||||
/// This type definition will be used in the `dispatch_error_impl_fn` and is
|
||||
/// expected to be generated somewhere in scope for that to be possible.
|
||||
pub type_def: TokenStream2,
|
||||
// A function which will live in an impl block for our `DispatchError`,
|
||||
// to statically return details for known error types:
|
||||
pub dispatch_error_impl_fn: TokenStream2,
|
||||
}
|
||||
|
||||
impl ErrorDetails {
|
||||
fn emit_compile_error(err: &str) -> ErrorDetails {
|
||||
let err_lit_str = syn::LitStr::new(err, Span2::call_site());
|
||||
ErrorDetails {
|
||||
type_def: quote!(),
|
||||
dispatch_error_impl_fn: quote!(compile_error!(#err_lit_str)),
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// The purpose of this is to enumerate all of the possible `(module_index, error_index)` error
|
||||
/// variants, so that we can convert `u8` error codes inside a generated `DispatchError` into
|
||||
/// nicer error strings with documentation. To do this, we emit the type we'll return instances of,
|
||||
/// and a function that returns such an instance for all of the error codes seen in the metadata.
|
||||
pub fn generate_error_details(metadata: &RuntimeMetadataV14) -> ErrorDetails {
|
||||
let errors = match pallet_errors(metadata) {
|
||||
Ok(errors) => errors,
|
||||
Err(e) => {
|
||||
let err_string =
|
||||
format!("Failed to generate error details from metadata: {}", e);
|
||||
return ErrorDetails::emit_compile_error(&err_string)
|
||||
}
|
||||
};
|
||||
|
||||
let match_body_items = errors.into_iter().map(|err| {
|
||||
let docs = err.docs;
|
||||
let pallet_index = err.pallet_index;
|
||||
let error_index = err.error_index;
|
||||
let pallet_name = err.pallet;
|
||||
let error_name = err.error;
|
||||
|
||||
quote! {
|
||||
(#pallet_index, #error_index) => Some(ErrorDetails {
|
||||
pallet: #pallet_name,
|
||||
error: #error_name,
|
||||
docs: #docs
|
||||
})
|
||||
}
|
||||
});
|
||||
|
||||
/// The aim of this is to implement the `::subxt::HasModuleError` trait for
|
||||
/// the generated `DispatchError`, so that we can obtain the module error details,
|
||||
/// if applicable, from it.
|
||||
pub fn generate_has_module_error_impl(
|
||||
metadata: &RuntimeMetadataV14,
|
||||
types_mod_ident: &syn::Ident,
|
||||
) -> TokenStream2 {
|
||||
let dispatch_error_def = metadata
|
||||
.types
|
||||
.types()
|
||||
@@ -111,108 +65,28 @@ pub fn generate_error_details(metadata: &RuntimeMetadataV14) -> ErrorDetails {
|
||||
false
|
||||
};
|
||||
|
||||
let dispatch_error_impl_fn = if module_variant_is_struct {
|
||||
let trait_fn_body = if module_variant_is_struct {
|
||||
quote! {
|
||||
pub fn details(&self) -> Option<ErrorDetails> {
|
||||
if let Self::Module { error, index } = self {
|
||||
match (index, error) {
|
||||
#( #match_body_items ),*,
|
||||
_ => None
|
||||
}
|
||||
} else {
|
||||
None
|
||||
}
|
||||
if let &Self::Module { index, error } = self {
|
||||
Some((index, error))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
} else {
|
||||
quote! {
|
||||
pub fn details(&self) -> Option<ErrorDetails> {
|
||||
if let Self::Module (module_error) = self {
|
||||
match (module_error.index, module_error.error) {
|
||||
#( #match_body_items ),*,
|
||||
_ => None
|
||||
}
|
||||
} else {
|
||||
None
|
||||
}
|
||||
if let Self::Module (module_error) = self {
|
||||
Some((module_error.index, module_error.error))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
ErrorDetails {
|
||||
type_def: quote! {
|
||||
pub struct ErrorDetails {
|
||||
pub pallet: &'static str,
|
||||
pub error: &'static str,
|
||||
pub docs: &'static str,
|
||||
}
|
||||
},
|
||||
dispatch_error_impl_fn,
|
||||
}
|
||||
}
|
||||
|
||||
fn pallet_errors(
|
||||
metadata: &RuntimeMetadataV14,
|
||||
) -> Result<Vec<ErrorMetadata>, InvalidMetadataError> {
|
||||
let get_type_def_variant = |type_id: u32| {
|
||||
let ty = metadata
|
||||
.types
|
||||
.resolve(type_id)
|
||||
.ok_or(InvalidMetadataError::MissingType(type_id))?;
|
||||
if let scale_info::TypeDef::Variant(var) = ty.type_def() {
|
||||
Ok(var)
|
||||
} else {
|
||||
Err(InvalidMetadataError::TypeDefNotVariant(type_id))
|
||||
}
|
||||
};
|
||||
|
||||
let mut pallet_errors = vec![];
|
||||
for pallet in &metadata.pallets {
|
||||
let error = match &pallet.error {
|
||||
Some(err) => err,
|
||||
None => continue,
|
||||
};
|
||||
|
||||
let type_def_variant = get_type_def_variant(error.ty.id())?;
|
||||
for var in type_def_variant.variants().iter() {
|
||||
pallet_errors.push(ErrorMetadata {
|
||||
pallet_index: pallet.index,
|
||||
error_index: var.index(),
|
||||
pallet: pallet.name.clone(),
|
||||
error: var.name().clone(),
|
||||
docs: var.docs().join("\n"),
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
Ok(pallet_errors)
|
||||
}
|
||||
|
||||
/// Information about each error that we find in the metadata;
|
||||
/// used to generate the static error information.
|
||||
#[derive(Clone, Debug)]
|
||||
struct ErrorMetadata {
|
||||
pub pallet_index: u8,
|
||||
pub error_index: u8,
|
||||
pub pallet: String,
|
||||
pub error: String,
|
||||
pub docs: String,
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
enum InvalidMetadataError {
|
||||
MissingType(u32),
|
||||
TypeDefNotVariant(u32),
|
||||
}
|
||||
|
||||
impl std::fmt::Display for InvalidMetadataError {
|
||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||
match self {
|
||||
InvalidMetadataError::MissingType(n) => {
|
||||
write!(f, "Type {} missing from type registry", n)
|
||||
}
|
||||
InvalidMetadataError::TypeDefNotVariant(n) => {
|
||||
write!(f, "Type {} was not a variant/enum type", n)
|
||||
quote! {
|
||||
impl ::subxt::HasModuleError for #types_mod_ident::sp_runtime::DispatchError {
|
||||
fn module_error_indices(&self) -> Option<(u8,u8)> {
|
||||
#trait_fn_body
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -265,9 +265,8 @@ impl RuntimeGenerator {
|
||||
pallet.calls.as_ref().map(|_| pallet_mod_name)
|
||||
});
|
||||
|
||||
let error_details = errors::generate_error_details(&self.metadata);
|
||||
let error_type = error_details.type_def;
|
||||
let error_fn = error_details.dispatch_error_impl_fn;
|
||||
let has_module_error_impl =
|
||||
errors::generate_has_module_error_impl(&self.metadata, types_mod_ident);
|
||||
|
||||
let default_account_data_ident = format_ident!("DefaultAccountData");
|
||||
let default_account_data_impl = generate_default_account_data_impl(
|
||||
@@ -291,12 +290,8 @@ impl RuntimeGenerator {
|
||||
|
||||
/// The default error type returned when there is a runtime issue.
|
||||
pub type DispatchError = #types_mod_ident::sp_runtime::DispatchError;
|
||||
|
||||
// Statically generate error information so that we don't need runtime metadata for it.
|
||||
#error_type
|
||||
impl DispatchError {
|
||||
#error_fn
|
||||
}
|
||||
// Impl HasModuleError on DispatchError so we can pluck out module error details.
|
||||
#has_module_error_impl
|
||||
|
||||
#default_account_data_impl
|
||||
|
||||
|
||||
Reference in New Issue
Block a user