Signed Extensions
- In this lecture you will learn above one of the most advance FRAME concepts, Signed Extensions.
- They allow for a multitude of custom features to be added to FRAME transactions.
History
- Signed Extensions originally where added to implement tipping in a reasonable way.
- Originally, your dumb instructor (@kianenigma) had the idea of hard-coding it into the
UncheckedExtrinsic
, until @gavofyork jumped in with the idea of signed extensions.
Tipped Transaction Type. by kianenigma · Pull Request #2930 · paritytech/substrate > Extensible transactions (and tips) by gavofyork · Pull Request #3102 · paritytech/substrate
History
- In essence, they are a generic way to extend the transaction. Moreover, if they have additional payload, it is signed, therefore
SignedExtension
.
Anatomy
A signed extension can be either combination of the following things:
- Some additional data that is attached to the transaction.
- The tip!
- Some hooks that are executed before and after the transaction is executed.
- Before each transaction is executed, it must pay its fee upfront.
- Perhaps refund the fee partially 🤑.
Anatomy
- Some additional validation logic that is used to validate the transaction, and give feedback to the pool.
- Set priority of transaction priority based on some metric!
- Some additional data that must be present in the signed payload of each transaction.
- Data that the sender has, the chain also has, it is not communicated itself, but it is part of the signature payload.
- Spec version and genesis hash is part of all transaction's signature payload!
Anatomy: Let's Peek at the Trait
pub trait SignedExtension: Codec + Debug + Sync + Send + Clone + Eq + PartialEq + StaticTypeInfo { fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError>; fn validate(..) -> TransactionValidity; fn validate_unsigned(..) -> TransactionValidity; fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>; fn pre_dispatch_unsigned() -> Result<(), TransactionValidityError>; fn post_dispatch() -> Result<(), TransactionValidityError>; }
pub trait SignedExtension: Codec + Debug + Sync + Send + Clone + Eq + PartialEq + StaticTypeInfo { fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError>; fn validate(..) -> TransactionValidity; fn validate_unsigned(..) -> TransactionValidity; fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>; fn pre_dispatch_unsigned() -> Result<(), TransactionValidityError>; fn post_dispatch() -> Result<(), TransactionValidityError>; }
pub trait SignedExtension: Codec + Debug + Sync + Send + Clone + Eq + PartialEq + StaticTypeInfo { fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError>; fn validate(..) -> TransactionValidity; fn validate_unsigned(..) -> TransactionValidity; fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>; fn pre_dispatch_unsigned() -> Result<(), TransactionValidityError>; fn post_dispatch() -> Result<(), TransactionValidityError>; }
pub trait SignedExtension: Codec + Debug + Sync + Send + Clone + Eq + PartialEq + StaticTypeInfo { fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError>; fn validate(..) -> TransactionValidity; fn validate_unsigned(..) -> TransactionValidity; fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>; fn pre_dispatch_unsigned() -> Result<(), TransactionValidityError>; fn post_dispatch() -> Result<(), TransactionValidityError>; }
pub trait SignedExtension: Codec + Debug + Sync + Send + Clone + Eq + PartialEq + StaticTypeInfo { fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError>; fn validate(..) -> TransactionValidity; fn validate_unsigned(..) -> TransactionValidity; fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>; fn pre_dispatch_unsigned() -> Result<(), TransactionValidityError>; fn post_dispatch() -> Result<(), TransactionValidityError>; }
Grouping Signed Extension
Is also a signed extension itself!
You can look at the implementation yourself.. but the TLDR is:
Main takeaways:
type AdditionalSigned = (SE1::AdditionalSigned, SE2::AdditionalSigned)
,- all of hooks:
- Executes each individually, combines results
Usage In The Runtime
- Each runtime has a bunch of signed extensions. They can be grouped as a tuple
pub type SignedExtra = (
frame_system::CheckNonZeroSender<Runtime>,
frame_system::CheckSpecVersion<Runtime>,
frame_system::CheckTxVersion<Runtime>,
frame_system::CheckGenesis<Runtime>,
pallet_asset_tx_payment::ChargeAssetTxPayment<Runtime>,
);
type UncheckedExtrinsic = generic::UncheckedExtrinsic<Address, Call, Signature, SignedExtra>;
- Signed extensions might originate from a pallet, but are applied to ALL EXTRINSICS 😮💨!
Encoding
struct Foo(u32, u32);
impl SignedExtension for Foo {
type AdditionalSigned = u32;
fn additional_signed(&self) -> Result<Self::AdditionalSigned, TransactionValidityError> {
Ok(42u32)
}
}
pub struct UncheckedExtrinsic<Address, Call, Signature, (Foo)>
{
pub signature: Option<(Address, Signature, Extra)>,
pub function: Call,
}
- 2 u32 are decoded as,
42u32
is expected to be in the signature payload.
Transaction Pool Validation
- Each pallet also has
#[pallet::validate_unsigned]
. - This kind of overlaps with creating a signed extension and implementing
validate_unsigned
.
Transaction Pool Validation
- Recall that transaction pool validation should be minimum effort and static.
- In
executive
, we only do the following:- check signature.
- call
Extra::validate
/Extra::validate_unsigned
- call
ValidateUnsigned::validate
, if unsigned. - NOTE dispatching ✅!
Transaction Pool Validation
- Crucially, you should make sure that you re-execute anything that you do in transaction pool validation in dispatch as well:
/// Do any pre-flight stuff for a signed transaction.
///
/// Make sure to perform the same checks as in [`Self::validate`].
fn pre_dispatch() -> Result<Self::Pre, TransactionValidityError>;
- Because conditions that are not stateless might change over time!
Post Dispatch
- The dispatch result, plus generic type (
type Pre
) returned frompre_dispatch
is passed topost_dispatch
. - See
impl Applyable for CheckedExtrinsic
for more info.
Notable Signed Extensions
- These are some of the default signed extensions that come in FRAME.
- See if you can predict how they are made!
ChargeTransactionPayment
Charge payments, refund if Pays::Yes
.
type Pre = (
// tip
BalanceOf<T>,
// who paid the fee - this is an option to allow for a Default impl.
Self::AccountId,
// imbalance resulting from withdrawing the fee
<<T as Config>::OnChargeTransaction as OnChargeTransaction<T>>::LiquidityInfo,
);
check_genesis
Wants to make sure you are signing against the right chain.
Put the genesis hash in additional_signed
.
check_spec_version
and check_tx_version
work very similarly.
check_non_zero_sender
- interesting story: any account can sign on behalf of the
0x00
account. - discovered by @xlc.
- uses
pre_dispatch
andvalidate
to ensure the signing account is not0x00
.
check_nonce
pre_dispatch
: check nonce and actually update it.validate
: check the nonce, DO NOT WRITE ANYTHING, setprovides
andrequires
.
- remember that:
validate
is only for lightweight checks, no read/write.- anything you write to storage is reverted anyhow.
check_weight
- Check there is enough weight in
validate
. - Check there is enough weight, and update the consumed weight in
pre_dispatch
. - Updated consumed weight in
post_dispatch
.
Big Picture: Pipeline of Extension
- Signed extensions (or at least the
pre_dispatch
andvalidate
part) remind me of the extension system ofexpress.js
, if any of you know what that is
Big Picture: Pipeline of Extension
Exercises
- Walk over the notable signed extensions above and riddle each other about how they work.
- SignedExtensions are an important part of the transaction encoding. Try and encode a correct transaction against a template runtime in any language that you want, using only a scale-codec library.
- SignedExtensions that logs something on each transaction
- SignedExtension that keeps a counter of all transactions
- SignedExtensions that keeps a counter of all successful/failed transactions
- SignedExtension that tries to refund the transaction from each account as long as they submit less than 1tx/day.