Smart Order Router V2

Summary

The objective of this section is to explain the inner workings of SOR v2. SOR v1 is explained in detail here.

SOR v2 makes it possible to optimally choose trading routes across different and arbitrary types of pools. This is a major feature that enables Balancer to be the most flexible AMM protocol that exists. Traders however are completely shielded from all the variety of pool types and simply get to benefit from all the aggregated liquidity in the vault.

The reason why SOR v1 could not be used for Balancer v2 is that SOR v1 linearizes the spotPriceAfterSwap functions for pools. This linearization does not work well with stable pools or other arbitrary pools.

The only prerequisite for SOR v2 to work with a pool is that it has first and second (either numerically or analytically) differentiable spotPriceAfterSwap functions.

The final objective of SOR is to find a trade that maximizes the return for the user. One requirement for this to be true is that after the swap is done, each of the paths/routes used ends up having the same spot price. This means that there is not arbitrage possible after the trade (at least with the pools used) and therefore no money left on the table.

Glossary

  • tokenIn: the address of the token being sold by user

  • tokenOut: the address of the token being bought by user

  • swapType < 'swapExactIn' , 'swapExactOut'>: the type of swap being done. The user can select the amount they want to sell (in tokenIn) or the amount they want to buy (in tokenOut)

  • targetAmountSwap: the amount the user wants to buy OR sell, depending on swapType. If swapType is 'swapExactIn' , then targetAmountSwap is the amount the user wants to sell. Conversely, if swapType is 'swapExactOut' , then targetAmountSwap is the amount the user wants to buy.

  • pools: is a dictionary that is loaded from subgraph with all the pools available on Balancer

  • paths: a path is a sequence of pools that enable the trade tokenIn to tokenOut to happen. Paths can contain one hop (direct trades) or two hops. For example a trade of DAI for BAL could have direct trades with pools that contain both DAI and BAL and also multihop paths, for example DAI→WETH→BAL or DAI→USDC→BAL.

  • pairType < 'token->token' , 'token->BPT' ,'BPT->token' >: This is something new to SOR v2. Join/Exit pools with a single token are considered by SOR as a swap just like a swap between two underlying tokens in a pool. This allows for cool use cases like swapping DAI for GUSD in two hops if there is a big stable pool (say pool A) with <DAI, USDT, USDC> and another stable pool with <BPTA, GUSD>, BPTA being the BPT of pool A. So if a hop involves joining a pool, i.e. DAI for BPTA, this pairType would be 'token->BPT'

  • maxPools: is the maximum number of swaps the final solution of SOR can have. A multihop has two swaps, for example.

  • returnToken: is the token whose amount is unknown for the swap and SOR is calculating. Simply put, if swapType is 'swapExactIn' the user is inputting how much they want to sell (targetAmountSwap is in tokenIn) and will get from SOR a returnAmount which is how much they will get back in tokenOut. Conversely, if swapType is 'swapExactOut', then the user is inputting how much they want to buy (targetAmountSwap is in tokenOut) and will get from SOR a returnAmount which is how much they have to pay in tokenIn.

  • costReturnToken: is how much an additional swap would add in gas costs in terms of returnToken. For example, for a 'swapExactIn' swap of DAI for BAL, costReturnToken could be 0.1 BAL (BAL is the returnToken). This is simply calculated by:

costReturnToken=gasPricegasAddSwappriceReturnTokenInETHcostReturnToken = \frac{gasPrice \cdot gasAddSwap}{priceReturnTokenInETH}

Basic algorithm

The basic flow of SOR v2 is as follows

  1. Check how many pools (initialNumPaths) of the most liquid pools we need to be able to trade the targetAmountSwap. Usually targetAmountSwap is much lower than the liquidity available in the largest pools so this is usually 1. Start with next step with b = initialNumPaths where b is the number of paths being considered.

  2. Find out the best b paths and the best distribution of targetAmountSwap which is a list called swapAmounts with b amounts to be swapped in each of the b paths. The sum of all amounts in swapAmounts is always targetAmountSwap. This step also calculates the returnAmount the suggested swaps return.

  3. Compare returnAmount of previous step with bestReturnAmount (which is the best return amount so far) considering the additional costs of adding an extra path (costReturnToken * nSwapsInPath). If the returnAmount is better than bestReturnAmount then increment b and return to step 2). If not the algorithm has found the final solution. IMPORTANT: notice that if swapType is 'swapExactIn' having a better returnAmount means that it is higher than bestReturnAmount, i.e. the user is getting more tokenOut . Conversely, if swapType is 'swapExactOut' having a better returnAmount means that it is lower than bestReturnAmount, i.e. the user is paying less tokenIn for the desired amount (targetAmountSwap) of tokenOut they want to buy.

Deep dive into step 2)

The most challenging part of the algorithm is to find out what is the best combination of paths and swapAmounts for a swap given how many paths we want to include (variable bmentioned above).

Step 2) can be broken down into the following sub-steps:

2.1. Start swapAmounts list based on the previous bestSwapAmounts by adding another element to the list which is defined as targetAmountSwap/b and scale down all the elements of bestSwapAmounts by 1-1/b so that the sum still is equal to targetAmountSwap. For example, if the previous bestSwapAmounts for b=2 were [150, 30] then the new initial swapAmounts for b=3 will be [100, 20, 60]

