diff --git a/x/gamm/pool-models/stableswap/README.md b/x/gamm/pool-models/stableswap/README.md index 7db5f556ca1..597ad828ab2 100644 --- a/x/gamm/pool-models/stableswap/README.md +++ b/x/gamm/pool-models/stableswap/README.md @@ -1,4 +1,207 @@ -# Stableswap +# Solidly Stableswap + +Stableswaps are pools that offer low slippage for two assets that are intended to be tightly correlated. +There is a price ratio they are expected to be at, and the AMM offers low slippage around this price. +There is still price impact for each trade, and as the liquidity becomes more lop-sided, the slippage drastically increases. This package implements the Solidly stableswap curve, namely a CFMM with -invariant: `xy(x^2 + y^2) = k` +invariant: $f(x, y) = xy(x^2 + y^2) = k$ + +It is generalized to the multi-asset setting as $f(a_1, ..., a_n) = a_1 * ... * a_n (a_1^2 + ... + a_n^2)$ + +## Choice of curve + +{TODO: Include some high level summary of the curve} + +## Pool configuration + +One key concept, is that the pool has a native concept of + +### Scaling factor handling + +An important concept thats up to now, not been mentioned is how do we set the expected price ratio. +In the choice of curve section, we see that its the case that when `x_reserves ~= y_reserves`, that spot price is very close to `1`. However, there are a couple issues with just this in practice: + +* Precision of pegged coins may differ. e.g. 1 Foo = 10^12 base units, whereas 1 WrappedFoo = 10^6 base units, but 1 Foo should trade at around 1 Wrapped Foo. +* Related, I could have a token called TwoFoo which should trade around 1 TwoFoo = 2 Foo +* For staking derivatives, where value accrues within the token, the expected price to concentrate around dynamically changes (very slowly). + +To handle these cases, we introduce scaling factors. What we do is we have a scaling factor for every asset in the pool (defaulted to 1). +Then we map from 'true coin reserves' to 'amm math reasoned about reserves', by doing: +`amm_eq_asset_reserve = true_asset_reserve / asset_scaling_factor`. + + +Then we run the AMM equation around `amm_eq_asset_reserve`, to get an `amm_eq_asset_out`. +Then we get the `true_asset_out = amm_eq_asset_out * asset_out_scaling_factor` + +## Algorithm details + +The AMM pool interfaces requires implementing the following stateful methods: + +```golang + SwapOutAmtGivenIn(tokenIn sdk.Coins, tokenOutDenom string, swapFee sdk.Dec) (tokenOut sdk.Coin, err error) + SwapInAmtGivenOut(tokenOut sdk.Coins, tokenInDenom string, swapFee sdk.Dec) (tokenIn sdk.Coin, err error) + + SpotPrice(baseAssetDenom string, quoteAssetDenom string) (sdk.Dec, error) + + JoinPool(tokensIn sdk.Coins, swapFee sdk.Dec) (numShares sdk.Int, err error) + JoinPoolNoSwap(tokensIn sdk.Coins, swapFee sdk.Dec) (numShares sdk.Int, err error) + ExitPool(numShares sdk.Int, exitFee sdk.Dec) (exitedCoins sdk.Coins, err error) +``` + +The "constant" part of CFMM's imply that we can reason about all their necessary algorithms from just the CFMM equation. There are still multiple ways to solve each method. We detail below the ways in which we do so. This is organized by first discussing variable substitutions we do, to be in a more amenable form, and then the details of how we implement each method. + + + +### CFMM function + +Most operations we do only need to reason about two of the assets in a pool, and sometimes only one. +We wish to have a simpler CFMM function to work within these cases. +Due to the CFMM equation $f$ being a symmetric function, we can without loss of generality reorder the arguments to the function. Thus we put the assets of relevance at the beginning of the function. So if two assets $x, y$, we write: $f(x,y, a_3, ... a_n) = xy * a_3 * ... a_n (x^2 + y^2 + a_3^2 + ... + a_n^2)$. + +We then take a more convenient expression to work with, via variable substition. + + +$$\begin{equation} + v = + \begin{cases} + 1, & \text{if } n=2 \\ + \prod\negthinspace \negthinspace \thinspace^{n}_{i=3} \space a_i, & \text{otherwise} + \end{cases} + \end{equation}$$ + +$$\begin{equation} + w = + \begin{cases} + 0, & \text{if}\ n=2 \\ + \sum\negthinspace \negthinspace \thinspace^{n}_{i=3} \space {a_i^2}, & \text{otherwise} + \end{cases} + \end{equation}$$ + +$$\text{then } g(x,y,v,w) = xyv(x^2 + y^2 + w) = f(x,y, a_3, ... a_n)$$ + +As a corollary, notice that $g(x,y,v,w) = v * g(x,y,1,w)$, which will be useful when we have to compare before and after quantities. We will use $h(x,y,w) := g(x,y,1,w)$ as short-hand for this. + +### Swaps + +The question we need to answer for a swap is "suppose I want to swap $a$ units of $x$, how many units $b$ of $y$ would I get out". + +Since we only deal with two assets at a time, we can then work with our prior definition of $g$. Let the input asset's reserves be $x$, the output asset's reserves be $y$, and we compute $v$ and $w$ given the other asset reserves, whose reserves are untouched throughout the swap. + +First we note the direct way of solving this, its limitation, and then an iterative approximation approach that we implement. + +#### Direct swap solution + +The method to compute this under 0 swap fee is implied by the CFMM equation itself, since the constant refers to: +$g(x_0, y_0, v, w) = k = g(x_0 + a, y_0 - b, v, w)$. As $k$ is linearly related to $v$, and $v$ is unchanged throughout the swap, we can simplify the equation to be reasoning about $k' = \frac{k}{v}$ as the constant, and $h$ instead of $g$ + +We then model the solution by finding a function $\text{solve cfmm}(x, w, k') = y\text{ s.t. }h(x, y, w) = k'$. +Then we can solve the swap amount out by first computing $k'$ as $k' = h(x_0, y_0, w)$, and +computing $y_f := \text{solve cfmm}(x_0 + a, w, k')$. We then get that $b = y_0 - y_f$. + +So all we need is an equation for $\text{solve cfmm}$! Its essentially inverting a multi-variate polynomial, and in this case is solvable: [wolfram alpha link](https://www.wolframalpha.com/input?i=solve+for+y+in+x+*+y+*+%28x%5E2+%2B+y%5E2+%2B+w%29+%3D+k) + +Or if were clever with simplification in the two asset case, we can reduce it to: [desmos link](https://www.desmos.com/calculator/hag1f0wieg). + +These functions are a bit complex, which is fine as they are easy to prove correct. However, they are relatively expensive to compute, the latter needs precision on the order of x^4, and requires computing multiple cubic roots. + +Instead there is a more generic way to compute these, which we detail in the next subsection. + +#### Iterative search solution + +Instead of using the direct solution for $\text{solve cfmm}(x, w, k')$, instead notice that $h(x, y, w)$ is an increasing function in $y$. +So we can simply binary search for $y$ such that $h(x, y, w) = k'$, and we are guaranteed convergence within some error bound. + +In order to do a binary search, we need bounds on $y$. +The lowest lowerbound is $0$, and the largest upperbound is $\infty$. +The maximal upperbound is obviously unworkable, and in general binary searching around wide ranges is unfortunate, as we expect most trades to be centered around $y_0$. +This would suggest that we should do something smarter to iteratively approach the right value for the upperbound at least. +Notice that $h$ is super-linearly related in $y$, and at most cubically related to $y$. +This means that $\forall c \in \mathbb{R}^+, c * h(x,y,w) < h(x,c*y,w) < c^3 * h(x,y,w)$. +We can use this fact to get a pretty-good initial upperbound guess for $y$ using the linear estimate. In the lowerbound case, we leave it as lower-bounded by $0$, otherwise we would need to take a cubed root to get a better estimate. + +```python +def iterative_search(x_f, y_0, w, k, err_tolerance): + k_0 = h(x_f, y_0, w) + lowerbound, upperbound = y_0, y_0 + k_ratio = k_0 / k + if k_ratio < 1: + # k_0 < k. Need to find an upperbound. Worst case assume a linear relationship, gives an upperbound + # TODO: In the future, we can derive better bounds via reasoning about coefficients in the cubic + # These are quite close when we are in the "stable" part of the curve though. + upperbound = ceil(y_0 * k_ratio) + elif k_ratio > 1: + # need to find a lowerbound. We could use a cubic relation, but for now we just set it to 0. + lowerbound = 0 + else: + return y_0 # means x_f = x_0 + k_calculator = lambda y_est: h(x_f, y_est, w) + max_iteration_count = 100 + return binary_search(lowerbound, upperbound, k_calculator, k, err_tolerance) + +def binary_search(lowerbound, upperbound, approximation_fn, target, max_iteration_count, err_tolerance): + iter_count = 0 + cur_k_guess = 0 + while (not satisfies_bounds(cur_k_guess, target, err_tolerance)) and iter_count < max_iteration_count: + iter_count += 1 + cur_y_guess = (lowerbound + upperbound) / 2 + cur_k_guess = approximation_fn(cur_y_guess) + + if cur_k_guess > target: + upperbound = cur_y_guess + else if cur_k_guess < target: + lowerbound = cur_y_guess + + if iter_count == max_iteration_count: + return Error("max iteration count reached") + + return cur_y_guess +``` + +#### Using this in swap methods + +Detail how we take the previously discussed solver, and build SwapExactAmountIn and SwapExactAmountOut. + +### Spot Price + +Spot price for an AMM pool is the derivative of its `CalculateOutAmountGivenIn` equation. +However for the stableswap equation, this is painful: [wolfram alpha link](https://www.wolframalpha.com/input?i=dy%2Fdx+of+y+%3D+%28sqrt%28729+k%5E2+x%5E4+%2B+108+x%5E3+%28w+x+%2B+x%5E3%29%5E3%29+%2B+27+k+x%5E2%29%5E%281%2F3%29%2F%283+2%5E%281%2F3%29+x%29+-+%282%5E%281%2F3%29+%28w+x+%2B+x%5E3%29%29%2F%28sqrt%28729+k%5E2+x%5E4+%2B+108+x%5E3+%28w+x+%2B+x%5E3%29%5E3%29+%2B+27+k+x%5E2%29%5E%281%2F3%29+) + +So instead we compute the spot price by approximating the derivative via a small swap. + +Let $\epsilon$ be a sentinel very small swap in amount. + +Then $\text{spot price} = \frac{\text{CalculateOutAmountGivenIn}(\epsilon)}{\epsilon}$. + +### LP equations + +We divide this section into two parts, `JoinPoolNoSwap & ExitPool`, and `JoinPool`. + +#### JoinPoolNoSwap and ExitPool + +Both of these methods can be implemented via generic AMM techniques. +(Link to them or describe the idea) + +#### JoinPool + +The JoinPool API only supports JoinPoolNoSwap if + +## Code structure + +## Testing strategy + +* Simulator integrations: + * Pool creation + * JoinPool + ExitPool gives a token amount out that is lte input + * SingleTokenIn + ExitPool + Swap to base token gives a token amount that is less than input + * CFMM k adjusting in the correct direction after every action +* Fuzz test binary search algorithm, to see that it still works correctly across wide scale ranges +* Fuzz test approximate equality of iterative approximation swap algorithm and direct equation swap. +* Flow testing the entire stableswap scaling factor update process + +## Extensions + +* The astute observer may notice that the equation we are solving in $\text{solve cfmm}$ is actually a cubic polynomial in $y$, with an always-positive derivative. We should then be able to use newton's root finding algorithm to solve for the solution with quadratic convergence. We do not pursue this today, due to other engineering tradeoffs, and insufficient analysis being done.