use super::{
display_contract_exec_result,
prompt_confirm_tx,
runtime_api::api,
state_call,
submit_extrinsic,
Balance,
BalanceVariant,
Client,
CodeHash,
ContractMessageTranscoder,
DefaultConfig,
ExtrinsicOpts,
PairSigner,
StorageDeposit,
MAX_KEY_COL_WIDTH,
};
use crate::{
cmd::extrinsics::{
display_contract_exec_result_debug,
events::DisplayEvents,
ErrorVariant,
TokenMetadata,
},
DEFAULT_KEY_COL_WIDTH,
};
use anyhow::{
anyhow,
Result,
};
use contract_build::{
name_value_println,
util::decode_hex,
Verbosity,
};
use pallet_contracts_primitives::ContractInstantiateResult;
use scale::Encode;
use sp_core::{
crypto::Ss58Codec,
Bytes,
};
use sp_weights::Weight;
use subxt::{
blocks::ExtrinsicEvents,
Config,
OnlineClient,
};
#[derive(Debug, clap::Args)]
pub struct InstantiateCommand {
#[clap(name = "constructor", long, default_value = "new")]
constructor: String,
#[clap(long, num_args = 0..)]
args: Vec<String>,
#[clap(flatten)]
extrinsic_opts: ExtrinsicOpts,
#[clap(name = "value", long, default_value = "0")]
value: BalanceVariant,
#[clap(name = "gas", long)]
gas_limit: Option<u64>,
#[clap(long)]
proof_size: Option<u64>,
#[clap(long, value_parser = parse_hex_bytes)]
salt: Option<Bytes>,
#[clap(long, conflicts_with = "verbose")]
output_json: bool,
}
fn parse_hex_bytes(input: &str) -> Result<Bytes> {
let bytes = decode_hex(input)?;
Ok(bytes.into())
}
impl InstantiateCommand {
pub fn is_json(&self) -> bool {
self.output_json
}
pub fn run(&self) -> Result<(), ErrorVariant> {
let artifacts = self.extrinsic_opts.contract_artifacts()?;
let transcoder = artifacts.contract_transcoder()?;
let data = transcoder.encode(&self.constructor, &self.args)?;
let signer = super::pair_signer(self.extrinsic_opts.signer()?);
let url = self.extrinsic_opts.url_to_string();
let verbosity = self.extrinsic_opts.verbosity()?;
let code = if let Some(code) = artifacts.code {
Code::Upload(code.0)
} else {
let code_hash = artifacts.code_hash()?;
Code::Existing(code_hash.into())
};
let salt = self.salt.clone().map(|s| s.0).unwrap_or_default();
async_std::task::block_on(async move {
let client = OnlineClient::from_url(url.clone()).await?;
let token_metadata = TokenMetadata::query(&client).await?;
let args = InstantiateArgs {
constructor: self.constructor.clone(),
raw_args: self.args.clone(),
value: self.value.denominate_balance(&token_metadata)?,
gas_limit: self.gas_limit,
proof_size: self.proof_size,
storage_deposit_limit: self
.extrinsic_opts
.storage_deposit_limit
.as_ref()
.map(|bv| bv.denominate_balance(&token_metadata))
.transpose()?,
code,
data,
salt,
};
let exec = Exec {
args,
opts: self.extrinsic_opts.clone(),
url,
client,
verbosity,
signer,
transcoder,
output_json: self.output_json,
};
exec.exec(self.extrinsic_opts.dry_run).await
})
}
}
struct InstantiateArgs {
constructor: String,
raw_args: Vec<String>,
value: Balance,
gas_limit: Option<u64>,
proof_size: Option<u64>,
storage_deposit_limit: Option<Balance>,
code: Code,
data: Vec<u8>,
salt: Vec<u8>,
}
impl InstantiateArgs {
fn storage_deposit_limit_compact(&self) -> Option<scale::Compact<Balance>> {
self.storage_deposit_limit.map(Into::into)
}
}
pub struct Exec {
opts: ExtrinsicOpts,
args: InstantiateArgs,
verbosity: Verbosity,
url: String,
client: Client,
signer: PairSigner,
transcoder: ContractMessageTranscoder,
output_json: bool,
}
impl Exec {
async fn exec(&self, dry_run: bool) -> Result<(), ErrorVariant> {
tracing::debug!("instantiate data {:?}", self.args.data);
if dry_run {
let result = self.instantiate_dry_run().await?;
match result.result {
Ok(ref ret_val) => {
let dry_run_result = InstantiateDryRunResult {
result: String::from("Success!"),
contract: ret_val.account_id.to_ss58check(),
reverted: ret_val.result.did_revert(),
data: ret_val.result.data.clone().into(),
gas_consumed: result.gas_consumed,
gas_required: result.gas_required,
storage_deposit: StorageDeposit::from(&result.storage_deposit),
};
if self.output_json {
println!("{}", dry_run_result.to_json()?);
Ok(())
} else {
dry_run_result.print();
display_contract_exec_result_debug::<_, DEFAULT_KEY_COL_WIDTH>(
&result,
)?;
Ok(())
}
}
Err(ref err) => {
let metadata = self.client.metadata();
let object = ErrorVariant::from_dispatch_error(err, &metadata)?;
if self.output_json {
Err(object)
} else {
name_value_println!("Result", object, MAX_KEY_COL_WIDTH);
display_contract_exec_result::<_, MAX_KEY_COL_WIDTH>(&result)?;
Ok(())
}
}
}
} else {
let gas_limit = self.pre_submit_dry_run_gas_estimate().await?;
match self.args.code.clone() {
Code::Upload(code) => {
self.instantiate_with_code(code, gas_limit).await?;
}
Code::Existing(code_hash) => {
self.instantiate(code_hash, gas_limit).await?;
}
}
Ok(())
}
}
async fn instantiate_with_code(
&self,
code: Vec<u8>,
gas_limit: Weight,
) -> Result<(), ErrorVariant> {
if !self.opts.skip_confirm {
prompt_confirm_tx(|| self.print_default_instantiate_preview(gas_limit))?;
}
let call = api::tx().contracts().instantiate_with_code(
self.args.value,
gas_limit,
self.args.storage_deposit_limit_compact(),
code.to_vec(),
self.args.data.clone(),
self.args.salt.clone(),
);
let result = submit_extrinsic(&self.client, &call, &self.signer).await?;
let code_hash = result
.find_first::<api::contracts::events::CodeStored>()?
.map(|code_stored| code_stored.code_hash);
let instantiated = result
.find_first::<api::contracts::events::Instantiated>()?
.ok_or_else(|| anyhow!("Failed to find Instantiated event"))?;
let token_metadata = TokenMetadata::query(&self.client).await?;
self.display_result(&result, code_hash, instantiated.contract, &token_metadata)
.await
}
async fn instantiate(
&self,
code_hash: CodeHash,
gas_limit: Weight,
) -> Result<(), ErrorVariant> {
if !self.opts.skip_confirm {
prompt_confirm_tx(|| {
self.print_default_instantiate_preview(gas_limit);
name_value_println!(
"Code hash",
format!("{:?}", code_hash),
DEFAULT_KEY_COL_WIDTH
);
})?;
}
let call = api::tx().contracts().instantiate(
self.args.value,
gas_limit,
self.args.storage_deposit_limit_compact(),
code_hash,
self.args.data.clone(),
self.args.salt.clone(),
);
let result = submit_extrinsic(&self.client, &call, &self.signer).await?;
let instantiated = result
.find_first::<api::contracts::events::Instantiated>()?
.ok_or_else(|| anyhow!("Failed to find Instantiated event"))?;
let token_metadata = TokenMetadata::query(&self.client).await?;
self.display_result(&result, None, instantiated.contract, &token_metadata)
.await
}
async fn display_result(
&self,
result: &ExtrinsicEvents<DefaultConfig>,
code_hash: Option<CodeHash>,
contract_address: sp_core::crypto::AccountId32,
token_metadata: &TokenMetadata,
) -> Result<(), ErrorVariant> {
let events = DisplayEvents::from_events(
result,
Some(&self.transcoder),
&self.client.metadata(),
)?;
let contract_address = contract_address.to_ss58check();
if self.output_json {
let display_instantiate_result = InstantiateResult {
code_hash: code_hash.map(|ch| format!("{:?}", ch)),
contract: Some(contract_address),
events,
};
println!("{}", display_instantiate_result.to_json()?)
} else {
if let Some(code_hash) = code_hash {
name_value_println!("Code hash", format!("{:?}", code_hash));
}
name_value_println!("Contract", contract_address);
println!("{}", events.display_events(self.verbosity, token_metadata)?)
};
Ok(())
}
fn print_default_instantiate_preview(&self, gas_limit: Weight) {
name_value_println!("Constructor", self.args.constructor, DEFAULT_KEY_COL_WIDTH);
name_value_println!("Args", self.args.raw_args.join(" "), DEFAULT_KEY_COL_WIDTH);
name_value_println!("Gas limit", gas_limit.to_string(), DEFAULT_KEY_COL_WIDTH);
}
async fn instantiate_dry_run(
&self,
) -> Result<ContractInstantiateResult<<DefaultConfig as Config>::AccountId, Balance>>
{
let storage_deposit_limit = self.args.storage_deposit_limit;
let call_request = InstantiateRequest {
origin: self.signer.account_id().clone(),
value: self.args.value,
gas_limit: None,
storage_deposit_limit,
code: self.args.code.clone(),
data: self.args.data.clone(),
salt: self.args.salt.clone(),
};
state_call(&self.url, "ContractsApi_instantiate", &call_request).await
}
async fn pre_submit_dry_run_gas_estimate(&self) -> Result<Weight> {
if self.opts.skip_dry_run {
return match (self.args.gas_limit, self.args.proof_size) {
(Some(ref_time), Some(proof_size)) => Ok(Weight::from_parts(ref_time, proof_size)),
_ => {
Err(anyhow!(
"Weight args `--gas` and `--proof-size` required if `--skip-dry-run` specified"
))
}
}
}
if !self.output_json {
super::print_dry_running_status(&self.args.constructor);
}
let instantiate_result = self.instantiate_dry_run().await?;
match instantiate_result.result {
Ok(_) => {
if !self.output_json {
super::print_gas_required_success(instantiate_result.gas_required);
}
let ref_time = self
.args
.gas_limit
.unwrap_or_else(|| instantiate_result.gas_required.ref_time());
let proof_size = self
.args
.proof_size
.unwrap_or_else(|| instantiate_result.gas_required.proof_size());
Ok(Weight::from_parts(ref_time, proof_size))
}
Err(ref err) => {
let object =
ErrorVariant::from_dispatch_error(err, &self.client.metadata())?;
if self.output_json {
Err(anyhow!("{}", serde_json::to_string_pretty(&object)?))
} else {
name_value_println!("Result", object, MAX_KEY_COL_WIDTH);
display_contract_exec_result::<_, MAX_KEY_COL_WIDTH>(
&instantiate_result,
)?;
Err(anyhow!("Pre-submission dry-run failed. Use --skip-dry-run to skip this step."))
}
}
}
}
}
#[derive(serde::Serialize)]
pub struct InstantiateResult {
#[serde(skip_serializing_if = "Option::is_none")]
contract: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
code_hash: Option<String>,
events: DisplayEvents,
}
impl InstantiateResult {
pub fn to_json(&self) -> Result<String> {
Ok(serde_json::to_string_pretty(self)?)
}
}
#[derive(serde::Serialize)]
pub struct InstantiateDryRunResult {
pub result: String,
pub contract: String,
pub reverted: bool,
pub data: Bytes,
pub gas_consumed: Weight,
pub gas_required: Weight,
pub storage_deposit: StorageDeposit,
}
impl InstantiateDryRunResult {
pub fn to_json(&self) -> Result<String> {
Ok(serde_json::to_string_pretty(self)?)
}
pub fn print(&self) {
name_value_println!("Result", self.result, DEFAULT_KEY_COL_WIDTH);
name_value_println!("Contract", self.contract, DEFAULT_KEY_COL_WIDTH);
name_value_println!(
"Reverted",
format!("{:?}", self.reverted),
DEFAULT_KEY_COL_WIDTH
);
name_value_println!("Data", format!("{:?}", self.data), DEFAULT_KEY_COL_WIDTH);
}
}
#[derive(Encode)]
struct InstantiateRequest {
origin: <DefaultConfig as Config>::AccountId,
value: Balance,
gas_limit: Option<Weight>,
storage_deposit_limit: Option<Balance>,
code: Code,
data: Vec<u8>,
salt: Vec<u8>,
}
#[derive(Clone, Encode)]
enum Code {
Upload(Vec<u8>),
Existing(<DefaultConfig as Config>::Hash),
}