2.2 Given swapAmounts from above, find the best paths using getBestPathIds() (see detailed information about this function in appendix below).

2.3 Given current swapAmounts and bestPathIds find new swapAmounts that maximize the return for the given bestPathIds

2.4 Return to step 2.2) above with new swapAmounts and check if function getBestPathIds() returns the same bestPathIds. If yes, this means that we have converged to the best paths and swapAmounts and step 2) is finalized. If not, then go to step 2.3) again

Deep dive into step 2.3)

Step 2.3) finds the swapAmounts that will maximize the returns given an initial guess of swapAmounts and the list of paths that should be used (bestPathIds). This is all done in function iterateSwapAmounts() which will be described below in its different steps.

2.3.1 Call iterateSwapAmountsApproximation() and which does a first iteration in approximating swapAmounts to the optimal ones (this function is explained in detail in the appendix below). This function returns prices which are the prices after swap for each of the paths considering their respective amountSwap in swapAmounts as long as that amountSwap is NOT at the limit of the path (i.e. equal to path.limitAmount) AND is not zero.

2.3.2 Check if prices are the same within a given error margin. If this is true, then the objective of not leaving paths in a state that can be arbed has been achieved and the optimal distribution of swap amounts for these paths has been found. If prices are still not close enough to each other go back to step 2.3.1.

Appendix

getBestPathIds()

getBestPathIds() takes as inputs swapAmounts and all available paths and is expected to find the paths that give the best return for the given swapAmounts. The algorithm to do so is very simple: sort the swapAmounts by descending order and starting with the largest swapAmount, find the path that has the best return. Select this path and remove it from the list of available paths as we do not want to use the same path twice (as using it the first time would affect its price and slippage). The continue for each subsequent swapAmount until we chose a path for each swapAmount

iterateSwapAmountsApproximation()

iterateSwapAmountsApproximation() takes as inputs swapAmounts and bestPathIds and is expected to find new swapAmounts that have spot prices after swap as close as possible.

To help illustrate this, let's start with two paths (1 and 2) and initial swapAmounts equal to AiA_i and A2A_2 . Remember that always the sum of the individual amounts is targetAmountSwap.

The objective is to find a target spot price (targetSP) and A1A'_1 and A2A'_2 . such that:

SPaA1(A1)=SPaA2(A2)=targetSPA1+A2=A1+A2=ATSPaA_1(A_1) =SPaA_2(A_2)= targetSP \\ A_1'+A_2' = A_1+A_2 = A_T

To help the visualization, this is what we are looking to achieve:

To calculate a candidate for targetSP we need to use the derivatives of SPaS1SPaS_1 and SPaS2SPaS_2 at A1A_1 and A2A_2 respectively.

Using simple trigonometry we can say that:

SPaS1(A1)targetSP=SPaS1(A1)(A1A1)SPaS2(A2)targetSP=SPaS2(A2)(A2A2)SPaS_1(A_1)-targetSP = SPaS_1'(A_1)\cdot(A_1'-A_1)\\ SPaS_2(A_2)-targetSP = SPaS_2'(A_2)\cdot(A_2'-A_2)

The solution for this system of equations is:

targetSP=SPaS1(A1)SPaS1(A1)+SPaS2(A2)SPaS2(A2)1SPaS1(A1)+1SPaS1(A1)targetSP = \frac{\frac{SPaS_1(A_1)}{SPaS_1'(A_1)}+\frac{SPaS_2(A_2)}{SPaS_2'(A_2)}}{\frac{1}{SPaS_1'(A_1)}+\frac{1}{SPaS_1'(A_1)}}

In other words, the target spot price is the average of the spot prices after swap weighted by the inverse of their derivatives. The derivatives of SPaSSPaS can be seen as the slippage of that path. Generalizing for any number of paths we have:

targetSP=iSPaSi(Ai)SPaSi(Ai)i1SPaSi(Ai)targetSP = \frac{\sum_i{\frac{SPaS_i(A_i)}{SPaS'_i(A_i)}}} {\sum_i{\frac{1}{SPaS'_i(A_i)}}}

After calculating targetSP it's easy to replace it in the equations above to find each AiA'_i :

Ai=SPaSi(Ai)targetSPSPaSi(Ai)+AiA'_i = \frac{SPaS_i(Ai) - targetSP}{SPaS'_i(Ai)}+A_i

Notice that the paths have limits in the amounts they can be used to swap. The lower boundary is always zero, since you cannot swap a negative number. The upper boundary is usually defined by 50% of the balance a pool has in the token being swapped. These limits have to be taken into account and respected in the choice of swapAmounts. Function redistributeInputAmounts() does exactly that. Let's take a look at it into more detail below.

redistributeInputAmounts()

If after the calculations done in iterateSwapAmountsApproximation() above we end up with AiA'_i that is negative or above the limit of path ii , then we have to set it to 0 or AlimitiA_{limit_i} respectively. The excesses have to be 'redistributed' to the other viable paths (i.e. paths that do not have swap amounts below zero or above the path limit), otherwise the sum of AiA_i for all paths ii is not going to be equal to targetAmountSwap as it should.

Function redistributeInputAmounts() might need to be calculated several times in a row because as the excesses are redistributed to paths that are viable, their swap amounts might go below zero or beyond the limit. We call redistributeInputAmounts() iteratively until all swap amounts are above or equal to zero and below or equal to their path limit amounts.

Last updated