use clap::Parser;
use crate::{commands::tx, tx::builder, xdr};
#[derive(Parser, Debug, Clone)]
#[group(skip)]
pub struct Cmd {
#[command(flatten)]
pub tx: tx::Args,
#[clap(flatten)]
pub op: Args,
}
#[derive(Debug, clap::Args, Clone)]
pub struct Args {
#[arg(long)]
pub liquidity_pool_id: String,
#[arg(long)]
pub max_amount_a: builder::Amount,
#[arg(long)]
pub max_amount_b: builder::Amount,
#[arg(long, default_value = "1:1")]
pub min_price: String,
#[arg(long, default_value = "1:1")]
pub max_price: String,
}
fn parse_price(price: &str) -> Result<xdr::Price, tx::args::Error> {
let price_parts: Vec<&str> = price.split(':').collect();
if price_parts.len() != 2 {
return Err(tx::args::Error::InvalidPrice(price.to_string()));
}
let n: i32 = price_parts[0]
.parse()
.map_err(|_| tx::args::Error::InvalidPrice(price.to_string()))?;
let d: i32 = price_parts[1]
.parse()
.map_err(|_| tx::args::Error::InvalidPrice(price.to_string()))?;
if d == 0 {
return Err(tx::args::Error::InvalidPrice(
"denominator cannot be zero".to_string(),
));
}
Ok(xdr::Price { n, d })
}
impl TryFrom<&Cmd> for xdr::OperationBody {
type Error = tx::args::Error;
fn try_from(
Cmd {
tx: _,
op:
Args {
liquidity_pool_id,
max_amount_a,
max_amount_b,
min_price,
max_price,
},
}: &Cmd,
) -> Result<Self, Self::Error> {
let pool_id: xdr::PoolId = liquidity_pool_id
.parse()
.map_err(|_| tx::args::Error::InvalidPoolId(liquidity_pool_id.clone()))?;
let min_price_parsed = parse_price(min_price)?;
let max_price_parsed = parse_price(max_price)?;
Ok(xdr::OperationBody::LiquidityPoolDeposit(
xdr::LiquidityPoolDepositOp {
liquidity_pool_id: pool_id,
max_amount_a: max_amount_a.into(),
max_amount_b: max_amount_b.into(),
min_price: min_price_parsed,
max_price: max_price_parsed,
},
))
}
}