use super::{deposit_limit, GAS_LIMIT};
use crate::{
address::AddressMapper, AccountIdOf, BalanceOf, Code, CollectEvents, Config, ContractResult,
DebugInfo, DepositLimit, EventRecordOf, ExecReturnValue, InstantiateReturnValue, OriginFor,
Pallet, Weight,
};
use frame_support::pallet_prelude::DispatchResultWithPostInfo;
use paste::paste;
use sp_core::H160;
macro_rules! builder {
(
$method:ident($($field:ident: $type:ty,)*) -> $result:ty;
$($extra:item)*
) => {
paste!{
builder!([< $method:camel Builder >], $method($($field: $type,)* ) -> $result; $($extra)*);
}
};
(
$name:ident,
$method:ident($($field:ident: $type:ty,)*) -> $result:ty;
$($extra:item)*
) => {
#[doc = concat!("A builder to construct a ", stringify!($method), " call")]
pub struct $name<T: Config> {
$($field: $type,)*
}
#[allow(dead_code)]
impl<T: Config> $name<T>
where
BalanceOf<T>: Into<sp_core::U256> + TryFrom<sp_core::U256>,
crate::MomentOf<T>: Into<sp_core::U256>,
T::Hash: frame_support::traits::IsType<sp_core::H256>,
{
$(
#[doc = concat!("Set the ", stringify!($field))]
pub fn $field(mut self, value: $type) -> Self {
self.$field = value;
self
}
)*
#[doc = concat!("Build the ", stringify!($method), " call")]
pub fn build(self) -> $result {
Pallet::<T>::$method(
$(self.$field,)*
)
}
$($extra)*
}
}
}
pub struct Contract<T: Config> {
pub account_id: AccountIdOf<T>,
pub addr: H160,
}
builder!(
instantiate_with_code(
origin: OriginFor<T>,
value: BalanceOf<T>,
gas_limit: Weight,
storage_deposit_limit: BalanceOf<T>,
code: Vec<u8>,
data: Vec<u8>,
salt: Option<[u8; 32]>,
) -> DispatchResultWithPostInfo;
pub fn instantiate_with_code(origin: OriginFor<T>, code: Vec<u8>) -> Self {
Self {
origin,
value: 0u32.into(),
gas_limit: GAS_LIMIT,
storage_deposit_limit: deposit_limit::<T>(),
code,
data: vec![],
salt: Some([0; 32]),
}
}
);
builder!(
instantiate(
origin: OriginFor<T>,
value: BalanceOf<T>,
gas_limit: Weight,
storage_deposit_limit: BalanceOf<T>,
code_hash: sp_core::H256,
data: Vec<u8>,
salt: Option<[u8; 32]>,
) -> DispatchResultWithPostInfo;
pub fn instantiate(origin: OriginFor<T>, code_hash: sp_core::H256) -> Self {
Self {
origin,
value: 0u32.into(),
gas_limit: GAS_LIMIT,
storage_deposit_limit: deposit_limit::<T>(),
code_hash,
data: vec![],
salt: Some([0; 32]),
}
}
);
builder!(
bare_instantiate(
origin: OriginFor<T>,
value: BalanceOf<T>,
gas_limit: Weight,
storage_deposit_limit: DepositLimit<BalanceOf<T>>,
code: Code,
data: Vec<u8>,
salt: Option<[u8; 32]>,
debug: DebugInfo,
collect_events: CollectEvents,
) -> ContractResult<InstantiateReturnValue, BalanceOf<T>, EventRecordOf<T>>;
pub fn build_and_unwrap_result(self) -> InstantiateReturnValue {
self.build().result.unwrap()
}
pub fn build_and_unwrap_contract(self) -> Contract<T> {
let addr = self.build().result.unwrap().addr;
let account_id = T::AddressMapper::to_account_id(&addr);
Contract{ account_id, addr }
}
pub fn bare_instantiate(origin: OriginFor<T>, code: Code) -> Self {
Self {
origin,
value: 0u32.into(),
gas_limit: GAS_LIMIT,
storage_deposit_limit: DepositLimit::Balance(deposit_limit::<T>()),
code,
data: vec![],
salt: Some([0; 32]),
debug: DebugInfo::UnsafeDebug,
collect_events: CollectEvents::Skip,
}
}
);
builder!(
call(
origin: OriginFor<T>,
dest: H160,
value: BalanceOf<T>,
gas_limit: Weight,
storage_deposit_limit: BalanceOf<T>,
data: Vec<u8>,
) -> DispatchResultWithPostInfo;
pub fn call(origin: OriginFor<T>, dest: H160) -> Self {
CallBuilder {
origin,
dest,
value: 0u32.into(),
gas_limit: GAS_LIMIT,
storage_deposit_limit: deposit_limit::<T>(),
data: vec![],
}
}
);
builder!(
bare_call(
origin: OriginFor<T>,
dest: H160,
value: BalanceOf<T>,
gas_limit: Weight,
storage_deposit_limit: DepositLimit<BalanceOf<T>>,
data: Vec<u8>,
debug: DebugInfo,
collect_events: CollectEvents,
) -> ContractResult<ExecReturnValue, BalanceOf<T>, EventRecordOf<T>>;
pub fn build_and_unwrap_result(self) -> ExecReturnValue {
self.build().result.unwrap()
}
pub fn bare_call(origin: OriginFor<T>, dest: H160) -> Self {
Self {
origin,
dest,
value: 0u32.into(),
gas_limit: GAS_LIMIT,
storage_deposit_limit: DepositLimit::Balance(deposit_limit::<T>()),
data: vec![],
debug: DebugInfo::UnsafeDebug,
collect_events: CollectEvents::Skip,
}
}
);