mod error;
pub mod unresolved;
use error::Error;
use sui_types::Address;
use sui_types::Argument;
use sui_types::Command;
use sui_types::GasPayment;
use sui_types::Identifier;
use sui_types::Input;
use sui_types::MakeMoveVector;
use sui_types::MergeCoins;
use sui_types::MoveCall;
use sui_types::ObjectId;
use sui_types::ObjectReference;
use sui_types::Publish;
use sui_types::SplitCoins;
use sui_types::Transaction;
use sui_types::TransactionExpiration;
use sui_types::TransferObjects;
use sui_types::TypeTag;
use sui_types::Upgrade;
use base64ct::Encoding;
use serde::Serialize;
#[derive(Clone, Default, Debug)]
pub struct TransactionBuilder {
inputs: Vec<unresolved::Input>,
commands: Vec<Command>,
gas: Vec<unresolved::Input>,
gas_budget: Option<u64>,
gas_price: Option<u64>,
sender: Option<Address>,
sponsor: Option<Address>,
expiration: TransactionExpiration,
}
struct RawBytes(Vec<u8>);
pub struct Serialized<'a, T: Serialize>(pub &'a T);
pub struct Function {
package: Address,
module: Identifier,
function: Identifier,
type_args: Vec<TypeTag>,
}
impl TransactionBuilder {
pub fn new() -> Self {
Self::default()
}
pub fn input(&mut self, i: impl Into<unresolved::Input>) -> Argument {
let input = i.into();
self.inputs.push(input);
Argument::Input((self.inputs.len() - 1) as u16)
}
pub fn gas(&self) -> Argument {
Argument::Gas
}
pub fn add_gas_objects<O, I>(&mut self, gas: I)
where
O: Into<unresolved::Input>,
I: IntoIterator<Item = O>,
{
self.gas.extend(gas.into_iter().map(|x| x.into()));
}
pub fn set_gas_budget(&mut self, budget: u64) {
self.gas_budget = Some(budget);
}
pub fn set_gas_price(&mut self, price: u64) {
self.gas_price = Some(price);
}
pub fn set_sender(&mut self, sender: Address) {
self.sender = Some(sender);
}
pub fn set_sponsor(&mut self, sponsor: Address) {
self.sponsor = Some(sponsor);
}
pub fn set_expiration(&mut self, epoch: u64) {
self.expiration = TransactionExpiration::Epoch(epoch);
}
pub fn move_call(&mut self, function: Function, arguments: Vec<Argument>) -> Argument {
let cmd = Command::MoveCall(MoveCall {
package: function.package.into(),
module: function.module,
function: function.function,
type_arguments: function.type_args,
arguments,
});
self.commands.push(cmd);
Argument::Result(self.commands.len() as u16 - 1)
}
pub fn transfer_objects(&mut self, objects: Vec<Argument>, address: Argument) {
let cmd = Command::TransferObjects(TransferObjects { objects, address });
self.commands.push(cmd);
}
pub fn split_coins(&mut self, coin: Argument, amounts: Vec<Argument>) -> Argument {
let cmd = Command::SplitCoins(SplitCoins { coin, amounts });
self.commands.push(cmd);
Argument::Result(self.commands.len() as u16 - 1)
}
pub fn merge_coins(&mut self, coin: Argument, coins_to_merge: Vec<Argument>) {
let cmd = Command::MergeCoins(MergeCoins {
coin,
coins_to_merge,
});
self.commands.push(cmd);
}
pub fn make_move_vec(&mut self, type_: Option<TypeTag>, elements: Vec<Argument>) -> Argument {
let cmd = Command::MakeMoveVector(MakeMoveVector { type_, elements });
self.commands.push(cmd);
Argument::Result(self.commands.len() as u16 - 1)
}
pub fn publish(&mut self, modules: Vec<Vec<u8>>, dependencies: Vec<ObjectId>) -> Argument {
let cmd = Command::Publish(Publish {
modules,
dependencies,
});
self.commands.push(cmd);
Argument::Result(self.commands.len() as u16 - 1)
}
pub fn upgrade(
&mut self,
modules: Vec<Vec<u8>>,
dependencies: Vec<ObjectId>,
package: ObjectId,
ticket: Argument,
) -> Argument {
let cmd = Command::Upgrade(Upgrade {
modules,
dependencies,
package,
ticket,
});
self.commands.push(cmd);
Argument::Result(self.commands.len() as u16 - 1)
}
pub fn finish(self) -> Result<Transaction, Error> {
let Some(sender) = self.sender else {
return Err(Error::MissingSender);
};
if self.gas.is_empty() {
return Err(Error::MissingGasObjects);
}
let Some(budget) = self.gas_budget else {
return Err(Error::MissingGasBudget);
};
let Some(price) = self.gas_price else {
return Err(Error::MissingGasPrice);
};
Ok(Transaction {
kind: sui_types::TransactionKind::ProgrammableTransaction(
sui_types::ProgrammableTransaction {
inputs: self
.inputs
.into_iter()
.map(try_from_unresolved_input_arg)
.collect::<Result<Vec<_>, _>>()?,
commands: self.commands,
},
),
sender,
gas_payment: {
GasPayment {
objects: self
.gas
.into_iter()
.map(try_from_gas_unresolved_input_to_unresolved_obj_ref)
.collect::<Result<Vec<_>, _>>()?
.into_iter()
.map(try_from_unresolved_obj_ref)
.collect::<Result<Vec<_>, _>>()?,
owner: self.sponsor.unwrap_or(sender),
price,
budget,
}
},
expiration: self.expiration,
})
}
}
impl Function {
pub fn new(
package: Address,
module: Identifier,
function: Identifier,
type_args: Vec<TypeTag>,
) -> Self {
Self {
package,
module,
function,
type_args,
}
}
}
impl From<RawBytes> for unresolved::Input {
fn from(raw: RawBytes) -> Self {
Self {
kind: Some(unresolved::InputKind::Pure),
value: Some(unresolved::Value::String(base64ct::Base64::encode_string(
&raw.0,
))),
object_id: None,
version: None,
digest: None,
mutable: None,
}
}
}
impl<'a, T: Serialize> From<Serialized<'a, T>> for unresolved::Input {
fn from(value: Serialized<'a, T>) -> Self {
Self::from(RawBytes(bcs::to_bytes(value.0).unwrap()))
}
}
fn try_from_gas_unresolved_input_to_unresolved_obj_ref(
input: unresolved::Input,
) -> Result<unresolved::ObjectReference, Error> {
match input.kind {
Some(unresolved::InputKind::ImmutableOrOwned) => {
let object_id = input.object_id.ok_or(Error::MissingObjectId)?;
let version = input.version;
let digest = input.digest;
Ok(unresolved::ObjectReference {
object_id,
version,
digest,
})
}
_ => Err(Error::WrongGasObject),
}
}
fn try_from_unresolved_obj_ref(obj: unresolved::ObjectReference) -> Result<ObjectReference, Error> {
let obj_id = obj.object_id;
let version = obj.version.ok_or(Error::MissingVersion(obj_id))?;
let digest = obj.digest.ok_or(Error::MissingDigest(obj_id))?;
Ok(ObjectReference::new(obj_id, version, digest))
}
fn try_from_unresolved_input_arg(value: unresolved::Input) -> Result<Input, Error> {
if let Some(kind) = value.kind {
match kind {
unresolved::InputKind::Pure => {
let Some(value) = value.value else {
return Err(Error::MissingPureValue);
};
match value {
unresolved::Value::String(v) => {
let bytes = base64ct::Base64::decode_vec(&v).map_err(Error::Decoding)?;
Ok(Input::Pure { value: bytes })
}
_ => Err(Error::Input(
"expected a base64 string value for the Pure input argument".to_string(),
)),
}
}
unresolved::InputKind::ImmutableOrOwned => {
let Some(object_id) = value.object_id else {
return Err(Error::MissingObjectId);
};
let Some(version) = value.version else {
return Err(Error::MissingVersion(object_id));
};
let Some(digest) = value.digest else {
return Err(Error::MissingDigest(object_id));
};
Ok(Input::ImmutableOrOwned(ObjectReference::new(
object_id, version, digest,
)))
}
unresolved::InputKind::Shared => {
let Some(object_id) = value.object_id else {
return Err(Error::MissingObjectId);
};
let Some(initial_shared_version) = value.version else {
return Err(Error::MissingInitialSharedVersion(object_id));
};
let Some(mutable) = value.mutable else {
return Err(Error::SharedObjectMutability(object_id));
};
Ok(Input::Shared {
object_id,
initial_shared_version,
mutable,
})
}
unresolved::InputKind::Receiving => {
let Some(object_id) = value.object_id else {
return Err(Error::MissingObjectId);
};
let Some(version) = value.version else {
return Err(Error::MissingVersion(object_id));
};
let Some(digest) = value.digest else {
return Err(Error::MissingDigest(object_id));
};
Ok(Input::Receiving(ObjectReference::new(
object_id, version, digest,
)))
}
unresolved::InputKind::Literal => Err(Error::UnsupportedLiteral),
}
} else {
Err(Error::Input(
"unresolved::Input must have a kind that is not None".to_string(),
))
}
}
#[cfg(test)]
mod tests {
use std::str::FromStr;
use anyhow::Context;
use base64ct::Encoding;
use serde::de;
use serde::Deserialize;
use serde::Deserializer;
use sui_crypto::ed25519::Ed25519PrivateKey;
use sui_crypto::SuiSigner;
use sui_graphql_client::faucet::CoinInfo;
use sui_graphql_client::faucet::FaucetClient;
use sui_graphql_client::Client;
use sui_graphql_client::PaginationFilter;
use sui_types::Address;
use sui_types::ExecutionStatus;
use sui_types::IdOperation;
use sui_types::ObjectId;
use sui_types::ObjectType;
use sui_types::TransactionEffects;
use sui_types::TypeTag;
use crate::unresolved::Input;
use crate::Function;
use crate::Serialized;
use crate::TransactionBuilder;
use sui_types::TransactionDigest;
#[derive(serde::Deserialize, Debug)]
struct MovePackageData {
#[serde(deserialize_with = "bcs_from_str")]
modules: Vec<Vec<u8>>,
#[serde(deserialize_with = "deps_from_str")]
dependencies: Vec<ObjectId>,
digest: Vec<u8>,
}
fn bcs_from_str<'de, D>(deserializer: D) -> Result<Vec<Vec<u8>>, D::Error>
where
D: Deserializer<'de>,
{
let bcs = Vec::<String>::deserialize(deserializer)?;
bcs.into_iter()
.map(|s| base64ct::Base64::decode_vec(&s).map_err(de::Error::custom))
.collect()
}
fn deps_from_str<'de, D>(deserializer: D) -> Result<Vec<ObjectId>, D::Error>
where
D: Deserializer<'de>,
{
let deps = Vec::<String>::deserialize(deserializer)?;
deps.into_iter()
.map(|s| ObjectId::from_str(&s).map_err(de::Error::custom))
.collect()
}
fn move_package_data(file: &str) -> MovePackageData {
let data = std::fs::read_to_string(file)
.with_context(|| {
format!(
"Failed to read {file}. \
Run `make test-with-localnet` from the root of the repository that will \
generate the right json files with the package data and then run the tests."
)
})
.unwrap();
serde_json::from_str(&data).unwrap()
}
fn helper_address_pk() -> (Address, Ed25519PrivateKey) {
let pk = Ed25519PrivateKey::generate(rand::thread_rng());
let address = pk.public_key().derive_address();
(address, pk)
}
async fn helper_setup(
tx: &mut TransactionBuilder,
client: &Client,
) -> (Address, Ed25519PrivateKey, Vec<CoinInfo>) {
let (address, pk) = helper_address_pk();
let coins = FaucetClient::local()
.request_and_wait(address)
.await
.unwrap()
.unwrap()
.sent;
let tx_digest = coins.first().unwrap().transfer_tx_digest;
wait_for_tx(client, tx_digest).await;
let gas = coins.last().unwrap().id;
let gas_obj: Input = (&client.object(gas.into(), None).await.unwrap().unwrap()).into();
tx.add_gas_objects(vec![gas_obj.with_owned_kind()]);
tx.set_gas_budget(500000000);
tx.set_gas_price(1000);
tx.set_sender(address);
(address, pk, coins)
}
async fn wait_for_tx(client: &Client, digest: TransactionDigest) {
while client.transaction(digest).await.unwrap().is_none() {
tokio::time::sleep(tokio::time::Duration::from_millis(100)).await;
}
}
async fn wait_for_tx_and_check_effects_status_success(
client: &Client,
digest: TransactionDigest,
effects: Result<Option<TransactionEffects>, sui_graphql_client::error::Error>,
) {
assert!(effects.is_ok(), "Execution failed. Effects: {:?}", effects);
wait_for_tx(client, digest).await;
let status = effects.unwrap();
let expected_status = ExecutionStatus::Success;
assert_eq!(&expected_status, status.as_ref().unwrap().status());
}
#[tokio::test]
async fn test_finish() {
let mut tx = TransactionBuilder::new();
let coin_obj_id = "0x19406ea4d9609cd9422b85e6bf2486908f790b778c757aff805241f3f609f9b4";
let coin_digest = "7opR9rFUYivSTqoJHvFb9p6p54THyHTatMG6id4JKZR9";
let coin_version = 2;
let coin = tx.input(Input::owned(
coin_obj_id.parse().unwrap(),
coin_version,
coin_digest.parse().unwrap(),
));
let addr = Address::generate(rand::thread_rng());
let recipient = tx.input(Serialized(&addr));
let result = tx.clone().finish();
assert!(result.is_err());
tx.transfer_objects(vec![coin], recipient);
tx.set_gas_budget(500000000);
tx.set_gas_price(1000);
tx.add_gas_objects(vec![Input::immutable(
"0xd8792bce2743e002673752902c0e7348dfffd78638cb5367b0b85857bceb9821"
.parse()
.unwrap(),
2,
"2ZigdvsZn5BMeszscPQZq9z8ebnS2FpmAuRbAi9ednCk"
.parse()
.unwrap(),
)]);
tx.set_sender(
"0xc574ea804d9c1a27c886312e96c0e2c9cfd71923ebaeb3000d04b5e65fca2793"
.parse()
.unwrap(),
);
let tx = tx.finish();
assert!(tx.is_ok());
}
#[tokio::test]
async fn test_transfer_obj_execution() {
let mut tx = TransactionBuilder::new();
let (_, pk, coins) = helper_setup(&mut tx, &Client::new_localhost()).await;
let client = Client::new_localhost();
let first = coins.first().unwrap().id;
let coin: Input = (&client.object(first.into(), None).await.unwrap().unwrap()).into();
let coin_input = tx.input(coin.with_owned_kind());
let recipient = Address::generate(rand::thread_rng());
let recipient_input = tx.input(Serialized(&recipient));
tx.transfer_objects(vec![coin_input], recipient_input);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
let recipient_coins = client
.coins(recipient, None, PaginationFilter::default())
.await
.unwrap();
assert_eq!(recipient_coins.data().len(), 1);
}
#[tokio::test]
async fn test_move_call() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (_, pk, _) = helper_setup(&mut tx, &client).await;
let function = Function::new(
"0x1".parse().unwrap(),
"option".parse().unwrap(),
"is_none".parse().unwrap(),
vec![TypeTag::U64],
);
let input = tx.input(Serialized(&vec![1u64]));
tx.move_call(function, vec![input]);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
}
#[tokio::test]
async fn test_split_transfer() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (_, pk, _) = helper_setup(&mut tx, &client).await;
let amount = tx.input(Serialized(&1_000_000_000u64));
let result = tx.split_coins(tx.gas(), vec![amount]);
let recipient_address = Address::generate(rand::thread_rng());
let recipient = tx.input(Serialized(&recipient_address));
tx.transfer_objects(vec![result], recipient);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
let recipient_coins = client
.coins(recipient_address, None, PaginationFilter::default())
.await
.unwrap();
assert_eq!(recipient_coins.data().len(), 1);
}
#[tokio::test]
async fn test_split_without_transfer_should_fail() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (_, pk, coins) = helper_setup(&mut tx, &client).await;
let coin = coins.first().unwrap().id;
let coin_obj: Input = (&client.object(coin.into(), None).await.unwrap().unwrap()).into();
let coin_input = tx.input(coin_obj.with_owned_kind());
let amount = tx.input(Serialized(&1_000_000_000u64));
tx.split_coins(coin_input, vec![amount]);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
assert!(effects.is_ok());
loop {
let tx_digest = client.transaction(tx.digest()).await.unwrap();
if tx_digest.is_some() {
break;
}
}
assert!(effects.is_ok());
let status = effects.unwrap();
let expected_status = ExecutionStatus::Success;
assert_ne!(&expected_status, status.as_ref().unwrap().status());
}
#[tokio::test]
async fn test_merge_coins() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (address, pk, coins) = helper_setup(&mut tx, &client).await;
let coin1 = coins.first().unwrap().id;
let coin1_obj: Input = (&client.object(coin1.into(), None).await.unwrap().unwrap()).into();
let coin_to_merge = tx.input(coin1_obj.with_owned_kind());
let mut coins_to_merge = vec![];
for c in coins[1..&coins.len() - 1].iter() {
let coin: Input = (&client.object(c.id.into(), None).await.unwrap().unwrap()).into();
coins_to_merge.push(tx.input(coin.with_owned_kind()));
}
tx.merge_coins(coin_to_merge, coins_to_merge);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
let coins_after = client
.coins(address, None, PaginationFilter::default())
.await
.unwrap();
assert_eq!(coins_after.data().len(), 2);
}
#[tokio::test]
async fn test_make_move_vec() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (_, pk, _) = helper_setup(&mut tx, &client).await;
let input = tx.input(Serialized(&1u64));
tx.make_move_vec(Some(TypeTag::U64), vec![input]);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
}
#[tokio::test]
async fn test_publish() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (address, pk, _) = helper_setup(&mut tx, &client).await;
let package = move_package_data("package_test_example_v1.json");
let sender = tx.input(Serialized(&address));
let upgrade_cap = tx.publish(package.modules, package.dependencies);
tx.transfer_objects(vec![upgrade_cap], sender);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
}
#[tokio::test]
async fn test_upgrade() {
let client = Client::new_localhost();
let mut tx = TransactionBuilder::new();
let (address, pk, coins) = helper_setup(&mut tx, &client).await;
let package = move_package_data("package_test_example_v2.json");
let sender = tx.input(Serialized(&address));
let upgrade_cap = tx.publish(package.modules, package.dependencies);
tx.transfer_objects(vec![upgrade_cap], sender);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
let mut package_id: Option<ObjectId> = None;
let mut created_objs = vec![];
if let Ok(Some(ref effects)) = effects {
match effects {
TransactionEffects::V2(e) => {
for obj in e.changed_objects.clone() {
if obj.id_operation == IdOperation::Created {
let change = obj.output_state;
match change {
sui_types::ObjectOut::PackageWrite { .. } => {
package_id = Some(obj.object_id);
}
sui_types::ObjectOut::ObjectWrite { .. } => {
created_objs.push(obj.object_id);
}
_ => {}
}
}
}
}
_ => panic!("Expected V2 effects"),
}
}
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
let mut tx = TransactionBuilder::new();
let mut upgrade_cap = None;
for o in created_objs {
let obj = client.object(*o.as_address(), None).await.unwrap().unwrap();
match obj.object_type() {
ObjectType::Struct(x) if x.name.to_string() == "UpgradeCap" => {
match obj.owner() {
sui_types::Owner::Address(_) => {
let obj: Input = (&obj).into();
upgrade_cap = Some(tx.input(obj.with_owned_kind()))
}
sui_types::Owner::Shared(_) => {
upgrade_cap = Some(tx.input(&obj))
}
sui_types::Owner::Object(_) => {
panic!("Upgrade capability controlled by object")
}
sui_types::Owner::Immutable => panic!("Upgrade capability is stored immutably and cannot be used for upgrades"),
};
break;
}
_ => {}
};
}
let upgrade_policy = tx.input(Serialized(&0u8));
let updated_package = move_package_data("package_test_example_v2.json");
let digest_arg = tx.input(Serialized(&updated_package.digest));
let upgrade_ticket = tx.move_call(
Function::new(
"0x2".parse().unwrap(),
"package".parse().unwrap(),
"authorize_upgrade".parse().unwrap(),
vec![],
),
vec![upgrade_cap.unwrap(), upgrade_policy, digest_arg],
);
let upgrade_receipt = tx.upgrade(
updated_package.modules,
updated_package.dependencies,
package_id.unwrap(),
upgrade_ticket,
);
tx.move_call(
Function::new(
"0x2".parse().unwrap(),
"package".parse().unwrap(),
"commit_upgrade".parse().unwrap(),
vec![],
),
vec![upgrade_cap.unwrap(), upgrade_receipt],
);
let gas = coins.last().unwrap().id;
let gas_obj: Input = (&client.object(gas.into(), None).await.unwrap().unwrap()).into();
tx.add_gas_objects(vec![gas_obj.with_owned_kind()]);
tx.set_gas_budget(500000000);
tx.set_gas_price(1000);
tx.set_sender(address);
let tx = tx.finish().unwrap();
let sig = pk.sign_transaction(&tx).unwrap();
let effects = client.execute_tx(vec![sig], &tx).await;
wait_for_tx_and_check_effects_status_success(&client, tx.digest(), effects).await;
}
}