From b17e935b879ec1092b2ad2f0ffd01e2ad5264031 Mon Sep 17 00:00:00 2001 From: Alexander Burkut Date: Thu, 12 Dec 2024 15:31:24 +0300 Subject: [PATCH] implement getTopPoolsForToken and added integration tests --- src/dex/cables/cables-integration.test.ts | 108 +++++++++++++++---- src/dex/cables/cables.ts | 122 +++++++++++++++++++++- 2 files changed, 211 insertions(+), 19 deletions(-) diff --git a/src/dex/cables/cables-integration.test.ts b/src/dex/cables/cables-integration.test.ts index aa9c4f50b..e619f5ba9 100644 --- a/src/dex/cables/cables-integration.test.ts +++ b/src/dex/cables/cables-integration.test.ts @@ -294,26 +294,98 @@ describe('Cables', function () { ); }); - it.skip('getTopPoolsForToken', async function () { - // We have to check without calling initializePricing, because - // pool-tracker is not calling that function - const cables = new Cables(network, dexKey, dexHelper); - const poolLiquidity = await cables.getTopPoolsForToken( - tokens[tokenASymbol].address, - 10, - ); - console.log( - `${tokenASymbol} Top Pools:`, - JSON.stringify(poolLiquidity, null, 2), - ); + describe('getTopPoolsForToken', () => { + it('USDC getTopPoolsForToken', async function () { + // We have to check without calling initializePricing, because + // pool-tracker is not calling that function + const tokenSymbol = 'USDC'; + const cables = new Cables(network, dexKey, dexHelper); + const poolLiquidity = await cables.getTopPoolsForToken( + tokens[tokenSymbol].address, + 10, + ); + console.log( + `${tokenASymbol} Top Pools:`, + JSON.stringify(poolLiquidity, null, 2), + ); - if (!cables.hasConstantPriceLargeAmounts) { - checkPoolsLiquidity( - poolLiquidity, - Tokens[network][tokenASymbol].address, - dexKey, + if (!cables.hasConstantPriceLargeAmounts) { + checkPoolsLiquidity( + poolLiquidity, + Tokens[network][tokenASymbol].address, + dexKey, + ); + } + }); + + it('WETH getTopPoolsForToken', async function () { + // We have to check without calling initializePricing, because + // pool-tracker is not calling that function + const tokenSymbol = 'WETH'; + const cables = new Cables(network, dexKey, dexHelper); + const poolLiquidity = await cables.getTopPoolsForToken( + tokens[tokenSymbol].address, + 10, ); - } + console.log( + `${tokenASymbol} Top Pools:`, + JSON.stringify(poolLiquidity, null, 2), + ); + + if (!cables.hasConstantPriceLargeAmounts) { + checkPoolsLiquidity( + poolLiquidity, + Tokens[network][tokenASymbol].address, + dexKey, + ); + } + }); + + it('ETH getTopPoolsForToken', async function () { + // We have to check without calling initializePricing, because + // pool-tracker is not calling that function + const tokenSymbol = 'ETH'; + const cables = new Cables(network, dexKey, dexHelper); + const poolLiquidity = await cables.getTopPoolsForToken( + tokens[tokenSymbol].address, + 10, + ); + console.log( + `${tokenASymbol} Top Pools:`, + JSON.stringify(poolLiquidity, null, 2), + ); + + if (!cables.hasConstantPriceLargeAmounts) { + checkPoolsLiquidity( + poolLiquidity, + Tokens[network][tokenASymbol].address, + dexKey, + ); + } + }); + + it('ARB getTopPoolsForToken', async function () { + // We have to check without calling initializePricing, because + // pool-tracker is not calling that function + const tokenSymbol = 'ARB'; + const cables = new Cables(network, dexKey, dexHelper); + const poolLiquidity = await cables.getTopPoolsForToken( + tokens[tokenSymbol].address, + 10, + ); + console.log( + `${tokenASymbol} Top Pools:`, + JSON.stringify(poolLiquidity, null, 2), + ); + + if (!cables.hasConstantPriceLargeAmounts) { + checkPoolsLiquidity( + poolLiquidity, + Tokens[network][tokenASymbol].address, + dexKey, + ); + } + }); }); }); }); diff --git a/src/dex/cables/cables.ts b/src/dex/cables/cables.ts index c81785384..f9f889358 100644 --- a/src/dex/cables/cables.ts +++ b/src/dex/cables/cables.ts @@ -56,6 +56,8 @@ import BigNumber from 'bignumber.js'; import { ethers } from 'ethers'; import { BI_MAX_UINT256 } from '../../bigint-constants'; import { SpecialDex } from '../../executor/types'; +import { uin256DecodeToFloat } from '../../lib/decoders'; +import _ from 'lodash'; export class Cables extends SimpleExchange implements IDex { public static dexKeysWithNetwork: { key: string; networks: Network[] }[] = @@ -690,7 +692,125 @@ export class Cables extends SimpleExchange implements IDex { tokenAddress: Address, limit: number, ): Promise { - return []; + const isETH = tokenAddress.toLowerCase() === ETHER_ADDRESS; + const denormalizedTokenAddress = isETH ? NULL_ADDRESS : tokenAddress; + + const tokens = (await this.getCachedTokens()) as { [key: string]: Token }; + const token = Object.values(tokens).find( + token => + token.address.toLowerCase() === denormalizedTokenAddress.toLowerCase(), + ); + + if (!token) { + return []; + } + + const pairs = (await this.getCachedPairs()) as { + [key: string]: { base: string; quote: string }; + }; + + const connectorTokens = Object.keys(pairs) + .filter(pairKey => { + const { base, quote } = pairs[pairKey]; + + if ( + base.toLowerCase() === token.symbol?.toLowerCase() || + quote.toLowerCase() === token.symbol?.toLowerCase() + ) { + return true; + } + + return false; + }) + .map(pairKey => { + const { base, quote } = pairs[pairKey]; + + if (base.toLowerCase() === token.symbol?.toLowerCase()) { + return tokens[quote]; + } + + if (quote.toLowerCase() === token.symbol?.toLowerCase()) { + return tokens[base]; + } + }); + + if (connectorTokens.length === 0) { + return []; + } + + connectorTokens.push(token); + + const tokensBalanceMultiCall = connectorTokens.map(token => { + let erc20BalanceCalldata; + if (token?.address.toLowerCase() === NULL_ADDRESS) { + erc20BalanceCalldata = this.dexHelper.multiContract.methods + .getEthBalance(this.mainnetRFQAddress) + .encodeABI(); + } else { + erc20BalanceCalldata = this.erc20Interface.encodeFunctionData( + 'balanceOf', + [this.mainnetRFQAddress], + ); + } + + return { + target: + token?.address.toLowerCase() === NULL_ADDRESS + ? this.dexHelper.config.data.multicallV2Address.toLowerCase() + : token?.address, + callData: erc20BalanceCalldata, + }; + }); + + const res = ( + await this.dexHelper.multiContract.methods + .aggregate(tokensBalanceMultiCall) + .call() + ).returnData; + + const balances = res.map((item: any) => { + if (item === '0x') { + return 0n; + } + return BigInt(item.toString()); + }); + + const connectorsPricesUSD = await Promise.all( + connectorTokens.map(async (token, index) => + this.dexHelper.getTokenUSDPrice(token!, balances[index]), + ), + ); + + const extendedConnectors = connectorTokens.map((token, index) => ({ + ...token, + usdPrice: connectorsPricesUSD[index], + })); + + const tokenUSDPrice = _.last(extendedConnectors)!.usdPrice; + extendedConnectors.pop(); // to remove token which was added before + + const connectors = extendedConnectors.map(connector => { + return { + exchange: this.dexKey, + address: this.mainnetRFQAddress, + connectorTokens: [ + { + address: + connector.address === NULL_ADDRESS + ? ETHER_ADDRESS + : connector!.address, + decimals: connector!.decimals, + }, + ], + liquidityUSD: Number(connector!.usdPrice) + Number(tokenUSDPrice), + }; + }); + + return _.slice( + _.sortBy(connectors as PoolLiquidity[], [pool => -1 * pool.liquidityUSD]), + 0, + limit, + ); } /**