use anyhow::{anyhow, Context};
use penumbra_sdk_asset::{asset, Value};
use penumbra_sdk_num::Amount;
use penumbra_sdk_proto::{
penumbra::core::component::dex::v1 as pb, serializers::bech32str, DomainType,
};
use rand_core::CryptoRngCore;
use serde::{Deserialize, Serialize};
use crate::{DirectedTradingPair, TradingPair};
use super::{trading_function::TradingFunction, Reserves};
pub const MAX_RESERVE_AMOUNT: u128 = (1 << 80) - 1;
pub const MAX_FEE_BPS: u32 = 5000;
#[derive(Clone, Serialize, Deserialize, PartialEq, Eq)]
#[serde(try_from = "pb::Position", into = "pb::Position")]
pub struct Position {
pub state: State,
pub reserves: Reserves,
pub phi: TradingFunction,
pub nonce: [u8; 32],
pub close_on_fill: bool,
}
impl std::fmt::Debug for Position {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("Position")
.field("state", &self.state)
.field("reserves", &self.reserves)
.field("phi", &self.phi)
.field("nonce", &hex::encode(self.nonce))
.finish()
}
}
impl Position {
pub fn new<R: CryptoRngCore>(
mut rng: R,
pair: DirectedTradingPair,
fee: u32,
p: Amount,
q: Amount,
reserves: Reserves,
) -> Position {
let mut p = p;
let mut q = q;
let mut reserves = reserves;
let mut nonce_bytes = [0u8; 32];
rng.fill_bytes(&mut nonce_bytes);
let canonical_tp: TradingPair = pair.into();
if pair.start != canonical_tp.asset_1() {
std::mem::swap(&mut p, &mut q);
reserves = Reserves {
r1: reserves.r2,
r2: reserves.r1,
};
}
let phi = TradingFunction::new(canonical_tp, fee, p, q);
Position {
phi,
nonce: nonce_bytes,
state: State::Opened,
reserves,
close_on_fill: false,
}
}
pub fn new_with_nonce(
nonce: [u8; 32],
pair: DirectedTradingPair,
fee: u32,
p: Amount,
q: Amount,
reserves: Reserves,
) -> Position {
let mut p = p;
let mut q = q;
let mut reserves = reserves;
let canonical_tp: TradingPair = pair.into();
if pair.start != canonical_tp.asset_1() {
std::mem::swap(&mut p, &mut q);
reserves = Reserves {
r1: reserves.r2,
r2: reserves.r1,
};
}
let phi = TradingFunction::new(canonical_tp, fee, p, q);
Position {
phi,
nonce,
state: State::Opened,
reserves,
close_on_fill: false,
}
}
pub fn id(&self) -> Id {
let mut state = blake2b_simd::Params::default()
.personal(b"penumbra_lp_id")
.to_state();
state.update(&self.nonce);
state.update(&self.phi.pair.asset_1().to_bytes());
state.update(&self.phi.pair.asset_2().to_bytes());
state.update(&self.phi.component.fee.to_le_bytes());
state.update(&self.phi.component.p.to_le_bytes());
state.update(&self.phi.component.q.to_le_bytes());
let hash = state.finalize();
let mut bytes = [0; 32];
bytes[0..32].copy_from_slice(&hash.as_bytes()[0..32]);
Id(bytes)
}
pub fn check_stateless(&self) -> anyhow::Result<()> {
if self.reserves.r1.value() > MAX_RESERVE_AMOUNT
|| self.reserves.r2.value() > MAX_RESERVE_AMOUNT
{
Err(anyhow::anyhow!(format!(
"Reserve amounts are out-of-bounds (limit: {MAX_RESERVE_AMOUNT})"
)))
} else if self.reserves.r1.value() == 0 && self.reserves.r2.value() == 0 {
Err(anyhow::anyhow!(
"initial reserves must provision some amount of either asset",
))
} else if self.phi.component.p == 0u64.into() || self.phi.component.q == 0u64.into() {
Err(anyhow::anyhow!(
"trading function coefficients must be nonzero"
))
} else if self.phi.component.p.value() > MAX_RESERVE_AMOUNT
|| self.phi.component.q.value() > MAX_RESERVE_AMOUNT
{
Err(anyhow!("trading function coefficients are too large"))
} else if self.phi.pair.asset_1() == self.phi.pair.asset_2() {
Err(anyhow!("cyclical pairs aren't allowed"))
} else if self.phi.component.fee > MAX_FEE_BPS {
Err(anyhow!("fee cannot be greater than 50% (5000bps)"))
} else {
Ok(())
}
}
pub fn reserves_for(&self, asset: asset::Id) -> Option<Amount> {
if asset == self.phi.pair.asset_1() {
Some(self.reserves.r1)
} else if asset == self.phi.pair.asset_2() {
Some(self.reserves.r2)
} else {
None
}
}
pub fn reserves_1(&self) -> Value {
Value {
amount: self.reserves.r1,
asset_id: self.phi.pair.asset_1(),
}
}
pub fn reserves_2(&self) -> Value {
Value {
amount: self.reserves.r2,
asset_id: self.phi.pair.asset_2(),
}
}
}
#[derive(PartialEq, Eq, PartialOrd, Ord, Copy, Clone, Serialize, Deserialize)]
#[serde(try_from = "pb::PositionId", into = "pb::PositionId")]
pub struct Id(pub [u8; 32]);
impl std::fmt::Debug for Id {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.write_str(&bech32str::encode(
&self.0,
bech32str::lp_id::BECH32_PREFIX,
bech32str::Bech32m,
))
}
}
impl std::fmt::Display for Id {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
f.write_str(&bech32str::encode(
&self.0,
bech32str::lp_id::BECH32_PREFIX,
bech32str::Bech32m,
))
}
}
impl std::str::FromStr for Id {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
let inner = bech32str::decode(s, bech32str::lp_id::BECH32_PREFIX, bech32str::Bech32m)?;
pb::PositionId {
inner,
alt_bech32m: String::new(),
}
.try_into()
}
}
#[derive(Debug, PartialEq, Eq, Copy, Clone, Serialize, Deserialize)]
#[serde(try_from = "pb::PositionState", into = "pb::PositionState")]
pub enum State {
Opened,
Closed,
Withdrawn {
sequence: u64,
},
}
impl std::fmt::Display for State {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
State::Opened => write!(f, "opened"),
State::Closed => write!(f, "closed"),
State::Withdrawn { sequence } => write!(f, "withdrawn_{}", sequence),
}
}
}
impl std::str::FromStr for State {
type Err = anyhow::Error;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"opened" => Ok(State::Opened),
"closed" => Ok(State::Closed),
_ => {
let mut parts = s.splitn(2, '_');
if parts.next() == Some("withdrawn") {
let sequence = parts
.next()
.ok_or_else(|| anyhow::anyhow!("missing sequence number"))?
.parse()?;
Ok(State::Withdrawn { sequence })
} else {
Err(anyhow::anyhow!("unknown position state"))
}
}
}
}
}
impl DomainType for Position {
type Proto = pb::Position;
}
impl DomainType for Id {
type Proto = pb::PositionId;
}
impl TryFrom<pb::PositionId> for Id {
type Error = anyhow::Error;
fn try_from(value: pb::PositionId) -> Result<Self, Self::Error> {
match (value.inner.is_empty(), value.alt_bech32m.is_empty()) {
(false, true) => Ok(Id(value
.inner
.as_slice()
.try_into()
.context("expected 32-byte id")?)),
(true, false) => value.alt_bech32m.parse(),
(false, false) => Err(anyhow::anyhow!(
"AssetId proto has both inner and alt_bech32m fields set"
)),
(true, true) => Err(anyhow::anyhow!(
"AssetId proto has neither inner nor alt_bech32m fields set"
)),
}
}
}
impl From<Id> for pb::PositionId {
fn from(value: Id) -> Self {
Self {
inner: value.0.to_vec(),
alt_bech32m: String::new(),
}
}
}
impl DomainType for State {
type Proto = pb::PositionState;
}
impl From<State> for pb::PositionState {
fn from(v: State) -> Self {
pb::PositionState {
state: match v {
State::Opened => pb::position_state::PositionStateEnum::Opened,
State::Closed => pb::position_state::PositionStateEnum::Closed,
State::Withdrawn { .. } => pb::position_state::PositionStateEnum::Withdrawn,
} as i32,
sequence: match v {
State::Withdrawn { sequence } => sequence,
_ => 0,
},
}
}
}
impl TryFrom<pb::PositionState> for State {
type Error = anyhow::Error;
fn try_from(v: pb::PositionState) -> Result<Self, Self::Error> {
let position_state =
pb::position_state::PositionStateEnum::try_from(v.state).map_err(|e| {
anyhow::anyhow!(
"invalid position state enum value: {}, error: {}",
v.state,
e
)
})?;
match position_state {
pb::position_state::PositionStateEnum::Opened => Ok(State::Opened),
pb::position_state::PositionStateEnum::Closed => Ok(State::Closed),
pb::position_state::PositionStateEnum::Withdrawn => Ok(State::Withdrawn {
sequence: v.sequence,
}),
_ => Err(anyhow!("unknown position state")),
}
}
}
impl From<Position> for pb::Position {
fn from(p: Position) -> Self {
Self {
state: Some(p.state.into()),
reserves: Some(p.reserves.into()),
phi: Some(p.phi.into()),
nonce: p.nonce.to_vec(),
close_on_fill: p.close_on_fill,
}
}
}
impl TryFrom<pb::Position> for Position {
type Error = anyhow::Error;
fn try_from(p: pb::Position) -> Result<Self, Self::Error> {
Ok(Self {
state: p
.state
.ok_or_else(|| anyhow::anyhow!("missing state in Position message"))?
.try_into()?,
reserves: p
.reserves
.ok_or_else(|| anyhow::anyhow!("missing reserves in Position message"))?
.try_into()?,
phi: p
.phi
.ok_or_else(|| anyhow::anyhow!("missing trading function"))?
.try_into()?,
nonce: p
.nonce
.as_slice()
.try_into()
.context("expected 32-byte nonce")?,
close_on_fill: p.close_on_fill,
})
}
}
#[cfg(test)]
mod test {
use super::*;
use ark_ff::Zero;
use decaf377::Fq;
use penumbra_sdk_asset::asset;
use rand_core::OsRng;
#[test]
fn position_state_fromstr() {
assert_eq!("opened".parse::<State>().unwrap(), State::Opened);
assert_eq!("closed".parse::<State>().unwrap(), State::Closed);
assert_eq!(
"withdrawn_0".parse::<State>().unwrap(),
State::Withdrawn { sequence: 0 }
);
assert_eq!(
"withdrawn_1".parse::<State>().unwrap(),
State::Withdrawn { sequence: 1 }
);
assert!("withdrawn".parse::<State>().is_err());
assert!("withdrawn_".parse::<State>().is_err());
assert!("withdrawn_1_".parse::<State>().is_err());
assert!("withdrawn_1_2".parse::<State>().is_err());
assert!("withdrawn_1_2_3".parse::<State>().is_err());
}
fn assert_position_similar(p1: Position, p2: Position) {
assert_eq!(p1.reserves.r1, p2.reserves.r1);
assert_eq!(p1.reserves.r2, p2.reserves.r2);
assert_eq!(p1.phi.component.p, p2.phi.component.p);
assert_eq!(p1.phi.component.q, p2.phi.component.q);
}
fn assert_position_not_similar(p1: Position, p2: Position) {
let different_reserves = p1.reserves.r1 != p2.reserves.r1;
let different_reserves = different_reserves || p1.reserves.r2 != p2.reserves.r2;
let different_prices = p1.phi.component.p != p2.phi.component.p;
let different_prices = different_prices || p1.phi.component.q != p2.phi.component.q;
assert!(different_prices || different_reserves);
}
#[test]
fn test_position() {
let small_id = asset::Id(Fq::zero());
let big_id = asset::Id(Fq::from(1u64));
let pair_1 = DirectedTradingPair::new(small_id, big_id);
let pair_2 = DirectedTradingPair::new(big_id, small_id);
let price100i: (Amount, Amount) = (1u64.into(), 100u64.into());
let reserves_1 = Reserves {
r1: 150u64.into(),
r2: 0u64.into(),
};
let reserves_2 = reserves_1.flip();
let a_position_1 = Position::new(
OsRng,
pair_1,
0u32,
price100i.0,
price100i.1,
reserves_1.clone(),
);
let a_position_2 = Position::new(
OsRng,
pair_1,
0u32,
price100i.0,
price100i.1,
reserves_2.clone(),
);
let a_position_3 = Position::new(
OsRng,
pair_1,
0u32,
price100i.1,
price100i.0,
reserves_1.clone(),
);
let a_position_4 = Position::new(
OsRng,
pair_1,
0u32,
price100i.1,
price100i.0,
reserves_2.clone(),
);
let b_position_1 = Position::new(
OsRng,
pair_2,
0u32,
price100i.0,
price100i.1,
reserves_1.clone(),
);
let b_position_2 = Position::new(
OsRng,
pair_2,
0u32,
price100i.0,
price100i.1,
reserves_2.clone(),
);
let b_position_3 = Position::new(
OsRng,
pair_2,
0u32,
price100i.1,
price100i.0,
reserves_1.clone(),
);
let b_position_4 = Position::new(
OsRng,
pair_2,
0u32,
price100i.1,
price100i.0,
reserves_2.clone(),
);
assert_position_not_similar(a_position_1.clone(), b_position_2.clone());
assert_position_not_similar(a_position_2.clone(), b_position_1.clone());
assert_position_similar(a_position_3, b_position_2);
assert_position_similar(a_position_4, b_position_1);
assert_position_similar(b_position_3, a_position_2);
assert_position_similar(b_position_4, a_position_1);
}
}