use {
crate::{
checks::{check_account_for_balance_with_commitment, get_fee_for_messages},
cli::CliError,
compute_budget::{simulate_and_update_compute_unit_limit, UpdateComputeUnitLimitResult},
stake,
},
clap::ArgMatches,
solana_clap_utils::{
compute_budget::ComputeUnitLimit, input_parsers::lamports_of_sol, offline::SIGN_ONLY_ARG,
},
solana_cli_output::display::build_balance_message,
solana_commitment_config::CommitmentConfig,
solana_hash::Hash,
solana_message::Message,
solana_pubkey::Pubkey,
solana_rpc_client::rpc_client::RpcClient,
};
#[derive(Debug, PartialEq, Eq, Clone, Copy)]
pub enum SpendAmount {
All,
Available,
Some(u64),
RentExempt,
AllForAccountCreation { create_account_min_balance: u64 },
}
impl Default for SpendAmount {
fn default() -> Self {
Self::Some(u64::default())
}
}
impl SpendAmount {
pub fn new(amount: Option<u64>, sign_only: bool) -> Self {
match amount {
Some(lamports) => Self::Some(lamports),
None if !sign_only => Self::All,
_ => panic!("ALL amount not supported for sign-only operations"),
}
}
pub fn new_from_matches(matches: &ArgMatches<'_>, name: &str) -> Self {
let sign_only = matches.is_present(SIGN_ONLY_ARG.name);
let amount = lamports_of_sol(matches, name);
if amount.is_some() {
return SpendAmount::new(amount, sign_only);
}
match matches.value_of(name).unwrap_or("ALL") {
"ALL" if !sign_only => SpendAmount::All,
"AVAILABLE" if !sign_only => SpendAmount::Available,
_ => panic!("Only specific amounts are supported for sign-only operations"),
}
}
}
struct SpendAndFee {
spend: u64,
fee: u64,
}
pub fn resolve_spend_tx_and_check_account_balance<F>(
rpc_client: &RpcClient,
sign_only: bool,
amount: SpendAmount,
blockhash: &Hash,
from_pubkey: &Pubkey,
compute_unit_limit: ComputeUnitLimit,
build_message: F,
commitment: CommitmentConfig,
) -> Result<(Message, u64), CliError>
where
F: Fn(u64) -> Message,
{
resolve_spend_tx_and_check_account_balances(
rpc_client,
sign_only,
amount,
blockhash,
from_pubkey,
from_pubkey,
compute_unit_limit,
build_message,
commitment,
)
}
pub fn resolve_spend_tx_and_check_account_balances<F>(
rpc_client: &RpcClient,
sign_only: bool,
amount: SpendAmount,
blockhash: &Hash,
from_pubkey: &Pubkey,
fee_pubkey: &Pubkey,
compute_unit_limit: ComputeUnitLimit,
build_message: F,
commitment: CommitmentConfig,
) -> Result<(Message, u64), CliError>
where
F: Fn(u64) -> Message,
{
if sign_only {
let (message, SpendAndFee { spend, fee: _ }) = resolve_spend_message(
rpc_client,
amount,
None,
0,
from_pubkey,
fee_pubkey,
0,
compute_unit_limit,
build_message,
)?;
Ok((message, spend))
} else {
let account = rpc_client
.get_account_with_commitment(from_pubkey, commitment)?
.value
.unwrap_or_default();
let mut from_balance = account.lamports;
let from_rent_exempt_minimum =
if amount == SpendAmount::RentExempt || amount == SpendAmount::Available {
rpc_client.get_minimum_balance_for_rent_exemption(account.data.len())?
} else {
0
};
if amount == SpendAmount::Available && account.owner == solana_sdk_ids::stake::id() {
let state = stake::get_account_stake_state(
rpc_client,
from_pubkey,
account,
true,
None,
false,
None,
)?;
let mut subtract_rent_exempt_minimum = false;
if let Some(active_stake) = state.active_stake {
from_balance = from_balance.saturating_sub(active_stake);
subtract_rent_exempt_minimum = true;
}
if let Some(activating_stake) = state.activating_stake {
from_balance = from_balance.saturating_sub(activating_stake);
subtract_rent_exempt_minimum = true;
}
if subtract_rent_exempt_minimum {
from_balance = from_balance.saturating_sub(from_rent_exempt_minimum);
}
}
let (message, SpendAndFee { spend, fee }) = resolve_spend_message(
rpc_client,
amount,
Some(blockhash),
from_balance,
from_pubkey,
fee_pubkey,
from_rent_exempt_minimum,
compute_unit_limit,
build_message,
)?;
if from_pubkey == fee_pubkey {
if from_balance == 0 || from_balance < spend.saturating_add(fee) {
return Err(CliError::InsufficientFundsForSpendAndFee(
build_balance_message(spend, false, false),
build_balance_message(fee, false, false),
*from_pubkey,
));
}
} else {
if from_balance < spend {
return Err(CliError::InsufficientFundsForSpend(
build_balance_message(spend, false, false),
*from_pubkey,
));
}
if !check_account_for_balance_with_commitment(rpc_client, fee_pubkey, fee, commitment)?
{
return Err(CliError::InsufficientFundsForFee(
build_balance_message(fee, false, false),
*fee_pubkey,
));
}
}
Ok((message, spend))
}
}
fn resolve_spend_message<F>(
rpc_client: &RpcClient,
amount: SpendAmount,
blockhash: Option<&Hash>,
from_account_transferable_balance: u64,
from_pubkey: &Pubkey,
fee_pubkey: &Pubkey,
from_rent_exempt_minimum: u64,
compute_unit_limit: ComputeUnitLimit,
build_message: F,
) -> Result<(Message, SpendAndFee), CliError>
where
F: Fn(u64) -> Message,
{
let (fee, compute_unit_info) = match blockhash {
Some(blockhash) => {
let lamports = if from_pubkey == fee_pubkey {
match amount {
SpendAmount::Some(lamports) => lamports,
SpendAmount::AllForAccountCreation {
create_account_min_balance,
} => create_account_min_balance,
SpendAmount::All | SpendAmount::Available | SpendAmount::RentExempt => 0,
}
} else {
match amount {
SpendAmount::Some(lamports) => lamports,
SpendAmount::AllForAccountCreation { .. }
| SpendAmount::All
| SpendAmount::Available => from_account_transferable_balance,
SpendAmount::RentExempt => {
from_account_transferable_balance.saturating_sub(from_rent_exempt_minimum)
}
}
};
let mut dummy_message = build_message(lamports);
dummy_message.recent_blockhash = *blockhash;
let compute_unit_info =
if let UpdateComputeUnitLimitResult::UpdatedInstructionIndex(ix_index) =
simulate_and_update_compute_unit_limit(
&compute_unit_limit,
rpc_client,
&mut dummy_message,
)?
{
Some((ix_index, dummy_message.instructions[ix_index].data.clone()))
} else {
None
};
(
get_fee_for_messages(rpc_client, &[&dummy_message])?,
compute_unit_info,
)
}
None => (0, None), };
let (mut message, spend_and_fee) = match amount {
SpendAmount::Some(lamports) => (
build_message(lamports),
SpendAndFee {
spend: lamports,
fee,
},
),
SpendAmount::All | SpendAmount::AllForAccountCreation { .. } | SpendAmount::Available => {
let lamports = if from_pubkey == fee_pubkey {
from_account_transferable_balance.saturating_sub(fee)
} else {
from_account_transferable_balance
};
(
build_message(lamports),
SpendAndFee {
spend: lamports,
fee,
},
)
}
SpendAmount::RentExempt => {
let mut lamports = if from_pubkey == fee_pubkey {
from_account_transferable_balance.saturating_sub(fee)
} else {
from_account_transferable_balance
};
lamports = lamports.saturating_sub(from_rent_exempt_minimum);
(
build_message(lamports),
SpendAndFee {
spend: lamports,
fee,
},
)
}
};
if let Some((ix_index, ix_data)) = compute_unit_info {
message.instructions[ix_index].data = ix_data;
}
Ok((message, spend_and_fee))
}