use std::sync::Arc;
use axum::extract::State;
use axum::{Extension, Json};
use axum_extra::extract::WithRejection;
use fastcrypto::encoding::{Encoding, Hex};
use fastcrypto::hash::HashFunction;
use futures::StreamExt;
use shared_crypto::intent::{Intent, IntentMessage};
use sui_json_rpc_types::{
StakeStatus, SuiObjectDataOptions, SuiTransactionBlockEffectsAPI,
SuiTransactionBlockResponseOptions,
};
use sui_sdk::rpc_types::SuiExecutionStatus;
use sui_types::base_types::{ObjectRef, SuiAddress};
use sui_types::crypto::{DefaultHash, SignatureScheme, ToFromBytes};
use sui_types::error::SuiError;
use sui_types::signature::{GenericSignature, VerifyParams};
use sui_types::signature_verification::{
verify_sender_signed_data_message_signatures, VerifiedDigestCache,
};
use sui_types::transaction::{Transaction, TransactionData, TransactionDataAPI};
use crate::errors::Error;
use crate::types::{
Amount, ConstructionCombineRequest, ConstructionCombineResponse, ConstructionDeriveRequest,
ConstructionDeriveResponse, ConstructionHashRequest, ConstructionMetadata,
ConstructionMetadataRequest, ConstructionMetadataResponse, ConstructionParseRequest,
ConstructionParseResponse, ConstructionPayloadsRequest, ConstructionPayloadsResponse,
ConstructionPreprocessRequest, ConstructionPreprocessResponse, ConstructionSubmitRequest,
InternalOperation, MetadataOptions, SignatureType, SigningPayload, TransactionIdentifier,
TransactionIdentifierResponse,
};
use crate::{OnlineServerContext, SuiEnv};
pub async fn derive(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionDeriveRequest>, Error>,
) -> Result<ConstructionDeriveResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let address: SuiAddress = request.public_key.try_into()?;
Ok(ConstructionDeriveResponse {
account_identifier: address.into(),
})
}
pub async fn payloads(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionPayloadsRequest>, Error>,
) -> Result<ConstructionPayloadsResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let metadata = request.metadata.ok_or(Error::MissingMetadata)?;
let address = metadata.sender;
let data = request
.operations
.into_internal()?
.try_into_data(metadata)?;
let intent_msg = IntentMessage::new(Intent::sui_transaction(), data);
let intent_msg_bytes = bcs::to_bytes(&intent_msg)?;
let mut hasher = DefaultHash::default();
hasher.update(bcs::to_bytes(&intent_msg).expect("Message serialization should not fail"));
let digest = hasher.finalize().digest;
Ok(ConstructionPayloadsResponse {
unsigned_transaction: Hex::from_bytes(&intent_msg_bytes),
payloads: vec![SigningPayload {
account_identifier: address.into(),
hex_bytes: Hex::encode(digest),
signature_type: Some(SignatureType::Ed25519),
}],
})
}
pub async fn combine(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionCombineRequest>, Error>,
) -> Result<ConstructionCombineResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let unsigned_tx = request.unsigned_transaction.to_vec()?;
let intent_msg: IntentMessage<TransactionData> = bcs::from_bytes(&unsigned_tx)?;
let sig = request
.signatures
.first()
.ok_or_else(|| Error::MissingInput("Signature".to_string()))?;
let sig_bytes = sig.hex_bytes.to_vec()?;
let pub_key = sig.public_key.hex_bytes.to_vec()?;
let flag = vec![match sig.signature_type {
SignatureType::Ed25519 => SignatureScheme::ED25519,
SignatureType::Ecdsa => SignatureScheme::Secp256k1,
}
.flag()];
let signed_tx = Transaction::from_generic_sig_data(
intent_msg.value,
vec![GenericSignature::from_bytes(
&[&*flag, &*sig_bytes, &*pub_key].concat(),
)?],
);
let place_holder_epoch = 0;
verify_sender_signed_data_message_signatures(
&signed_tx,
place_holder_epoch,
&VerifyParams::default(),
Arc::new(VerifiedDigestCache::new_empty()), )?;
let signed_tx_bytes = bcs::to_bytes(&signed_tx)?;
Ok(ConstructionCombineResponse {
signed_transaction: Hex::from_bytes(&signed_tx_bytes),
})
}
pub async fn submit(
State(context): State<OnlineServerContext>,
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionSubmitRequest>, Error>,
) -> Result<TransactionIdentifierResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let signed_tx: Transaction = bcs::from_bytes(&request.signed_transaction.to_vec()?)?;
let tx_data = signed_tx.data().transaction_data().clone();
let dry_run = context
.client
.read_api()
.dry_run_transaction_block(tx_data)
.await?;
if let SuiExecutionStatus::Failure { error } = dry_run.effects.status() {
return Err(Error::TransactionDryRunError(error.clone()));
};
let response = context
.client
.quorum_driver_api()
.execute_transaction_block(
signed_tx,
SuiTransactionBlockResponseOptions::new()
.with_input()
.with_effects()
.with_balance_changes(),
None,
)
.await?;
if let SuiExecutionStatus::Failure { error } = response
.effects
.expect("Execute transaction should return effects")
.status()
{
return Err(Error::TransactionExecutionError(error.to_string()));
}
Ok(TransactionIdentifierResponse {
transaction_identifier: TransactionIdentifier {
hash: response.digest,
},
metadata: None,
})
}
pub async fn preprocess(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionPreprocessRequest>, Error>,
) -> Result<ConstructionPreprocessResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let internal_operation = request.operations.into_internal()?;
let sender = internal_operation.sender();
let budget = request.metadata.and_then(|m| m.budget);
Ok(ConstructionPreprocessResponse {
options: Some(MetadataOptions {
internal_operation,
budget,
}),
required_public_keys: vec![sender.into()],
})
}
pub async fn hash(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionHashRequest>, Error>,
) -> Result<TransactionIdentifierResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let tx_bytes = request.signed_transaction.to_vec()?;
let tx: Transaction = bcs::from_bytes(&tx_bytes)?;
Ok(TransactionIdentifierResponse {
transaction_identifier: TransactionIdentifier { hash: *tx.digest() },
metadata: None,
})
}
pub async fn metadata(
State(context): State<OnlineServerContext>,
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionMetadataRequest>, Error>,
) -> Result<ConstructionMetadataResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let option = request.options.ok_or(Error::MissingMetadata)?;
let budget = option.budget;
let sender = option.internal_operation.sender();
let currency = match &option.internal_operation {
InternalOperation::PayCoin { currency, .. } => Some(currency.clone()),
_ => None,
};
let coin_type = currency.as_ref().map(|c| c.metadata.coin_type.clone());
let mut gas_price = context
.client
.governance_api()
.get_reference_gas_price()
.await?;
gas_price += 100;
let (total_required_amount, objects) = match &option.internal_operation {
InternalOperation::PaySui { amounts, .. } => {
let amount = amounts.iter().sum::<u64>();
(Some(amount), vec![])
}
InternalOperation::PayCoin { amounts, .. } => {
let amount = amounts.iter().sum::<u64>();
let coin_objs: Vec<ObjectRef> = context
.client
.coin_read_api()
.select_coins(sender, coin_type, amount.into(), vec![])
.await
.ok()
.unwrap_or_default()
.iter()
.map(|coin| coin.object_ref())
.collect();
(Some(0), coin_objs) }
InternalOperation::Stake { amount, .. } => (*amount, vec![]),
InternalOperation::WithdrawStake { sender, stake_ids } => {
let stake_ids = if stake_ids.is_empty() {
context
.client
.governance_api()
.get_stakes(*sender)
.await?
.into_iter()
.flat_map(|s| {
s.stakes.into_iter().filter_map(|s| {
if let StakeStatus::Active { .. } = s.status {
Some(s.staked_sui_id)
} else {
None
}
})
})
.collect()
} else {
stake_ids.clone()
};
if stake_ids.is_empty() {
return Err(Error::InvalidInput("No active stake to withdraw".into()));
}
let responses = context
.client
.read_api()
.multi_get_object_with_options(stake_ids, SuiObjectDataOptions::default())
.await?;
let stake_refs = responses
.into_iter()
.map(|stake| stake.into_object().map(|o| o.object_ref()))
.collect::<Result<Vec<_>, _>>()
.map_err(SuiError::from)?;
(Some(0), stake_refs)
}
};
let budget = match budget {
Some(budget) => budget,
None => {
let data = option
.internal_operation
.try_into_data(ConstructionMetadata {
sender,
coins: vec![],
objects: objects.clone(),
total_coin_value: 1_000_000_000 * 1_000_000_000,
gas_price,
budget: 50_000_000_000,
currency: currency.clone(),
})?;
let dry_run = context
.client
.read_api()
.dry_run_transaction_block(data)
.await?;
let effects = dry_run.effects;
if let SuiExecutionStatus::Failure { error } = effects.status() {
return Err(Error::TransactionDryRunError(error.to_string()));
}
effects.gas_cost_summary().computation_cost + effects.gas_cost_summary().storage_cost
}
};
let coins = if let Some(amount) = total_required_amount {
let total_amount = amount + budget;
context
.client
.coin_read_api()
.select_coins(sender, None, total_amount.into(), vec![])
.await
.ok()
} else {
None
};
let coins = if let Some(coins) = coins {
coins
} else {
context
.client
.coin_read_api()
.get_coins_stream(sender, None)
.collect::<Vec<_>>()
.await
};
let total_coin_value = coins.iter().fold(0, |sum, coin| sum + coin.balance);
let coins = coins
.into_iter()
.map(|c| c.object_ref())
.collect::<Vec<_>>();
Ok(ConstructionMetadataResponse {
metadata: ConstructionMetadata {
sender,
coins,
objects,
total_coin_value: total_coin_value.into(),
gas_price,
budget,
currency,
},
suggested_fee: vec![Amount::new(budget as i128, None)],
})
}
pub async fn parse(
Extension(env): Extension<SuiEnv>,
WithRejection(Json(request), _): WithRejection<Json<ConstructionParseRequest>, Error>,
) -> Result<ConstructionParseResponse, Error> {
env.check_network_identifier(&request.network_identifier)?;
let data = if request.signed {
let tx: Transaction = bcs::from_bytes(&request.transaction.to_vec()?)?;
tx.into_data().intent_message().value.clone()
} else {
let intent: IntentMessage<TransactionData> =
bcs::from_bytes(&request.transaction.to_vec()?)?;
intent.value
};
let account_identifier_signers = if request.signed {
vec![data.sender().into()]
} else {
vec![]
};
let operations = data.try_into()?;
Ok(ConstructionParseResponse {
operations,
account_identifier_signers,
metadata: None,
})
}