diff --git a/.changeset/fuzzy-coins-sin.md b/.changeset/fuzzy-coins-sin.md new file mode 100644 index 00000000..2053c322 --- /dev/null +++ b/.changeset/fuzzy-coins-sin.md @@ -0,0 +1,5 @@ +--- +"@balancer/sdk": patch +--- + +Refactor nestedPoolState logic to fetch mainTokens from api diff --git a/.changeset/pink-phones-suffer.md b/.changeset/pink-phones-suffer.md new file mode 100644 index 00000000..87ca46ac --- /dev/null +++ b/.changeset/pink-phones-suffer.md @@ -0,0 +1,5 @@ +--- +"@balancer/sdk": patch +--- + +Fix getPoolStateWithBalancesV3 with less 18 decimals tokens diff --git a/.changeset/two-kiwis-wash.md b/.changeset/two-kiwis-wash.md new file mode 100644 index 00000000..69813990 --- /dev/null +++ b/.changeset/two-kiwis-wash.md @@ -0,0 +1,5 @@ +--- +"@balancer/sdk": patch +--- + +Fix circular dependency on price impact implementation diff --git a/src/data/providers/balancer-api/modules/nested-pool-state/index.ts b/src/data/providers/balancer-api/modules/nested-pool-state/index.ts index 4dcbe52d..5512f036 100644 --- a/src/data/providers/balancer-api/modules/nested-pool-state/index.ts +++ b/src/data/providers/balancer-api/modules/nested-pool-state/index.ts @@ -14,11 +14,6 @@ export type PoolGetPool = { protocolVersion: 1 | 2 | 3; address: Address; type: string; - allTokens: { - address: Address; - decimals: number; - isMainToken: boolean; - }[]; poolTokens: Token[]; }; @@ -48,11 +43,6 @@ export class NestedPools { protocolVersion address type - allTokens { - address - decimals - isMainToken - } poolTokens { index address @@ -258,14 +248,10 @@ export function mapPoolToNestedPoolStateV2(pool: PoolGetPool): NestedPoolState { }); }); - const mainTokens = pool.allTokens - .filter((t) => t.isMainToken) - .map((t) => { - return { - address: t.address, - decimals: t.decimals, - }; - }); + // mainTokens are pool tokens filtering out nested pools and phantomBPTs + const mainTokens = pools + .flatMap((p) => p.tokens) + .filter((t) => !pools.find((p) => p.address === t.address)); return { protocolVersion: 2, diff --git a/src/entities/priceImpact/addLiquidityNested.ts b/src/entities/priceImpact/addLiquidityNested.ts index 12a6cc90..9ec46e34 100644 --- a/src/entities/priceImpact/addLiquidityNested.ts +++ b/src/entities/priceImpact/addLiquidityNested.ts @@ -13,11 +13,12 @@ import { AddLiquidityKind, AddLiquidityUnbalancedInput, } from '../addLiquidity/types'; -import { PriceImpact } from '.'; import { ChainId } from '@/utils'; import { TokenAmount } from '../tokenAmount'; import { AddLiquidityBoostedUnbalancedInput } from '../addLiquidityBoosted/types'; import { AddLiquidityBoostedV3 } from '../addLiquidityBoosted'; +import { addLiquidityUnbalanced } from './addLiquidityUnbalanced'; +import { addLiquidityUnbalancedBoosted } from './addLiquidityUnbalancedBoosted'; type AddResult = { priceImpactAmount: PriceImpactAmount; @@ -111,7 +112,7 @@ async function getAddUnbalancedResult( ...pool, protocolVersion, }; - const priceImpactAmount = await PriceImpact.addLiquidityUnbalanced( + const priceImpactAmount = await addLiquidityUnbalanced( addLiquidityInput, poolState, ); @@ -135,7 +136,7 @@ async function getAddBoostedUnbalancedResult( kind: AddLiquidityKind.Unbalanced, }; - const priceImpactAmount = await PriceImpact.addLiquidityUnbalancedBoosted( + const priceImpactAmount = await addLiquidityUnbalancedBoosted( addLiquidityInput, { ...pool, protocolVersion: 3 }, ); diff --git a/src/entities/priceImpact/addLiquidityUnbalanced.ts b/src/entities/priceImpact/addLiquidityUnbalanced.ts new file mode 100644 index 00000000..61bfd2ec --- /dev/null +++ b/src/entities/priceImpact/addLiquidityUnbalanced.ts @@ -0,0 +1,184 @@ +import { abs, max, min } from '@/utils'; +import { AddLiquidity } from '../addLiquidity'; +import { AddLiquidityUnbalancedInput } from '../addLiquidity/types'; +import { PriceImpactAmount } from '../priceImpactAmount'; +import { RemoveLiquidity } from '../removeLiquidity'; +import { + RemoveLiquidityInput, + RemoveLiquidityKind, +} from '../removeLiquidity/types'; +import { Token } from '../token'; +import { TokenAmount } from '../tokenAmount'; +import { PoolState } from '../types'; +import { priceImpactABA } from './helper'; +import { SwapKind } from '@/types'; +import { Swap, SwapInput } from '../swap'; + +export const addLiquidityUnbalanced = async ( + input: AddLiquidityUnbalancedInput, + poolState: PoolState, +): Promise => { + // inputs are being validated within AddLiquidity + + // simulate adding liquidity to get amounts in + const addLiquidity = new AddLiquidity(); + let amountsIn: TokenAmount[]; + let bptOut: TokenAmount; + let poolTokens: Token[]; + try { + const queryResult = await addLiquidity.query(input, poolState); + amountsIn = queryResult.amountsIn; + bptOut = queryResult.bptOut; + poolTokens = amountsIn.map((a) => a.token); + } catch (err) { + throw new Error( + `addLiquidityUnbalanced operation will fail at SC level with user defined input.\n${err}`, + ); + } + + // simulate removing liquidity to get amounts out + const removeLiquidity = new RemoveLiquidity(); + const removeLiquidityInput: RemoveLiquidityInput = { + chainId: input.chainId, + rpcUrl: input.rpcUrl, + bptIn: bptOut.toInputAmount(), + kind: RemoveLiquidityKind.Proportional, + }; + const { amountsOut } = await removeLiquidity.query( + removeLiquidityInput, + poolState, + ); + + // deltas between unbalanced and proportional amounts + const deltas = amountsOut.map((a, i) => a.amount - amountsIn[i].amount); + + // get how much BPT each delta would mint + const deltaBPTs: bigint[] = []; + for (let i = 0; i < deltas.length; i++) { + if (deltas[i] === 0n) { + deltaBPTs.push(0n); + } else { + try { + deltaBPTs.push(await queryAddLiquidityForTokenDelta(i)); + } catch (err) { + throw new Error( + `Unexpected error while calculating addLiquidityUnbalanced PI at Delta add step:\n${err}`, + ); + } + } + } + + // zero out deltas by swapping between tokens from proportionalAmounts + // to exactAmountsIn, leaving the remaining delta within a single token + let remainingDeltaIndex = 0; + if (deltaBPTs.some((deltaBPT) => deltaBPT !== 0n)) { + remainingDeltaIndex = await zeroOutDeltas(deltas, deltaBPTs); + } + + // get relevant amount for price impact calculation + const deltaAmount = TokenAmount.fromRawAmount( + amountsIn[remainingDeltaIndex].token, + abs(deltas[remainingDeltaIndex]), + ); + + // calculate price impact using ABA method + return priceImpactABA( + amountsIn[remainingDeltaIndex], + amountsIn[remainingDeltaIndex].sub(deltaAmount), + ); + + // helper functions + + async function zeroOutDeltas(deltas: bigint[], deltaBPTs: bigint[]) { + let minNegativeDeltaIndex = deltaBPTs.findIndex( + (deltaBPT) => deltaBPT === max(deltaBPTs.filter((a) => a < 0n)), + ); + const nonZeroDeltasBPTs = deltaBPTs.filter((d) => d !== 0n); + for (let i = 0; i < nonZeroDeltasBPTs.length - 1; i++) { + const minPositiveDeltaIndex = deltaBPTs.findIndex( + (deltaBPT) => deltaBPT === min(deltaBPTs.filter((a) => a > 0n)), + ); + minNegativeDeltaIndex = deltaBPTs.findIndex( + (deltaBPT) => deltaBPT === max(deltaBPTs.filter((a) => a < 0n)), + ); + + let swapKind: SwapKind; + let givenTokenIndex: number; + let resultTokenIndex: number; + let inputAmountRaw = 0n; + let outputAmountRaw = 0n; + if ( + deltaBPTs[minPositiveDeltaIndex] < + abs(deltaBPTs[minNegativeDeltaIndex]) + ) { + swapKind = SwapKind.GivenIn; + givenTokenIndex = minPositiveDeltaIndex; + resultTokenIndex = minNegativeDeltaIndex; + inputAmountRaw = abs(deltas[givenTokenIndex]); + } else { + swapKind = SwapKind.GivenOut; + givenTokenIndex = minNegativeDeltaIndex; + resultTokenIndex = minPositiveDeltaIndex; + outputAmountRaw = abs(deltas[givenTokenIndex]); + } + try { + const swapInput: SwapInput = { + chainId: input.chainId, + paths: [ + { + tokens: [ + poolTokens[ + minPositiveDeltaIndex + ].toInputToken(), + poolTokens[ + minNegativeDeltaIndex + ].toInputToken(), + ], + pools: [poolState.id], + inputAmountRaw, + outputAmountRaw, + protocolVersion: poolState.protocolVersion, + }, + ], + swapKind, + }; + const swap = new Swap(swapInput); + const result = await swap.query(input.rpcUrl); + const resultAmount = + result.swapKind === SwapKind.GivenIn + ? result.expectedAmountOut + : result.expectedAmountIn; + + deltas[givenTokenIndex] = 0n; + deltaBPTs[givenTokenIndex] = 0n; + deltas[resultTokenIndex] = + deltas[resultTokenIndex] + resultAmount.amount; + deltaBPTs[resultTokenIndex] = + await queryAddLiquidityForTokenDelta(resultTokenIndex); + } catch (err) { + throw new Error( + `Unexpected error while calculating addLiquidityUnbalanced PI at Swap step:\n${err}`, + ); + } + } + return minNegativeDeltaIndex; + } + + async function queryAddLiquidityForTokenDelta( + tokenIndex: number, + ): Promise { + const absDelta = TokenAmount.fromRawAmount( + poolTokens[tokenIndex], + abs(deltas[tokenIndex]), + ); + const { bptOut: deltaBPT } = await addLiquidity.query( + { + ...input, + amountsIn: [absDelta.toInputAmount()], + }, + poolState, + ); + const signal = deltas[tokenIndex] >= 0n ? 1n : -1n; + return deltaBPT.amount * signal; + } +}; diff --git a/src/entities/priceImpact/addLiquidityUnbalancedBoosted.ts b/src/entities/priceImpact/addLiquidityUnbalancedBoosted.ts index 98b84263..ae5a9919 100644 --- a/src/entities/priceImpact/addLiquidityUnbalancedBoosted.ts +++ b/src/entities/priceImpact/addLiquidityUnbalancedBoosted.ts @@ -5,7 +5,7 @@ import { RemoveLiquidityKind } from '../removeLiquidity/types'; import { Swap, SwapInput, TokenApi } from '../swap'; import { TokenAmount } from '../tokenAmount'; import { PoolStateWithUnderlyings, PoolTokenWithUnderlying } from '../types'; -import { priceImpactABA } from '.'; +import { priceImpactABA } from './helper'; import { AddLiquidityBoostedUnbalancedInput } from '../addLiquidityBoosted/types'; import { AddLiquidityBoostedV3 } from '../addLiquidityBoosted'; import { RemoveLiquidityBoostedV3 } from '../removeLiquidityBoosted'; diff --git a/src/entities/priceImpact/helper.ts b/src/entities/priceImpact/helper.ts new file mode 100644 index 00000000..a96f1a01 --- /dev/null +++ b/src/entities/priceImpact/helper.ts @@ -0,0 +1,18 @@ +import { MathSol } from '@/utils'; +import { PriceImpactAmount } from '../priceImpactAmount'; +import { TokenAmount } from '../tokenAmount'; + +/** + * Applies the ABA method to calculate the price impact of an operation. + * @param initialA amount of token A at the begginig of the ABA process, i.e. A -> B amountIn + * @param finalA amount of token A at the end of the ABA process, i.e. B -> A amountOut + * @returns + */ + +export const priceImpactABA = (initialA: TokenAmount, finalA: TokenAmount) => { + const priceImpact = MathSol.divDownFixed( + initialA.scale18 - finalA.scale18, + initialA.scale18 * 2n, + ); + return PriceImpactAmount.fromRawAmount(priceImpact); +}; diff --git a/src/entities/priceImpact/index.ts b/src/entities/priceImpact/index.ts index 5d228868..79048f9b 100644 --- a/src/entities/priceImpact/index.ts +++ b/src/entities/priceImpact/index.ts @@ -1,6 +1,3 @@ -import { MathSol, abs, max, min } from '../../utils'; -import { SwapKind } from '../../types'; - import { AddLiquidity } from '../addLiquidity'; import { AddLiquidityKind, @@ -17,7 +14,6 @@ import { } from '../removeLiquidity/types'; import { RemoveLiquidityNested } from '../removeLiquidityNested'; import { RemoveLiquidityNestedSingleTokenInputV2 } from '../removeLiquidityNested/removeLiquidityNestedV2/types'; -import { Swap, SwapInput } from '../swap'; import { TokenAmount } from '../tokenAmount'; import { NestedPoolState, PoolState, PoolStateWithUnderlyings } from '../types'; import { getSortedTokens } from '../utils'; @@ -26,7 +22,10 @@ import { AddLiquidityNested } from '../addLiquidityNested'; import { AddLiquidityBoostedUnbalancedInput } from '../addLiquidityBoosted/types'; import { addLiquidityUnbalancedBoosted } from './addLiquidityUnbalancedBoosted'; import { addLiquidityNested } from './addLiquidityNested'; -import { Token } from '../token'; +import { priceImpactABA } from './helper'; +import { addLiquidityUnbalanced } from './addLiquidityUnbalanced'; + +export * from './helper'; export class PriceImpact { /** @@ -106,171 +105,7 @@ export class PriceImpact { input: AddLiquidityUnbalancedInput, poolState: PoolState, ): Promise => { - // inputs are being validated within AddLiquidity - - // simulate adding liquidity to get amounts in - const addLiquidity = new AddLiquidity(); - let amountsIn: TokenAmount[]; - let bptOut: TokenAmount; - let poolTokens: Token[]; - try { - const queryResult = await addLiquidity.query(input, poolState); - amountsIn = queryResult.amountsIn; - bptOut = queryResult.bptOut; - poolTokens = amountsIn.map((a) => a.token); - } catch (err) { - throw new Error( - `addLiquidityUnbalanced operation will fail at SC level with user defined input.\n${err}`, - ); - } - - // simulate removing liquidity to get amounts out - const removeLiquidity = new RemoveLiquidity(); - const removeLiquidityInput: RemoveLiquidityInput = { - chainId: input.chainId, - rpcUrl: input.rpcUrl, - bptIn: bptOut.toInputAmount(), - kind: RemoveLiquidityKind.Proportional, - }; - const { amountsOut } = await removeLiquidity.query( - removeLiquidityInput, - poolState, - ); - - // deltas between unbalanced and proportional amounts - const deltas = amountsOut.map((a, i) => a.amount - amountsIn[i].amount); - - // get how much BPT each delta would mint - const deltaBPTs: bigint[] = []; - for (let i = 0; i < deltas.length; i++) { - if (deltas[i] === 0n) { - deltaBPTs.push(0n); - } else { - try { - deltaBPTs.push(await queryAddLiquidityForTokenDelta(i)); - } catch (err) { - throw new Error( - `Unexpected error while calculating addLiquidityUnbalanced PI at Delta add step:\n${err}`, - ); - } - } - } - - // zero out deltas by swapping between tokens from proportionalAmounts - // to exactAmountsIn, leaving the remaining delta within a single token - let remainingDeltaIndex = 0; - if (deltaBPTs.some((deltaBPT) => deltaBPT !== 0n)) { - remainingDeltaIndex = await zeroOutDeltas(deltas, deltaBPTs); - } - - // get relevant amount for price impact calculation - const deltaAmount = TokenAmount.fromRawAmount( - amountsIn[remainingDeltaIndex].token, - abs(deltas[remainingDeltaIndex]), - ); - - // calculate price impact using ABA method - return priceImpactABA( - amountsIn[remainingDeltaIndex], - amountsIn[remainingDeltaIndex].sub(deltaAmount), - ); - - // helper functions - - async function zeroOutDeltas(deltas: bigint[], deltaBPTs: bigint[]) { - let minNegativeDeltaIndex = deltaBPTs.findIndex( - (deltaBPT) => deltaBPT === max(deltaBPTs.filter((a) => a < 0n)), - ); - const nonZeroDeltasBPTs = deltaBPTs.filter((d) => d !== 0n); - for (let i = 0; i < nonZeroDeltasBPTs.length - 1; i++) { - const minPositiveDeltaIndex = deltaBPTs.findIndex( - (deltaBPT) => - deltaBPT === min(deltaBPTs.filter((a) => a > 0n)), - ); - minNegativeDeltaIndex = deltaBPTs.findIndex( - (deltaBPT) => - deltaBPT === max(deltaBPTs.filter((a) => a < 0n)), - ); - - let swapKind: SwapKind; - let givenTokenIndex: number; - let resultTokenIndex: number; - let inputAmountRaw = 0n; - let outputAmountRaw = 0n; - if ( - deltaBPTs[minPositiveDeltaIndex] < - abs(deltaBPTs[minNegativeDeltaIndex]) - ) { - swapKind = SwapKind.GivenIn; - givenTokenIndex = minPositiveDeltaIndex; - resultTokenIndex = minNegativeDeltaIndex; - inputAmountRaw = abs(deltas[givenTokenIndex]); - } else { - swapKind = SwapKind.GivenOut; - givenTokenIndex = minNegativeDeltaIndex; - resultTokenIndex = minPositiveDeltaIndex; - outputAmountRaw = abs(deltas[givenTokenIndex]); - } - try { - const swapInput: SwapInput = { - chainId: input.chainId, - paths: [ - { - tokens: [ - poolTokens[ - minPositiveDeltaIndex - ].toInputToken(), - poolTokens[ - minNegativeDeltaIndex - ].toInputToken(), - ], - pools: [poolState.id], - inputAmountRaw, - outputAmountRaw, - protocolVersion: poolState.protocolVersion, - }, - ], - swapKind, - }; - const swap = new Swap(swapInput); - const result = await swap.query(input.rpcUrl); - const resultAmount = - result.swapKind === SwapKind.GivenIn - ? result.expectedAmountOut - : result.expectedAmountIn; - - deltas[givenTokenIndex] = 0n; - deltaBPTs[givenTokenIndex] = 0n; - deltas[resultTokenIndex] = - deltas[resultTokenIndex] + resultAmount.amount; - deltaBPTs[resultTokenIndex] = - await queryAddLiquidityForTokenDelta(resultTokenIndex); - } catch (err) { - throw new Error( - `Unexpected error while calculating addLiquidityUnbalanced PI at Swap step:\n${err}`, - ); - } - } - return minNegativeDeltaIndex; - } - - async function queryAddLiquidityForTokenDelta( - tokenIndex: number, - ): Promise { - const absDelta = TokenAmount.fromRawAmount( - poolTokens[tokenIndex], - abs(deltas[tokenIndex]), - ); - const { bptOut: deltaBPT } = await addLiquidity.query( - { - ...input, - amountsIn: [absDelta.toInputAmount()], - }, - poolState, - ); - const signal = deltas[tokenIndex] >= 0n ? 1n : -1n; - return deltaBPT.amount * signal; - } + return addLiquidityUnbalanced(input, poolState); }; static async addLiquidityUnbalancedBoosted( @@ -393,17 +228,3 @@ export class PriceImpact { return priceImpactABA(bptAmountIn, bptOut); }; } - -/** - * Applies the ABA method to calculate the price impact of an operation. - * @param initialA amount of token A at the begginig of the ABA process, i.e. A -> B amountIn - * @param finalA amount of token A at the end of the ABA process, i.e. B -> A amountOut - * @returns - */ -export const priceImpactABA = (initialA: TokenAmount, finalA: TokenAmount) => { - const priceImpact = MathSol.divDownFixed( - initialA.scale18 - finalA.scale18, - initialA.scale18 * 2n, - ); - return PriceImpactAmount.fromRawAmount(priceImpact); -}; diff --git a/test/v3/utils/getPoolStateWithBalancesV3.integration.test.ts b/test/v3/utils/getPoolStateWithBalancesV3.integration.test.ts new file mode 100644 index 00000000..e97bdde9 --- /dev/null +++ b/test/v3/utils/getPoolStateWithBalancesV3.integration.test.ts @@ -0,0 +1,101 @@ +// pnpm test -- v3/utils/getPoolStateWithBalancesV3.integration.test.ts + +import { config } from 'dotenv'; +config(); + +import { + Hex, + PoolState, + ChainId, + PoolType, + getPoolStateWithBalancesV3, + PoolStateWithBalances, +} from '@/index'; +import { POOLS, TOKENS } from '../../lib/utils'; +import { ANVIL_NETWORKS, startFork } from '../../anvil/anvil-global-setup'; + +const protocolVersion = 3; + +const chainId = ChainId.SEPOLIA; + +const poolId = POOLS[chainId].MOCK_USDC_DAI_POOL.id; +const USDC = TOKENS[chainId].USDC; +const DAI = TOKENS[chainId].DAI; + +describe('add liquidity test', () => { + let poolState: PoolState; + let rpcUrl: string; + + beforeAll(async () => { + // setup mock api + const api = new MockApi(); + + // get pool state from api + poolState = await api.getPool(poolId); + + ({ rpcUrl } = await startFork( + ANVIL_NETWORKS[ChainId[chainId]], + undefined, + 7057106n, + )); + }); + + describe('getPoolStateWithBalancesV3', () => { + test('<18 decimals tokens', async () => { + const poolStateWithBalances = await getPoolStateWithBalancesV3( + poolState, + chainId, + rpcUrl, + ); + + const mockData: PoolStateWithBalances = { + ...poolState, + tokens: [ + { + address: USDC.address, + decimals: USDC.decimals, + index: 0, + balance: '9585.21526', + }, + { + address: DAI.address, + decimals: DAI.decimals, + index: 1, + balance: '10256.288668913000293429', + }, + ], + totalShares: '9912.817276660069114899', + }; + + expect(poolStateWithBalances).to.deep.eq(mockData); + }); + }); +}); + +/*********************** Mock To Represent API Requirements **********************/ +class MockApi { + public async getPool(id: Hex): Promise { + const tokens = [ + { + address: USDC.address, + decimals: USDC.decimals, + index: 0, + }, + { + address: DAI.address, + decimals: DAI.decimals, + index: 1, + }, + ]; + + return { + id, + address: id, + type: PoolType.Weighted, + tokens, + protocolVersion, + }; + } +} + +/******************************************************************************/