use crate::construct_runtime::Pallet;
use proc_macro2::TokenStream;
use quote::quote;
use std::str::FromStr;
use syn::Ident;
pub fn expand_outer_inherent(
runtime: &Ident,
block: &TokenStream,
unchecked_extrinsic: &TokenStream,
pallet_decls: &[Pallet],
scrate: &TokenStream,
) -> TokenStream {
let mut pallet_names = Vec::new();
let mut pallet_attrs = Vec::new();
let mut query_inherent_part_macros = Vec::new();
for pallet_decl in pallet_decls {
if pallet_decl.exists_part("Inherent") {
let name = &pallet_decl.name;
let path = &pallet_decl.path;
let attr = pallet_decl.cfg_pattern.iter().fold(TokenStream::new(), |acc, pattern| {
let attr = TokenStream::from_str(&format!("#[cfg({})]", pattern.original()))
.expect("was successfully parsed before; qed");
quote! {
#acc
#attr
}
});
pallet_names.push(name);
pallet_attrs.push(attr);
query_inherent_part_macros.push(quote! {
#path::__substrate_inherent_check::is_inherent_part_defined!(#name);
});
}
}
quote! {
#( #query_inherent_part_macros )*
trait InherentDataExt {
fn create_extrinsics(&self) ->
#scrate::__private::Vec<<#block as #scrate::sp_runtime::traits::Block>::Extrinsic>;
fn check_extrinsics(&self, block: &#block) -> #scrate::inherent::CheckInherentsResult;
}
impl InherentDataExt for #scrate::inherent::InherentData {
fn create_extrinsics(&self) ->
#scrate::__private::Vec<<#block as #scrate::sp_runtime::traits::Block>::Extrinsic>
{
use #scrate::{inherent::ProvideInherent, traits::InherentBuilder};
let mut inherents = #scrate::__private::Vec::new();
#(
#pallet_attrs
if let Some(inherent) = #pallet_names::create_inherent(self) {
let inherent = <#unchecked_extrinsic as InherentBuilder>::new_inherent(
inherent.into(),
);
inherents.push(inherent);
}
)*
inherents
}
fn check_extrinsics(&self, block: &#block) -> #scrate::inherent::CheckInherentsResult {
use #scrate::inherent::{ProvideInherent, IsFatalError};
use #scrate::traits::{IsSubType, ExtrinsicCall};
use #scrate::sp_runtime::traits::Block as _;
use #scrate::__private::{sp_inherents::Error, log};
let mut result = #scrate::inherent::CheckInherentsResult::new();
fn handle_put_error_result(res: Result<(), Error>) {
const LOG_TARGET: &str = "runtime::inherent";
match res {
Ok(()) => (),
Err(Error::InherentDataExists(id)) =>
log::debug!(
target: LOG_TARGET,
"Some error already reported for inherent {:?}, new non fatal \
error is ignored",
id
),
Err(Error::FatalErrorReported) =>
log::error!(
target: LOG_TARGET,
"Fatal error already reported, unexpected considering there is \
only one fatal error",
),
Err(_) =>
log::error!(
target: LOG_TARGET,
"Unexpected error from `put_error` operation",
),
}
}
for xt in block.extrinsics() {
if !(#scrate::sp_runtime::traits::ExtrinsicLike::is_bare(xt)) {
break
}
let mut is_inherent = false;
#(
#pallet_attrs
{
let call = <#unchecked_extrinsic as ExtrinsicCall>::call(xt);
if let Some(call) = IsSubType::<_>::is_sub_type(call) {
if #pallet_names::is_inherent(call) {
is_inherent = true;
if let Err(e) = #pallet_names::check_inherent(call, self) {
handle_put_error_result(result.put_error(
#pallet_names::INHERENT_IDENTIFIER, &e
));
if e.is_fatal_error() {
return result;
}
}
}
}
}
)*
if !is_inherent {
break
}
}
#(
#pallet_attrs
match #pallet_names::is_inherent_required(self) {
Ok(Some(e)) => {
let found = block.extrinsics().iter().any(|xt| {
let is_bare = #scrate::sp_runtime::traits::ExtrinsicLike::is_bare(xt);
if is_bare {
let call = <
#unchecked_extrinsic as ExtrinsicCall
>::call(xt);
if let Some(call) = IsSubType::<_>::is_sub_type(call) {
#pallet_names::is_inherent(&call)
} else {
false
}
} else {
false
}
});
if !found {
handle_put_error_result(result.put_error(
#pallet_names::INHERENT_IDENTIFIER, &e
));
if e.is_fatal_error() {
return result;
}
}
},
Ok(None) => (),
Err(e) => {
handle_put_error_result(result.put_error(
#pallet_names::INHERENT_IDENTIFIER, &e
));
if e.is_fatal_error() {
return result;
}
},
}
)*
result
}
}
impl #scrate::traits::IsInherent<<#block as #scrate::sp_runtime::traits::Block>::Extrinsic> for #runtime {
fn is_inherent(ext: &<#block as #scrate::sp_runtime::traits::Block>::Extrinsic) -> bool {
use #scrate::inherent::ProvideInherent;
use #scrate::traits::{IsSubType, ExtrinsicCall};
let is_bare = #scrate::sp_runtime::traits::ExtrinsicLike::is_bare(ext);
if !is_bare {
return false
}
#(
#pallet_attrs
{
let call = <#unchecked_extrinsic as ExtrinsicCall>::call(ext);
if let Some(call) = IsSubType::<_>::is_sub_type(call) {
if <#pallet_names as ProvideInherent>::is_inherent(&call) {
return true;
}
}
}
)*
false
}
}
impl #scrate::traits::EnsureInherentsAreFirst<#block> for #runtime {
fn ensure_inherents_are_first(block: &#block) -> Result<u32, u32> {
use #scrate::inherent::ProvideInherent;
use #scrate::traits::{IsSubType, ExtrinsicCall};
use #scrate::sp_runtime::traits::Block as _;
let mut num_inherents = 0u32;
for (i, xt) in block.extrinsics().iter().enumerate() {
if <Self as #scrate::traits::IsInherent<_>>::is_inherent(xt) {
if num_inherents != i as u32 {
return Err(i as u32);
}
num_inherents += 1; }
}
Ok(num_inherents)
}
}
}
}