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
//! StableSwapALend — Aave-style lending pools (Aave, sAAVE, IB, aETH).
//!
//! a_precision=100, no -1 offset, fee after denormalize, dynamic fee with avg xp.
//! Vyper: https://github.com/curvefi/curve-contract/blob/master/contracts/pool-templates/a/SwapTemplateA.vy
//!
//! IMPORTANT: Uses PRECISION_MUL directly (not stored_rates/PRECISION).
//! Normalization: xp[k] = balance[k] * precision_mul[k]

use alloy_primitives::U256;

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.checked_mul(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::*;

    #[test]
    fn get_d_balanced_pool() {
        let balance = U256::from(1_000_000_000_000_000_000u128);
        let amp = U256::from(100u64 * A_PRECISION as u64);
        let d = get_d(&[balance, balance, balance], amp).expect("converge");
        let expected = balance * U256::from(3u64);
        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_000_000_000_000_000u128);
        let xp = [balance, balance, balance];
        let amp = U256::from(100u64 * 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_000_000_000_000_000u128);
        // When xpi == xpj, 4*xpi*xpj/xps2 = 1, so fee should equal base fee
        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_000_000_000_000_000u128);
        let xpj = U256::from(500_000_000_000_000_000u128);
        let result = dynamic_fee(xpi, xpj, fee, multiplier);
        assert!(result > fee);
    }

    #[test]
    fn dynamic_fee_low_multiplier_returns_base() {
        let fee = U256::from(4_000_000u64);
        let fee_denom = U256::from(FEE_DENOMINATOR);
        let xpi = U256::from(2_000_000_000_000_000_000u128);
        let xpj = U256::from(500_000_000_000_000_000u128);
        // multiplier <= FEE_DENOMINATOR should return base fee
        let result = dynamic_fee(xpi, xpj, fee, fee_denom);
        assert_eq!(result, fee);
    }
}