use sui_types::Address;
use sui_types::Command;
use sui_types::ObjectDigest;
use sui_types::ObjectId;
use sui_types::TransactionExpiration;
use sui_types::Version;
#[derive(serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedTransaction")]
pub struct Transaction {
#[serde(flatten)]
pub ptb: ProgrammableTransaction,
pub sender: Address,
#[serde(skip_serializing_if = "Option::is_none")]
pub gas_payment: Option<GasPayment>,
pub expiration: TransactionExpiration,
}
#[derive(serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedProgrammableTransaction")]
pub struct ProgrammableTransaction {
pub inputs: Vec<Input>,
pub commands: Vec<Command>,
}
#[derive(serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedGasPayment")]
pub struct GasPayment {
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub objects: Vec<ObjectReference>,
pub owner: Address,
#[serde(
with = "OptionReadableDisplay",
default,
skip_serializing_if = "Option::is_none"
)]
pub price: Option<u64>,
#[serde(
with = "OptionReadableDisplay",
default,
skip_serializing_if = "Option::is_none"
)]
pub budget: Option<u64>,
}
#[derive(Clone, Debug, serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedObjectReference")]
pub struct ObjectReference {
pub object_id: ObjectId,
#[serde(
with = "OptionReadableDisplay",
default,
skip_serializing_if = "Option::is_none"
)]
pub version: Option<Version>,
#[serde(skip_serializing_if = "Option::is_none")]
pub digest: Option<ObjectDigest>,
}
#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedInputKind")]
#[serde(rename_all = "snake_case")]
pub enum InputKind {
Pure,
Shared,
Receiving,
ImmutableOrOwned,
Literal,
}
#[derive(Clone, Debug, Default, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedInput")]
pub struct Input {
#[serde(skip_serializing_if = "Option::is_none")]
pub kind: Option<InputKind>,
#[serde(skip_serializing_if = "Option::is_none")]
pub value: Option<Value>,
#[serde(skip_serializing_if = "Option::is_none")]
pub object_id: Option<ObjectId>,
#[serde(
with = "OptionReadableDisplay",
default,
skip_serializing_if = "Option::is_none",
alias = "initial_shared_version"
)]
pub version: Option<Version>,
#[serde(skip_serializing_if = "Option::is_none")]
pub digest: Option<ObjectDigest>,
#[serde(skip_serializing_if = "Option::is_none")]
pub mutable: Option<bool>,
}
#[derive(Clone, Debug, PartialEq, Eq, serde::Serialize, serde::Deserialize)]
#[serde(rename = "UnresolvedValue")]
#[serde(try_from = "serde_json::Value", into = "serde_json::Value")]
pub enum Value {
Null,
Bool(bool),
Number(u64),
String(String),
Array(Vec<Value>),
}
impl Input {
pub fn owned(object_id: ObjectId, version: u64, digest: ObjectDigest) -> Self {
Self {
kind: Some(InputKind::ImmutableOrOwned),
object_id: Some(object_id),
version: Some(version),
digest: Some(digest),
..Default::default()
}
}
pub fn immutable(object_id: ObjectId, version: u64, digest: ObjectDigest) -> Self {
Self {
kind: Some(InputKind::ImmutableOrOwned),
object_id: Some(object_id),
version: Some(version),
digest: Some(digest),
..Default::default()
}
}
pub fn receiving(object_id: ObjectId, version: u64, digest: ObjectDigest) -> Self {
Self {
kind: Some(InputKind::Receiving),
object_id: Some(object_id),
version: Some(version),
digest: Some(digest),
..Default::default()
}
}
pub fn shared(object_id: ObjectId, initial_shared_version: u64, mutable: bool) -> Self {
Self {
kind: Some(InputKind::Shared),
object_id: Some(object_id),
version: Some(initial_shared_version),
mutable: Some(mutable),
..Default::default()
}
}
pub fn by_id(object_id: ObjectId) -> Self {
Self {
object_id: Some(object_id),
..Default::default()
}
}
pub fn with_immutable_kind(self) -> Self {
Self {
kind: Some(InputKind::ImmutableOrOwned),
..self
}
}
pub fn with_owned_kind(self) -> Self {
Self {
kind: Some(InputKind::ImmutableOrOwned),
..self
}
}
pub fn with_receiving_kind(self) -> Self {
Self {
kind: Some(InputKind::Receiving),
..self
}
}
pub fn with_shared_kind(self) -> Self {
Self {
kind: Some(InputKind::Shared),
..self
}
}
pub fn with_version(self, version: u64) -> Self {
Self {
version: Some(version),
..self
}
}
pub fn with_digest(self, digest: ObjectDigest) -> Self {
Self {
digest: Some(digest),
..self
}
}
pub fn with_initial_shared_version(self, initial_version: u64) -> Self {
Self {
kind: Some(InputKind::Shared),
version: Some(initial_version),
..self
}
}
pub fn by_val(self) -> Self {
Self {
kind: Some(InputKind::Shared),
mutable: Some(true),
..self
}
}
pub fn by_ref(self) -> Self {
Self {
kind: Some(InputKind::Shared),
mutable: Some(false),
..self
}
}
pub fn by_mut(self) -> Self {
Self {
kind: Some(InputKind::Shared),
mutable: Some(true),
..self
}
}
}
impl TryFrom<serde_json::Value> for Value {
type Error = &'static str;
fn try_from(value: serde_json::Value) -> Result<Self, Self::Error> {
let v = match value {
serde_json::Value::Null => Self::Null,
serde_json::Value::Bool(b) => Self::Bool(b),
serde_json::Value::Number(n) => {
Self::Number(n.as_u64().ok_or("expected unsigned integer")?)
}
serde_json::Value::String(s) => Self::String(s),
serde_json::Value::Array(a) => Self::Array(
a.into_iter()
.map(Self::try_from)
.collect::<Result<_, _>>()?,
),
serde_json::Value::Object(_) => return Err("objects are not supported"),
};
Ok(v)
}
}
impl From<Value> for serde_json::Value {
fn from(value: Value) -> Self {
match value {
Value::Null => Self::Null,
Value::Bool(b) => Self::Bool(b),
Value::Number(n) => Self::Number(n.into()),
Value::String(s) => Self::String(s),
Value::Array(a) => Self::Array(a.into_iter().map(Into::into).collect()),
}
}
}
impl From<&sui_types::Object> for Input {
fn from(object: &sui_types::Object) -> Self {
use sui_types::Owner;
let input = Input::by_id(object.object_id())
.with_digest(object.digest())
.with_version(object.version());
match object.owner() {
Owner::Address(_) => input,
Owner::Object(_) => input,
Owner::Shared(at_version) => input.with_initial_shared_version(*at_version),
Owner::Immutable => input.with_immutable_kind(),
}
}
}
impl From<ObjectId> for Input {
fn from(object_id: ObjectId) -> Self {
Input::by_id(object_id)
}
}
pub(crate) type OptionReadableDisplay =
::serde_with::As<Option<::serde_with::IfIsHumanReadable<::serde_with::DisplayFromStr>>>;