1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
use crate::{address_h256_to_erdrs, mandos_to_erdrs_address, Interactor, InteractorResult};
use log::info;
use multiversx_sc_scenario::{
    multiversx_sc::{
        codec::{multi_types::IgnoreValue, CodecFrom, TopEncodeMulti},
        types::ContractCallWithEgld,
    },
    scenario_model::{ScCallStep, TransferStep, TxCall, TypedScCall},
    DebugApi,
};
use multiversx_sdk::data::transaction::Transaction;

fn contract_call_to_tx_data(contract_call: &ContractCallWithEgld<DebugApi, ()>) -> String {
    let mut result = String::from_utf8(
        contract_call
            .basic
            .endpoint_name
            .to_boxed_bytes()
            .into_vec(),
    )
    .unwrap();
    for argument in contract_call.basic.arg_buffer.raw_arg_iter() {
        result.push('@');
        result.push_str(hex::encode(argument.to_boxed_bytes().as_slice()).as_str());
    }
    result
}

impl Interactor {
    fn tx_call_to_blockchain_tx(&self, tx_call: &TxCall) -> Transaction {
        let contract_call = tx_call.to_contract_call();
        let contract_call_tx_data = contract_call_to_tx_data(&contract_call);
        let data = if contract_call_tx_data.is_empty() {
            None
        } else {
            Some(base64::encode(contract_call_tx_data))
        };

        Transaction {
            nonce: 0,
            value: contract_call.egld_payment.to_alloc().to_string(),
            sender: mandos_to_erdrs_address(&tx_call.from),
            receiver: address_h256_to_erdrs(&contract_call.basic.to.to_address()),
            gas_price: self.network_config.min_gas_price,
            gas_limit: tx_call.gas_limit.value,
            data,
            signature: None,
            chain_id: self.network_config.chain_id.clone(),
            version: self.network_config.min_transaction_version,
            options: 0,
        }
    }

    pub async fn sc_call<S>(&mut self, sc_call_step: S) -> String
    where
        ScCallStep: From<S>,
    {
        let sc_call_step: ScCallStep = sc_call_step.into();
        let sender_address = &sc_call_step.tx.from.value;
        let mut transaction = self.tx_call_to_blockchain_tx(&sc_call_step.tx);
        self.set_nonce_and_sign_tx(sender_address, &mut transaction)
            .await;
        let tx_hash = self.proxy.send_transaction(&transaction).await.unwrap();
        println!("sc call tx hash: {tx_hash}");
        info!("sc call tx hash: {}", tx_hash);
        tx_hash
    }

    pub async fn sc_call_get_result<OriginalResult, RequestedResult>(
        &mut self,
        typed_sc_call: TypedScCall<OriginalResult>,
    ) -> InteractorResult<RequestedResult>
    where
        OriginalResult: TopEncodeMulti,
        RequestedResult: CodecFrom<OriginalResult>,
    {
        let tx_hash = self.sc_call(typed_sc_call).await;
        let tx = self.retrieve_tx_on_network(tx_hash.as_str()).await;
        InteractorResult::new(tx)
    }

    pub async fn sc_call_get_raw_result(
        &mut self,
        sc_call_step: ScCallStep,
    ) -> InteractorResult<IgnoreValue> {
        let tx_hash = self.sc_call(sc_call_step).await;
        let tx = self.retrieve_tx_on_network(tx_hash.as_str()).await;
        InteractorResult::new(tx)
    }

    pub async fn multiple_sc_calls(&mut self, sc_call_steps: &[ScCallStep]) {
        let sender_address = &sc_call_steps.get(0).unwrap().tx.from.value;
        for sc_call_step in sc_call_steps {
            assert_eq!(
                &sc_call_step.tx.from.value, sender_address,
                "all calls are expected to have the same sender"
            );
            let mut transaction = self.tx_call_to_blockchain_tx(&sc_call_step.tx);
            self.set_nonce_and_sign_tx(sender_address, &mut transaction)
                .await;
            let _ = self.proxy.send_transaction(&transaction).await.unwrap();
        }
    }

    pub async fn transfer(&mut self, transfer_step: TransferStep) -> String {
        let sender_address = &transfer_step.tx.from.value;
        let mut transaction = self.tx_call_to_blockchain_tx(&transfer_step.tx.to_tx_call());
        self.set_nonce_and_sign_tx(sender_address, &mut transaction)
            .await;
        let tx_hash = self.proxy.send_transaction(&transaction).await.unwrap();
        println!("transfer tx hash: {tx_hash}");
        info!("transfer tx hash: {}", tx_hash);
        tx_hash
    }

    pub async fn transfer_get_raw_result(
        &mut self,
        transfer_step: TransferStep,
    ) -> InteractorResult<IgnoreValue> {
        let tx_hash = self.transfer(transfer_step).await;
        let tx = self.retrieve_tx_on_network(tx_hash.as_str()).await;
        InteractorResult::new(tx)
    }
}