use std::collections::HashMap;
use async_trait::async_trait;
use fuel_core_client::client::pagination::{PaginatedResult, PaginationRequest};
use fuel_tx::{Output, Receipt, TxId, TxPointer, UtxoId};
use fuel_types::{AssetId, Bytes32, ContractId, Nonce};
use fuels_core::types::{
bech32::{Bech32Address, Bech32ContractId},
coin::Coin,
coin_type::CoinType,
coin_type_id::CoinTypeId,
errors::Result,
input::Input,
message::Message,
transaction::{Transaction, TxPolicies},
transaction_builders::{BuildableTransaction, ScriptTransactionBuilder, TransactionBuilder},
transaction_response::TransactionResponse,
};
use crate::{
accounts_utils::{
add_base_change_if_needed, available_base_assets_and_amount, calculate_missing_base_amount,
extract_message_nonce, split_into_utxo_ids_and_nonces,
},
provider::{Provider, ResourceFilter},
};
#[cfg_attr(not(target_arch = "wasm32"), async_trait)]
pub trait ViewOnlyAccount: std::fmt::Debug + Send + Sync + Clone {
fn address(&self) -> &Bech32Address;
fn try_provider(&self) -> Result<&Provider>;
async fn get_transactions(
&self,
request: PaginationRequest<String>,
) -> Result<PaginatedResult<TransactionResponse, String>> {
Ok(self
.try_provider()?
.get_transactions_by_owner(self.address(), request)
.await?)
}
async fn get_coins(&self, asset_id: AssetId) -> Result<Vec<Coin>> {
Ok(self
.try_provider()?
.get_coins(self.address(), asset_id)
.await?)
}
async fn get_asset_balance(&self, asset_id: &AssetId) -> Result<u64> {
self.try_provider()?
.get_asset_balance(self.address(), *asset_id)
.await
}
async fn get_messages(&self) -> Result<Vec<Message>> {
Ok(self.try_provider()?.get_messages(self.address()).await?)
}
async fn get_balances(&self) -> Result<HashMap<String, u128>> {
self.try_provider()?.get_balances(self.address()).await
}
async fn get_spendable_resources(
&self,
asset_id: AssetId,
amount: u64,
excluded_coins: Option<Vec<CoinTypeId>>,
) -> Result<Vec<CoinType>> {
let (excluded_utxos, excluded_message_nonces) =
split_into_utxo_ids_and_nonces(excluded_coins);
let filter = ResourceFilter {
from: self.address().clone(),
asset_id: Some(asset_id),
amount,
excluded_utxos,
excluded_message_nonces,
};
self.try_provider()?.get_spendable_resources(filter).await
}
fn get_asset_outputs_for_amount(
&self,
to: &Bech32Address,
asset_id: AssetId,
amount: u64,
) -> Vec<Output> {
vec![
Output::coin(to.into(), amount, asset_id),
Output::change(self.address().into(), 0, asset_id),
]
}
async fn get_asset_inputs_for_amount(
&self,
asset_id: AssetId,
amount: u64,
excluded_coins: Option<Vec<CoinTypeId>>,
) -> Result<Vec<Input>>;
async fn adjust_for_fee<Tb: TransactionBuilder + Sync>(
&self,
tb: &mut Tb,
used_base_amount: u64,
) -> Result<()> {
let provider = self.try_provider()?;
let consensus_parameters = provider.consensus_parameters().await?;
let (base_assets, base_amount) =
available_base_assets_and_amount(tb, consensus_parameters.base_asset_id());
let missing_base_amount =
calculate_missing_base_amount(tb, base_amount, used_base_amount, provider).await?;
if missing_base_amount > 0 {
let new_base_inputs = self
.get_asset_inputs_for_amount(
*consensus_parameters.base_asset_id(),
missing_base_amount,
Some(base_assets),
)
.await
.unwrap_or_default();
tb.inputs_mut().extend(new_base_inputs);
};
add_base_change_if_needed(tb, self.address(), consensus_parameters.base_asset_id());
Ok(())
}
}
#[cfg_attr(not(target_arch = "wasm32"), async_trait)]
pub trait Account: ViewOnlyAccount {
fn add_witnesses<Tb: TransactionBuilder>(&self, _tb: &mut Tb) -> Result<()> {
Ok(())
}
async fn transfer(
&self,
to: &Bech32Address,
amount: u64,
asset_id: AssetId,
tx_policies: TxPolicies,
) -> Result<(TxId, Vec<Receipt>)> {
let provider = self.try_provider()?;
let inputs = self
.get_asset_inputs_for_amount(asset_id, amount, None)
.await?;
let outputs = self.get_asset_outputs_for_amount(to, asset_id, amount);
let mut tx_builder =
ScriptTransactionBuilder::prepare_transfer(inputs, outputs, tx_policies);
self.add_witnesses(&mut tx_builder)?;
let consensus_parameters = provider.consensus_parameters().await?;
let used_base_amount = if asset_id == *consensus_parameters.base_asset_id() {
amount
} else {
0
};
self.adjust_for_fee(&mut tx_builder, used_base_amount)
.await?;
let tx = tx_builder.build(provider).await?;
let tx_id = tx.id(consensus_parameters.chain_id());
let tx_status = provider.send_transaction_and_await_commit(tx).await?;
let receipts = tx_status.take_receipts_checked(None)?;
Ok((tx_id, receipts))
}
async fn force_transfer_to_contract(
&self,
to: &Bech32ContractId,
balance: u64,
asset_id: AssetId,
tx_policies: TxPolicies,
) -> Result<(String, Vec<Receipt>)> {
let provider = self.try_provider()?;
let zeroes = Bytes32::zeroed();
let plain_contract_id: ContractId = to.into();
let mut inputs = vec![Input::contract(
UtxoId::new(zeroes, 0),
zeroes,
zeroes,
TxPointer::default(),
plain_contract_id,
)];
inputs.extend(
self.get_asset_inputs_for_amount(asset_id, balance, None)
.await?,
);
let outputs = vec![
Output::contract(0, zeroes, zeroes),
Output::change(self.address().into(), 0, asset_id),
];
let mut tb = ScriptTransactionBuilder::prepare_contract_transfer(
plain_contract_id,
balance,
asset_id,
inputs,
outputs,
tx_policies,
);
self.add_witnesses(&mut tb)?;
self.adjust_for_fee(&mut tb, balance).await?;
let tx = tb.build(provider).await?;
let consensus_parameters = provider.consensus_parameters().await?;
let tx_id = tx.id(consensus_parameters.chain_id());
let tx_status = provider.send_transaction_and_await_commit(tx).await?;
let receipts = tx_status.take_receipts_checked(None)?;
Ok((tx_id.to_string(), receipts))
}
async fn withdraw_to_base_layer(
&self,
to: &Bech32Address,
amount: u64,
tx_policies: TxPolicies,
) -> Result<(TxId, Nonce, Vec<Receipt>)> {
let provider = self.try_provider()?;
let consensus_parameters = provider.consensus_parameters().await?;
let inputs = self
.get_asset_inputs_for_amount(*consensus_parameters.base_asset_id(), amount, None)
.await?;
let mut tb = ScriptTransactionBuilder::prepare_message_to_output(
to.into(),
amount,
inputs,
tx_policies,
*consensus_parameters.base_asset_id(),
);
self.add_witnesses(&mut tb)?;
self.adjust_for_fee(&mut tb, amount).await?;
let tx = tb.build(provider).await?;
let tx_id = tx.id(consensus_parameters.chain_id());
let tx_status = provider.send_transaction_and_await_commit(tx).await?;
let receipts = tx_status.take_receipts_checked(None)?;
let nonce = extract_message_nonce(&receipts)
.expect("MessageId could not be retrieved from tx receipts.");
Ok((tx_id, nonce, receipts))
}
}
#[cfg(test)]
mod tests {
use std::str::FromStr;
use fuel_crypto::{Message, SecretKey, Signature};
use fuel_tx::{Address, ConsensusParameters, Output, Transaction as FuelTransaction};
use fuels_core::{
traits::Signer,
types::{transaction::Transaction, DryRun, DryRunner},
};
use rand::{rngs::StdRng, RngCore, SeedableRng};
use super::*;
use crate::wallet::WalletUnlocked;
#[tokio::test]
async fn sign_and_verify() -> Result<()> {
let mut rng = StdRng::seed_from_u64(2322u64);
let mut secret_seed = [0u8; 32];
rng.fill_bytes(&mut secret_seed);
let secret = secret_seed.as_slice().try_into()?;
let wallet = WalletUnlocked::new_from_private_key(secret, None);
let message = Message::new("my message".as_bytes());
let signature = wallet.sign(message).await?;
assert_eq!(signature, Signature::from_str("0x8eeb238db1adea4152644f1cd827b552dfa9ab3f4939718bb45ca476d167c6512a656f4d4c7356bfb9561b14448c230c6e7e4bd781df5ee9e5999faa6495163d")?);
let recovered_address = signature.recover(&message)?;
assert_eq!(wallet.address().hash(), recovered_address.hash());
signature.verify(&recovered_address, &message)?;
Ok(())
}
#[derive(Default)]
struct MockDryRunner {
c_param: ConsensusParameters,
}
#[cfg_attr(not(target_arch = "wasm32"), async_trait)]
impl DryRunner for MockDryRunner {
async fn dry_run(&self, _: FuelTransaction) -> Result<DryRun> {
Ok(DryRun {
succeeded: true,
script_gas: 0,
variable_outputs: 0,
})
}
async fn consensus_parameters(&self) -> Result<ConsensusParameters> {
Ok(self.c_param.clone())
}
async fn estimate_gas_price(&self, _block_header: u32) -> Result<u64> {
Ok(0)
}
async fn estimate_predicates(
&self,
_: &FuelTransaction,
_: Option<u32>,
) -> Result<FuelTransaction> {
unimplemented!()
}
}
#[tokio::test]
async fn sign_tx_and_verify() -> std::result::Result<(), Box<dyn std::error::Error>> {
let secret = SecretKey::from_str(
"5f70feeff1f229e4a95e1056e8b4d80d0b24b565674860cc213bdb07127ce1b1",
)?;
let wallet = WalletUnlocked::new_from_private_key(secret, None);
let mut tb = {
let input_coin = Input::ResourceSigned {
resource: CoinType::Coin(Coin {
amount: 10000000,
owner: wallet.address().clone(),
..Default::default()
}),
};
let output_coin = Output::coin(
Address::from_str(
"0xc7862855b418ba8f58878db434b21053a61a2025209889cc115989e8040ff077",
)?,
1,
Default::default(),
);
let change = Output::change(wallet.address().into(), 0, Default::default());
ScriptTransactionBuilder::prepare_transfer(
vec![input_coin],
vec![output_coin, change],
Default::default(),
)
};
tb.add_signer(wallet.clone())?;
let tx = tb.build(MockDryRunner::default()).await?; let bytes = <[u8; Signature::LEN]>::try_from(tx.witnesses().first().unwrap().as_ref())?;
let tx_signature = Signature::from_bytes(bytes);
let message = Message::from_bytes(*tx.id(0.into()));
let signature = wallet.sign(message).await?;
assert_eq!(signature, tx_signature);
assert_eq!(signature, Signature::from_str("faa616776a1c336ef6257f7cb0cb5cd932180e2d15faba5f17481dae1cbcaf314d94617bd900216a6680bccb1ea62438e4ca93b0d5733d33788ef9d79cc24e9f")?);
let recovered_address = signature.recover(&message)?;
assert_eq!(wallet.address().hash(), recovered_address.hash());
signature.verify(&recovered_address, &message)?;
Ok(())
}
}