curve-math 0.1.0-alpha.2

Pure Rust port of Curve Finance AMM math (StableSwap + CryptoSwap). Wei-level precision, fuzz-verified against on-chain get_dy for 200+ pools.
Documentation
//! StableSwapNG — Next-gen StableSwap (plain + meta NG pools).
//!
//! a_precision=100, -1 offset, fee before denormalize, dynamic fee with avg xp.
//! Vyper: https://github.com/curvefi/stableswap-ng/blob/main/contracts/main/CurveStableSwapNG.vy

use alloy_primitives::U256;

pub const PRECISION: u128 = 1_000_000_000_000_000_000;
pub const FEE_DENOMINATOR: u64 = 10_000_000_000;
pub const A_PRECISION: u64 = 100;
const MAX_ITERATIONS: usize = 255;

pub fn get_d(xp: &[U256], amp: U256) -> Option<U256> {
    let n = U256::from(xp.len());
    let a_prec = U256::from(A_PRECISION);
    let sum: U256 = xp
        .iter()
        .try_fold(U256::ZERO, |acc, b| acc.checked_add(*b))?;
    if sum.is_zero() {
        return Some(U256::ZERO);
    }
    let ann = amp.checked_mul(n)?;
    let mut d = sum;
    for _ in 0..MAX_ITERATIONS {
        let mut d_p = d;
        for balance in xp {
            d_p = d_p.checked_mul(d)?.checked_div(*balance)?;
        }
        // NG divides by N^N at the end (pow_mod256), not per-element
        let mut n_pow_n = U256::from(1u64);
        for _ in 0..xp.len() {
            n_pow_n = n_pow_n.checked_mul(n)?;
        }
        d_p = d_p.checked_div(n_pow_n)?;
        let d_prev = d;
        let num = ann
            .checked_mul(sum)?
            .checked_div(a_prec)?
            .checked_add(d_p.checked_mul(n)?)?
            .checked_mul(d)?;
        let den = ann
            .checked_sub(a_prec)?
            .checked_mul(d)?
            .checked_div(a_prec)?
            .checked_add(n.checked_add(U256::from(1))?.checked_mul(d_p)?)?;
        if den.is_zero() {
            return None;
        }
        d = num.checked_div(den)?;
        let diff = if d > d_prev { d - d_prev } else { d_prev - d };
        if diff <= U256::from(1) {
            return Some(d);
        }
    }
    None
}

pub fn get_y(i: usize, j: usize, x_new: U256, xp: &[U256], d: U256, amp: U256) -> Option<U256> {
    let n = U256::from(xp.len());
    let a_prec = U256::from(A_PRECISION);
    let ann = amp.checked_mul(n)?;
    let mut s_prime = U256::ZERO;
    let mut c = d;
    #[allow(clippy::needless_range_loop)]
    for k in 0..xp.len() {
        let x_k = if k == i {
            x_new
        } else if k != j {
            xp[k]
        } else {
            continue;
        };
        s_prime = s_prime.checked_add(x_k)?;
        c = c.checked_mul(d)?.checked_div(x_k.checked_mul(n)?)?;
    }
    c = c
        .checked_mul(d)?
        .checked_mul(a_prec)?
        .checked_div(ann.checked_mul(n)?)?;
    let b = s_prime.checked_add(d.checked_mul(a_prec)?.checked_div(ann)?)?;
    let mut y = d;
    for _ in 0..MAX_ITERATIONS {
        let y_prev = y;
        let num = y.checked_mul(y)?.checked_add(c)?;
        let den = y
            .checked_mul(U256::from(2))?
            .checked_add(b)?
            .checked_sub(d)?;
        if den.is_zero() {
            return None;
        }
        y = num.checked_div(den)?;
        let diff = if y > y_prev { y - y_prev } else { y_prev - y };
        if diff <= U256::from(1) {
            return Some(y);
        }
    }
    None
}

pub fn dynamic_fee(xpi: U256, xpj: U256, fee: U256, fee_multiplier: U256) -> U256 {
    let fee_denom = U256::from(FEE_DENOMINATOR);
    if fee_multiplier <= fee_denom {
        return fee;
    }
    let xps2 = (xpi + xpj) * (xpi + xpj);
    if xps2.is_zero() {
        return fee;
    }
    (fee_multiplier * fee)
        / ((fee_multiplier - fee_denom) * U256::from(4u64) * xpi * xpj / xps2 + fee_denom)
}

#[cfg(test)]
mod tests {
    use super::*;

    fn wad() -> U256 {
        U256::from(PRECISION)
    }

    #[test]
    fn get_d_balanced_pool() {
        let balance = U256::from(1_000_000u64) * wad();
        let amp = U256::from(400u64 * A_PRECISION as u64);
        let d = get_d(&[balance, balance], amp).expect("converge");
        let expected = balance * U256::from(2u64);
        let diff = if d > expected {
            d - expected
        } else {
            expected - d
        };
        assert!(diff <= U256::from(1));
    }

    #[test]
    fn get_y_roundtrip() {
        let balance = U256::from(1_000_000u64) * wad();
        let xp = [balance, balance];
        let amp = U256::from(400u64 * A_PRECISION as u64);
        let d = get_d(&xp, amp).expect("d");
        let y = get_y(0, 1, xp[0], &xp, d, amp).expect("y");
        let diff = if y > xp[1] { y - xp[1] } else { xp[1] - y };
        assert!(diff <= U256::from(1));
    }

    #[test]
    fn dynamic_fee_at_peg_returns_base_fee() {
        let fee = U256::from(4_000_000u64);
        let multiplier = U256::from(20_000_000_000u64);
        let xp = U256::from(1_000_000u64) * wad();
        let result = dynamic_fee(xp, xp, fee, multiplier);
        assert_eq!(result, fee);
    }

    #[test]
    fn dynamic_fee_off_peg_increases() {
        let fee = U256::from(4_000_000u64);
        let multiplier = U256::from(20_000_000_000u64);
        let xpi = U256::from(2_000_000u64) * wad();
        let xpj = U256::from(500_000u64) * wad();
        let result = dynamic_fee(xpi, xpj, fee, multiplier);
        assert!(result > fee);
    }
}