mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-29 12:37:57 +00:00
565 lines
22 KiB
Rust
565 lines
22 KiB
Rust
// This file is part of Substrate.
|
|
|
|
// Copyright (C) 2017-2022 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
//! Proc macro of Support code for the runtime.
|
|
|
|
#![recursion_limit = "512"]
|
|
|
|
mod clone_no_bound;
|
|
mod construct_runtime;
|
|
mod crate_version;
|
|
mod debug_no_bound;
|
|
mod default_no_bound;
|
|
mod dummy_part_checker;
|
|
mod key_prefix;
|
|
mod match_and_insert;
|
|
mod pallet;
|
|
mod partial_eq_no_bound;
|
|
mod storage;
|
|
mod transactional;
|
|
|
|
use proc_macro::TokenStream;
|
|
use std::{cell::RefCell, str::FromStr};
|
|
pub(crate) use storage::INHERENT_INSTANCE_NAME;
|
|
|
|
thread_local! {
|
|
/// A global counter, can be used to generate a relatively unique identifier.
|
|
static COUNTER: RefCell<Counter> = RefCell::new(Counter(0));
|
|
}
|
|
|
|
/// Counter to generate a relatively unique identifier for macros querying for the existence of
|
|
/// pallet parts. This is necessary because declarative macros gets hoisted to the crate root,
|
|
/// which shares the namespace with other pallets containing the very same query macros.
|
|
struct Counter(u64);
|
|
|
|
impl Counter {
|
|
fn inc(&mut self) -> u64 {
|
|
let ret = self.0;
|
|
self.0 += 1;
|
|
ret
|
|
}
|
|
}
|
|
|
|
/// Get the value from the given environment variable set by cargo.
|
|
///
|
|
/// The value is parsed into the requested destination type.
|
|
fn get_cargo_env_var<T: FromStr>(version_env: &str) -> std::result::Result<T, ()> {
|
|
let version = std::env::var(version_env)
|
|
.unwrap_or_else(|_| panic!("`{}` is always set by cargo; qed", version_env));
|
|
|
|
T::from_str(&version).map_err(drop)
|
|
}
|
|
|
|
/// Declares strongly-typed wrappers around codec-compatible types in storage.
|
|
///
|
|
/// ## Example
|
|
///
|
|
/// ```nocompile
|
|
/// decl_storage! {
|
|
/// trait Store for Module<T: Config> as Example {
|
|
/// Foo get(fn foo) config(): u32=12;
|
|
/// Bar: map hasher(identity) u32 => u32;
|
|
/// pub Zed build(|config| vec![(0, 0)]): map hasher(identity) u32 => u32;
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
///
|
|
/// Declaration is set with the header `(pub) trait Store for Module<T: Config> as Example`,
|
|
/// with `Store` a (pub) trait generated associating each storage item to the `Module` and
|
|
/// `as Example` setting the prefix used for storage items of this module. `Example` must be unique:
|
|
/// another module with the same name and the same inner storage item name will conflict.
|
|
/// `Example` is called the module prefix.
|
|
///
|
|
/// note: For instantiable modules the module prefix is prepended with instance
|
|
/// prefix. Instance prefix is "" for default instance and "Instance$n" for instance number $n.
|
|
/// Thus, instance 3 of module Example has a module prefix of `Instance3Example`
|
|
///
|
|
/// Basic storage consists of a name and a type; supported types are:
|
|
///
|
|
/// * Value: `Foo: type`: Implements the
|
|
/// [`StorageValue`](../frame_support/storage/trait.StorageValue.html) trait using the
|
|
/// [`StorageValue generator`](../frame_support/storage/generator/trait.StorageValue.html).
|
|
///
|
|
/// The generator is implemented with:
|
|
/// * `module_prefix`: module_prefix
|
|
/// * `storage_prefix`: storage_name
|
|
///
|
|
/// Thus the storage value is finally stored at:
|
|
/// ```nocompile
|
|
/// Twox128(module_prefix) ++ Twox128(storage_prefix)
|
|
/// ```
|
|
///
|
|
/// * Map: `Foo: map hasher($hash) type => type`: Implements the
|
|
/// [`StorageMap`](../frame_support/storage/trait.StorageMap.html) trait using the [`StorageMap
|
|
/// generator`](../frame_support/storage/generator/trait.StorageMap.html). And
|
|
/// [`StoragePrefixedMap`](../frame_support/storage/trait.StoragePrefixedMap.html).
|
|
///
|
|
/// `$hash` representing a choice of hashing algorithms available in the
|
|
/// [`Hashable`](../frame_support/trait.Hashable.html) trait. You will generally want to use one
|
|
/// of three hashers:
|
|
/// * `blake2_128_concat`: The default, safe choice. Use if you are unsure or don't care. It is
|
|
/// secure against user-tainted keys, fairly fast and memory-efficient and supports iteration
|
|
/// over its keys and values. This must be used if the keys of your map can be selected *en
|
|
/// masse* by untrusted users.
|
|
/// * `twox_64_concat`: This is an insecure hasher and can only be used safely if you know that
|
|
/// the preimages cannot be chosen at will by untrusted users. It is memory-efficient, extremely
|
|
/// performant and supports iteration over its keys and values. You can safely use this is the
|
|
/// key is:
|
|
/// - A (slowly) incrementing index.
|
|
/// - Known to be the result of a cryptographic hash (though `identity` is a better choice
|
|
/// here).
|
|
/// - Known to be the public key of a cryptographic key pair in existence.
|
|
/// * `identity`: This is not a hasher at all, and just uses the key material directly. Since it
|
|
/// does no hashing or appending, it's the fastest possible hasher, however, it's also the least
|
|
/// secure. It can be used only if you know that the key will be cryptographically/securely
|
|
/// randomly distributed over the binary encoding space. In most cases this will not be true.
|
|
/// One case where it is true, however, if where the key is itself the result of a cryptographic
|
|
/// hash of some existent data.
|
|
///
|
|
/// Other hashers will tend to be "opaque" and not support iteration over the keys in the
|
|
/// map. It is not recommended to use these.
|
|
///
|
|
/// The generator is implemented with:
|
|
/// * `module_prefix`: $module_prefix
|
|
/// * `storage_prefix`: storage_name
|
|
/// * `Hasher`: $hash
|
|
///
|
|
/// Thus the keys are stored at:
|
|
/// ```nocompile
|
|
/// twox128(module_prefix) ++ twox128(storage_prefix) ++ hasher(encode(key))
|
|
/// ```
|
|
///
|
|
/// * Double map: `Foo: double_map hasher($hash1) u32, hasher($hash2) u32 => u32`: Implements the
|
|
/// [`StorageDoubleMap`](../frame_support/storage/trait.StorageDoubleMap.html) trait using the
|
|
/// [`StorageDoubleMap
|
|
/// generator`](../frame_support/storage/generator/trait.StorageDoubleMap.html). And
|
|
/// [`StoragePrefixedMap`](../frame_support/storage/trait.StoragePrefixedMap.html).
|
|
///
|
|
/// `$hash1` and `$hash2` representing choices of hashing algorithms available in the
|
|
/// [`Hashable`](../frame_support/trait.Hashable.html) trait. They must be chosen with care, see
|
|
/// generator documentation.
|
|
///
|
|
/// The generator is implemented with:
|
|
/// * `module_prefix`: $module_prefix
|
|
/// * `storage_prefix`: storage_name
|
|
/// * `Hasher1`: $hash1
|
|
/// * `Hasher2`: $hash2
|
|
///
|
|
/// Thus keys are stored at:
|
|
/// ```nocompile
|
|
/// Twox128(module_prefix) ++ Twox128(storage_prefix) ++ Hasher1(encode(key1)) ++
|
|
/// Hasher2(encode(key2)) ```
|
|
///
|
|
/// Supported hashers (ordered from least to best security):
|
|
///
|
|
/// * `identity` - Just the unrefined key material. Use only when it is known to be a secure hash
|
|
/// already. The most efficient and iterable over keys.
|
|
/// * `twox_64_concat` - TwoX with 64bit + key concatenated. Use only when an untrusted source
|
|
/// cannot select and insert key values. Very efficient and iterable over keys.
|
|
/// * `blake2_128_concat` - Blake2 with 128bit + key concatenated. Slower but safe to use in all
|
|
/// circumstances. Iterable over keys.
|
|
///
|
|
/// Deprecated hashers, which do not support iteration over keys include:
|
|
/// * `twox_128` - TwoX with 128bit.
|
|
/// * `twox_256` - TwoX with with 256bit.
|
|
/// * `blake2_128` - Blake2 with 128bit.
|
|
/// * `blake2_256` - Blake2 with 256bit.
|
|
///
|
|
/// Basic storage can be extended as such:
|
|
///
|
|
/// `#vis #name get(fn #getter) config(#field_name) build(#closure): #type = #default;`
|
|
///
|
|
/// * `#vis`: Set the visibility of the structure. `pub` or nothing.
|
|
/// * `#name`: Name of the storage item, used as a prefix in storage.
|
|
/// * \[optional\] `get(fn #getter)`: Implements the function #getter to `Module`.
|
|
/// * \[optional\] `config(#field_name)`: `field_name` is optional if get is set.
|
|
/// Will include the item in `GenesisConfig`.
|
|
/// * \[optional\] `build(#closure)`: Closure called with storage overlays.
|
|
/// * \[optional\] `max_values(#expr)`: `expr` is an expression returning a `u32`. It is used to
|
|
/// implement `StorageInfoTrait`. Note this attribute is not available for storage value as the
|
|
/// maximum number of values is 1.
|
|
/// * `#type`: Storage type.
|
|
/// * \[optional\] `#default`: Value returned when none.
|
|
///
|
|
/// Storage items are accessible in multiple ways:
|
|
///
|
|
/// * The structure: `Foo` or `Foo::<T>` depending if the value type is generic or not.
|
|
/// * The `Store` trait structure: `<Module<T> as Store>::Foo`
|
|
/// * The getter on the module that calls get on the structure: `Module::<T>::foo()`
|
|
///
|
|
/// ## GenesisConfig
|
|
///
|
|
/// An optional `GenesisConfig` struct for storage initialization can be defined, either
|
|
/// when at least one storage field requires default initialization
|
|
/// (both `get` and `config` or `build`), or specifically as in:
|
|
///
|
|
/// ```nocompile
|
|
/// decl_storage! {
|
|
/// trait Store for Module<T: Config> as Example {
|
|
///
|
|
/// // Your storage items
|
|
/// }
|
|
/// add_extra_genesis {
|
|
/// config(genesis_field): GenesisFieldType;
|
|
/// config(genesis_field2): GenesisFieldType;
|
|
/// ...
|
|
/// build(|_: &Self| {
|
|
/// // Modification of storage
|
|
/// })
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
///
|
|
/// This struct can be exposed as `ExampleConfig` by the `construct_runtime!` macro like follows:
|
|
///
|
|
/// ```nocompile
|
|
/// construct_runtime!(
|
|
/// pub enum Runtime with ... {
|
|
/// ...,
|
|
/// Example: example::{Pallet, Storage, ..., Config<T>},
|
|
/// ...,
|
|
/// }
|
|
/// );
|
|
/// ```
|
|
///
|
|
/// ### Module with Instances
|
|
///
|
|
/// The `decl_storage!` macro supports building modules with instances with the following syntax
|
|
/// (`DefaultInstance` type is optional):
|
|
///
|
|
/// ```nocompile
|
|
/// trait Store for Module<T: Config<I>, I: Instance=DefaultInstance> as Example {}
|
|
/// ```
|
|
///
|
|
/// Accessing the structure no requires the instance as generic parameter:
|
|
/// * `Foo::<I>` if the value type is not generic
|
|
/// * `Foo::<T, I>` if the value type is generic
|
|
///
|
|
/// ## Where clause
|
|
///
|
|
/// This macro supports a where clause which will be replicated to all generated types.
|
|
///
|
|
/// ```nocompile
|
|
/// trait Store for Module<T: Config> as Example where T::AccountId: std::fmt::Display {}
|
|
/// ```
|
|
///
|
|
/// ## Limitations
|
|
///
|
|
/// # Instancing and generic `GenesisConfig`
|
|
///
|
|
/// If your module supports instancing and you see an error like `parameter `I` is never used` for
|
|
/// your `decl_storage!`, you are hitting a limitation of the current implementation. You probably
|
|
/// try to use an associated type of a non-instantiable trait. To solve this, add the following to
|
|
/// your macro call:
|
|
///
|
|
/// ```nocompile
|
|
/// add_extra_genesis {
|
|
/// config(phantom): std::marker::PhantomData<I>,
|
|
/// }
|
|
/// ```
|
|
///
|
|
/// This adds a field to your `GenesisConfig` with the name `phantom` that you can initialize with
|
|
/// `Default::default()`.
|
|
///
|
|
/// ## PoV information
|
|
///
|
|
/// To implement the trait `StorageInfoTrait` for storages an additional attribute can be used
|
|
/// `generate_storage_info`:
|
|
/// ```nocompile
|
|
/// decl_storage! { generate_storage_info
|
|
/// trait Store for ...
|
|
/// }
|
|
/// ```
|
|
#[proc_macro]
|
|
pub fn decl_storage(input: TokenStream) -> TokenStream {
|
|
storage::decl_storage_impl(input)
|
|
}
|
|
|
|
/// Construct a runtime, with the given name and the given pallets.
|
|
///
|
|
/// The parameters here are specific types for `Block`, `NodeBlock`, and `UncheckedExtrinsic`
|
|
/// and the pallets that are used by the runtime.
|
|
/// `Block` is the block type that is used in the runtime and `NodeBlock` is the block type
|
|
/// that is used in the node. For instance they can differ in the extrinsics type.
|
|
///
|
|
/// # Example:
|
|
///
|
|
/// ```ignore
|
|
/// construct_runtime!(
|
|
/// pub enum Runtime where
|
|
/// Block = Block,
|
|
/// NodeBlock = node::Block,
|
|
/// UncheckedExtrinsic = UncheckedExtrinsic
|
|
/// {
|
|
/// System: frame_system::{Pallet, Call, Event<T>, Config<T>} = 0,
|
|
/// Test: path::to::test::{Pallet, Call} = 1,
|
|
///
|
|
/// // Pallets with instances
|
|
/// Test2_Instance1: test2::<Instance1>::{Pallet, Call, Storage, Event<T, I>, Config<T, I>, Origin<T, I>},
|
|
/// Test2_DefaultInstance: test2::{Pallet, Call, Storage, Event<T>, Config<T>, Origin<T>} = 4,
|
|
///
|
|
/// // Pallets declared with `pallet` attribute macro: no need to define the parts
|
|
/// Test3_Instance1: test3::<Instance1>,
|
|
/// Test3_DefaultInstance: test3,
|
|
///
|
|
/// // with `exclude_parts` keyword some part can be excluded.
|
|
/// Test4_Instance1: test4::<Instance1> exclude_parts { Call, Origin },
|
|
/// Test4_DefaultInstance: test4 exclude_parts { Storage },
|
|
///
|
|
/// // with `use_parts` keyword, a subset of the pallet parts can be specified.
|
|
/// Test4_Instance1: test4::<Instance1> use_parts { Pallet, Call},
|
|
/// Test4_DefaultInstance: test4 use_parts { Pallet },
|
|
/// }
|
|
/// )
|
|
/// ```
|
|
///
|
|
/// Each pallet is declared as such:
|
|
/// * `Identifier`: name given to the pallet that uniquely identifies it.
|
|
///
|
|
/// * `:`: colon separator
|
|
///
|
|
/// * `path::to::pallet`: identifiers separated by colons which declare the path to a pallet
|
|
/// definition.
|
|
///
|
|
/// * `::<InstanceN>` optional: specify the instance of the pallet to use. If not specified it will
|
|
/// use the default instance (or the only instance in case of non-instantiable pallets).
|
|
///
|
|
/// * `::{ Part1, Part2<T>, .. }` optional if pallet declared with `frame_support::pallet`: Comma
|
|
/// separated parts declared with their generic. If a pallet is declared with
|
|
/// `frame_support::pallet` macro then the parts can be automatically derived if not explicitly
|
|
/// provided. We provide support for the following module parts in a pallet:
|
|
///
|
|
/// - `Pallet` - Required for all pallets
|
|
/// - `Call` - If the pallet has callable functions
|
|
/// - `Storage` - If the pallet uses storage
|
|
/// - `Event` or `Event<T>` (if the event is generic) - If the pallet emits events
|
|
/// - `Origin` or `Origin<T>` (if the origin is generic) - If the pallet has instanciable origins
|
|
/// - `Config` or `Config<T>` (if the config is generic) - If the pallet builds the genesis
|
|
/// storage with `GenesisConfig`
|
|
/// - `Inherent` - If the pallet provides/can check inherents.
|
|
/// - `ValidateUnsigned` - If the pallet validates unsigned extrinsics.
|
|
///
|
|
/// It is important to list these parts here to export them correctly in the metadata or to make
|
|
/// the pallet usable in the runtime.
|
|
///
|
|
/// * `exclude_parts { Part1, Part2 }` optional: comma separated parts without generics. I.e. one of
|
|
/// `Pallet`, `Call`, `Storage`, `Event`, `Origin`, `Config`, `Inherent`, `ValidateUnsigned`. It
|
|
/// is incompatible with `use_parts`. This specifies the part to exclude. In order to select
|
|
/// subset of the pallet parts.
|
|
///
|
|
/// For example excluding the part `Call` can be useful if the runtime doesn't want to make the
|
|
/// pallet calls available.
|
|
///
|
|
/// * `use_parts { Part1, Part2 }` optional: comma separated parts without generics. I.e. one of
|
|
/// `Pallet`, `Call`, `Storage`, `Event`, `Origin`, `Config`, `Inherent`, `ValidateUnsigned`. It
|
|
/// is incompatible with `exclude_parts`. This specifies the part to use. In order to select a
|
|
/// subset of the pallet parts.
|
|
///
|
|
/// For example not using the part `Call` can be useful if the runtime doesn't want to make the
|
|
/// pallet calls available.
|
|
///
|
|
/// * `= $n` optional: number to define at which index the pallet variants in `OriginCaller`, `Call`
|
|
/// and `Event` are encoded, and to define the ModuleToIndex value.
|
|
///
|
|
/// if `= $n` is not given, then index is resolved in the same way as fieldless enum in Rust
|
|
/// (i.e. incrementedly from previous index):
|
|
/// ```nocompile
|
|
/// pallet1 .. = 2,
|
|
/// pallet2 .., // Here pallet2 is given index 3
|
|
/// pallet3 .. = 0,
|
|
/// pallet4 .., // Here pallet4 is given index 1
|
|
/// ```
|
|
///
|
|
/// # Note
|
|
///
|
|
/// The population of the genesis storage depends on the order of pallets. So, if one of your
|
|
/// pallets depends on another pallet, the pallet that is depended upon needs to come before
|
|
/// the pallet depending on it.
|
|
///
|
|
/// # Type definitions
|
|
///
|
|
/// * The macro generates a type alias for each pallet to their `Pallet`. E.g. `type System =
|
|
/// frame_system::Pallet<Runtime>`
|
|
#[proc_macro]
|
|
pub fn construct_runtime(input: TokenStream) -> TokenStream {
|
|
construct_runtime::construct_runtime(input)
|
|
}
|
|
|
|
/// Macro to define a pallet. Docs are at `frame_support::pallet`.
|
|
#[proc_macro_attribute]
|
|
pub fn pallet(attr: TokenStream, item: TokenStream) -> TokenStream {
|
|
pallet::pallet(attr, item)
|
|
}
|
|
|
|
/// Execute the annotated function in a new storage transaction.
|
|
///
|
|
/// The return type of the annotated function must be `Result`. All changes to storage performed
|
|
/// by the annotated function are discarded if it returns `Err`, or committed if `Ok`.
|
|
///
|
|
/// # Example
|
|
///
|
|
/// ```nocompile
|
|
/// #[transactional]
|
|
/// fn value_commits(v: u32) -> result::Result<u32, &'static str> {
|
|
/// Value::set(v);
|
|
/// Ok(v)
|
|
/// }
|
|
///
|
|
/// #[transactional]
|
|
/// fn value_rollbacks(v: u32) -> result::Result<u32, &'static str> {
|
|
/// Value::set(v);
|
|
/// Err("nah")
|
|
/// }
|
|
/// ```
|
|
#[proc_macro_attribute]
|
|
pub fn transactional(attr: TokenStream, input: TokenStream) -> TokenStream {
|
|
transactional::transactional(attr, input).unwrap_or_else(|e| e.to_compile_error().into())
|
|
}
|
|
|
|
/// Derive [`Clone`] but do not bound any generic. Docs are at `frame_support::CloneNoBound`.
|
|
#[proc_macro_derive(CloneNoBound)]
|
|
pub fn derive_clone_no_bound(input: TokenStream) -> TokenStream {
|
|
clone_no_bound::derive_clone_no_bound(input)
|
|
}
|
|
|
|
/// Derive [`Debug`] but do not bound any generics. Docs are at `frame_support::DebugNoBound`.
|
|
#[proc_macro_derive(DebugNoBound)]
|
|
pub fn derive_debug_no_bound(input: TokenStream) -> TokenStream {
|
|
debug_no_bound::derive_debug_no_bound(input)
|
|
}
|
|
|
|
/// Derive [`Debug`], if `std` is enabled it uses `frame_support::DebugNoBound`, if `std` is not
|
|
/// enabled it just returns `"<stripped>"`.
|
|
/// This behaviour is useful to prevent bloating the runtime WASM blob from unneeded code.
|
|
#[proc_macro_derive(RuntimeDebugNoBound)]
|
|
pub fn derive_runtime_debug_no_bound(input: TokenStream) -> TokenStream {
|
|
#[cfg(not(feature = "std"))]
|
|
{
|
|
let input: syn::DeriveInput = match syn::parse(input) {
|
|
Ok(input) => input,
|
|
Err(e) => return e.to_compile_error().into(),
|
|
};
|
|
|
|
let name = &input.ident;
|
|
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
|
|
|
|
quote::quote!(
|
|
const _: () = {
|
|
impl #impl_generics core::fmt::Debug for #name #ty_generics #where_clause {
|
|
fn fmt(&self, fmt: &mut core::fmt::Formatter) -> core::fmt::Result {
|
|
fmt.write_str("<stripped>")
|
|
}
|
|
}
|
|
};
|
|
)
|
|
.into()
|
|
}
|
|
|
|
#[cfg(feature = "std")]
|
|
{
|
|
debug_no_bound::derive_debug_no_bound(input)
|
|
}
|
|
}
|
|
|
|
/// Derive [`PartialEq`] but do not bound any generic. Docs are at
|
|
/// `frame_support::PartialEqNoBound`.
|
|
#[proc_macro_derive(PartialEqNoBound)]
|
|
pub fn derive_partial_eq_no_bound(input: TokenStream) -> TokenStream {
|
|
partial_eq_no_bound::derive_partial_eq_no_bound(input)
|
|
}
|
|
|
|
/// derive Eq but do no bound any generic. Docs are at `frame_support::EqNoBound`.
|
|
#[proc_macro_derive(EqNoBound)]
|
|
pub fn derive_eq_no_bound(input: TokenStream) -> TokenStream {
|
|
let input: syn::DeriveInput = match syn::parse(input) {
|
|
Ok(input) => input,
|
|
Err(e) => return e.to_compile_error().into(),
|
|
};
|
|
|
|
let name = &input.ident;
|
|
let (impl_generics, ty_generics, where_clause) = input.generics.split_for_impl();
|
|
|
|
quote::quote_spanned!(name.span() =>
|
|
const _: () = {
|
|
impl #impl_generics core::cmp::Eq for #name #ty_generics #where_clause {}
|
|
};
|
|
)
|
|
.into()
|
|
}
|
|
|
|
/// derive `Default` but do no bound any generic. Docs are at `frame_support::DefaultNoBound`.
|
|
#[proc_macro_derive(DefaultNoBound)]
|
|
pub fn derive_default_no_bound(input: TokenStream) -> TokenStream {
|
|
default_no_bound::derive_default_no_bound(input)
|
|
}
|
|
|
|
#[proc_macro_attribute]
|
|
pub fn require_transactional(attr: TokenStream, input: TokenStream) -> TokenStream {
|
|
transactional::require_transactional(attr, input)
|
|
.unwrap_or_else(|e| e.to_compile_error().into())
|
|
}
|
|
|
|
#[proc_macro]
|
|
pub fn crate_to_crate_version(input: TokenStream) -> TokenStream {
|
|
crate_version::crate_to_crate_version(input)
|
|
.unwrap_or_else(|e| e.to_compile_error())
|
|
.into()
|
|
}
|
|
|
|
/// The number of module instances supported by the runtime, starting at index 1,
|
|
/// and up to `NUMBER_OF_INSTANCE`.
|
|
pub(crate) const NUMBER_OF_INSTANCE: u8 = 16;
|
|
|
|
/// This macro is meant to be used by frame-support only.
|
|
/// It implements the trait `HasKeyPrefix` and `HasReversibleKeyPrefix` for tuple of `Key`.
|
|
#[proc_macro]
|
|
pub fn impl_key_prefix_for_tuples(input: TokenStream) -> TokenStream {
|
|
key_prefix::impl_key_prefix_for_tuples(input)
|
|
.unwrap_or_else(syn::Error::into_compile_error)
|
|
.into()
|
|
}
|
|
|
|
/// Internal macro use by frame_support to generate dummy part checker for old pallet declaration
|
|
#[proc_macro]
|
|
pub fn __generate_dummy_part_checker(input: TokenStream) -> TokenStream {
|
|
dummy_part_checker::generate_dummy_part_checker(input)
|
|
}
|
|
|
|
/// Macro that inserts some tokens after the first match of some pattern.
|
|
///
|
|
/// # Example:
|
|
///
|
|
/// ```nocompile
|
|
/// match_and_insert!(
|
|
/// target = [{ Some content with { at some point match pattern } other match pattern are ignored }]
|
|
/// pattern = [{ match pattern }] // the match pattern cannot contain any group: `[]`, `()`, `{}`
|
|
/// // can relax this constraint, but will require modifying the match logic in code
|
|
/// tokens = [{ expansion tokens }] // content inside braces can be anything including groups
|
|
/// );
|
|
/// ```
|
|
///
|
|
/// will generate:
|
|
///
|
|
/// ```nocompile
|
|
/// Some content with { at some point match pattern expansion tokens } other match patterns are
|
|
/// ignored
|
|
/// ```
|
|
#[proc_macro]
|
|
pub fn match_and_insert(input: TokenStream) -> TokenStream {
|
|
match_and_insert::match_and_insert(input)
|
|
}
|