diff --git a/.changeset/eleven-trains-kneel.md b/.changeset/eleven-trains-kneel.md new file mode 100644 index 00000000..753d4307 --- /dev/null +++ b/.changeset/eleven-trains-kneel.md @@ -0,0 +1,5 @@ +--- +"@balancer/sdk": minor +--- + +Add Fantom config. Update to have network specific vault addr. diff --git a/.changeset/few-mayflies-change.md b/.changeset/few-mayflies-change.md new file mode 100644 index 00000000..cb3c86ad --- /dev/null +++ b/.changeset/few-mayflies-change.md @@ -0,0 +1,10 @@ +--- +"@balancer/sdk": minor +--- + +Replace dataQueries with multicall (using Viem): + +- Add `BATCHSIZE` config per network +- Onchain calls correctly mapped to pool/version +- Filter bricked pools from vulnerability +- Fix scalingFactor scaling diff --git a/.github/workflows/checks.yml b/.github/workflows/checks.yml index 83ce0682..45400b72 100644 --- a/.github/workflows/checks.yml +++ b/.github/workflows/checks.yml @@ -47,6 +47,7 @@ jobs: env: ETHEREUM_RPC_URL: ${{ secrets.ETHEREUM_RPC_URL }} POLYGON_RPC_URL: ${{ secrets.POLYGON_RPC_URL }} + FANTOM_RPC_URL: ${{ secrets.FANTOM_RPC_URL }} build: name: Build diff --git a/debug/multicall.ts b/debug/multicall.ts new file mode 100644 index 00000000..1dd10c95 --- /dev/null +++ b/debug/multicall.ts @@ -0,0 +1,59 @@ +// bun run debug/multicall.ts +import { + OnChainPoolDataEnricher, + BATCHSIZE, + ProviderSwapOptions, + VAULT, +} from '../src'; +// rome-ignore lint/correctness/noUnusedVariables: +import { mainnet, polygonZkEvm } from 'viem/chains'; + +const chain = polygonZkEvm; +const rpc = 'https://rpc.ankr.com/polygon_zkevm'; + +const poolsQuery = `{ + pools(first: 10, orderBy: id, orderDirection: desc, where: { totalShares_gt: 0, poolType_contains: "ComposableStable" }) { + id + address + poolType + wrappedIndex + tokens { + address + decimals + index + } + } +}`; + +const pools = await fetch( + 'https://api.studio.thegraph.com/query/24660/balancer-polygon-zk-v2/version/latest', + // 'https://api.thegraph.com/subgraphs/name/balancer-labs/balancer-v2', + { + method: 'POST', + headers: { + 'Content-Type': 'application/json', + }, + body: JSON.stringify({ query: poolsQuery }), + }, +) + .then((res) => res.json()) + .then((res) => res.data); + +const onChainEnricher = new OnChainPoolDataEnricher( + chain.id, + rpc, + BATCHSIZE[chain.id], + VAULT[chain.id], +); +// const blockNumber = +const providerOptions: ProviderSwapOptions = { + // block: blockNumber, + timestamp: BigInt(Date.now()), +}; +const data = await onChainEnricher.fetchAdditionalPoolData( + pools, + providerOptions, +); +const enriched = onChainEnricher.enrichPoolsWithData(pools.pools, data); + +console.log(enriched, data.length); diff --git a/src/abi/balancerPoolDataQueries.ts b/src/abi/balancerPoolDataQueries.ts deleted file mode 100644 index 2ff1cf4c..00000000 --- a/src/abi/balancerPoolDataQueries.ts +++ /dev/null @@ -1,413 +0,0 @@ -export const balancerPoolDataQueriesAbi = [ - { - inputs: [ - { - internalType: 'contract IVault', - name: '_vault', - type: 'address', - }, - ], - stateMutability: 'nonpayable', - type: 'constructor', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getAmpForPools', - outputs: [ - { - internalType: 'uint256[]', - name: '', - type: 'uint256[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getInRecoveryModeForPools', - outputs: [ - { - internalType: 'bool[]', - name: '', - type: 'bool[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getIsPausedForPools', - outputs: [ - { - internalType: 'bool[]', - name: '', - type: 'bool[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getLinearTargetsForPools', - outputs: [ - { - internalType: 'uint256[][]', - name: '', - type: 'uint256[][]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getNormalizedWeightsForPools', - outputs: [ - { internalType: 'uint256[][]', name: '', type: 'uint256[][]' }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { internalType: 'bytes32[]', name: 'poolIds', type: 'bytes32[]' }, - { - components: [ - { - internalType: 'bool', - name: 'loadTokenBalanceUpdatesAfterBlock', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadTotalSupply', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadSwapFees', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadLinearWrappedTokenRates', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadLinearTargets', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadNormalizedWeights', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadScalingFactors', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadAmps', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadRates', - type: 'bool', - }, - { - internalType: 'uint256', - name: 'blockNumber', - type: 'uint256', - }, - { - internalType: 'enum TotalSupplyType[]', - name: 'totalSupplyTypes', - type: 'uint8[]', - }, - { - internalType: 'enum SwapFeeType[]', - name: 'swapFeeTypes', - type: 'uint8[]', - }, - { - internalType: 'uint256[]', - name: 'linearPoolIdxs', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'weightedPoolIdxs', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'scalingFactorPoolIdxs', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'ampPoolIdxs', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'ratePoolIdxs', - type: 'uint256[]', - }, - ], - internalType: 'struct PoolDataQueryConfig', - name: 'config', - type: 'tuple', - }, - ], - name: 'getPoolData', - outputs: [ - { - internalType: 'uint256[][]', - name: 'balances', - type: 'uint256[][]', - }, - { - internalType: 'uint256[]', - name: 'totalSupplies', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'swapFees', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'linearWrappedTokenRates', - type: 'uint256[]', - }, - { - internalType: 'uint256[][]', - name: 'linearTargets', - type: 'uint256[][]', - }, - { - internalType: 'uint256[][]', - name: 'weights', - type: 'uint256[][]', - }, - { - internalType: 'uint256[][]', - name: 'scalingFactors', - type: 'uint256[][]', - }, - { - internalType: 'uint256[]', - name: 'amps', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'rates', - type: 'uint256[]', - }, - { - internalType: 'uint256[]', - name: 'ignoreIdxs', - type: 'uint256[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'bytes32[]', - name: 'poolIds', - type: 'bytes32[]', - }, - { - components: [ - { - internalType: 'bool', - name: 'loadInRecoveryMode', - type: 'bool', - }, - { - internalType: 'bool', - name: 'loadIsPaused', - type: 'bool', - }, - ], - internalType: 'struct PoolStatusQueryConfig', - name: 'config', - type: 'tuple', - }, - ], - name: 'getPoolStatus', - outputs: [ - { - internalType: 'bool[]', - name: 'isPaused', - type: 'bool[]', - }, - { - internalType: 'bool[]', - name: 'inRecoveryMode', - type: 'bool[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'bytes32[]', - name: 'poolIds', - type: 'bytes32[]', - }, - { - internalType: 'uint256', - name: 'blockNumber', - type: 'uint256', - }, - ], - name: 'getPoolTokenBalancesWithUpdatesAfterBlock', - outputs: [ - { - internalType: 'uint256[][]', - name: '', - type: 'uint256[][]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getRateForPools', - outputs: [ - { - internalType: 'uint256[]', - name: '', - type: 'uint256[]', - }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getScalingFactorsForPools', - outputs: [ - { internalType: 'uint256[][]', name: '', type: 'uint256[][]' }, - ], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - { - internalType: 'enum SwapFeeType[]', - name: 'swapFeeTypes', - type: 'uint8[]', - }, - ], - name: 'getSwapFeePercentageForPools', - outputs: [{ internalType: 'uint256[]', name: '', type: 'uint256[]' }], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - { - internalType: 'enum TotalSupplyType[]', - name: 'totalSupplyTypes', - type: 'uint8[]', - }, - ], - name: 'getTotalSupplyForPools', - outputs: [{ internalType: 'uint256[]', name: '', type: 'uint256[]' }], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [ - { - internalType: 'address[]', - name: 'poolAddresses', - type: 'address[]', - }, - ], - name: 'getWrappedTokenRateForLinearPools', - outputs: [{ internalType: 'uint256[]', name: '', type: 'uint256[]' }], - stateMutability: 'view', - type: 'function', - }, - { - inputs: [], - name: 'vault', - outputs: [ - { internalType: 'contract IVault', name: '', type: 'address' }, - ], - stateMutability: 'view', - type: 'function', - }, -] as const; diff --git a/src/abi/index.ts b/src/abi/index.ts index db6ac5cb..a13e6d1b 100644 --- a/src/abi/index.ts +++ b/src/abi/index.ts @@ -1,5 +1,3 @@ export * from './vault'; export * from './balancerQueries'; -export * from './balancerPoolDataQueries'; export * from './erc20'; -export * from './tokenRates'; diff --git a/src/abi/tokenRates.ts b/src/abi/tokenRates.ts deleted file mode 100644 index 8270fe1c..00000000 --- a/src/abi/tokenRates.ts +++ /dev/null @@ -1,20 +0,0 @@ -export const tokenRatesFragmentAbi = [ - { - inputs: [], - name: 'getTokenRates', - outputs: [ - { - internalType: 'uint256', - name: 'rate0', - type: 'uint256', - }, - { - internalType: 'uint256', - name: 'rate1', - type: 'uint256', - }, - ], - stateMutability: 'view', - type: 'function', - }, -] as const; diff --git a/src/data/enrichers/onChainPoolDataEnricher.ts b/src/data/enrichers/onChainPoolDataEnricher.ts index 3903e565..2522c07e 100644 --- a/src/data/enrichers/onChainPoolDataEnricher.ts +++ b/src/data/enrichers/onChainPoolDataEnricher.ts @@ -1,5 +1,10 @@ -import { Address, createPublicClient, formatUnits, Hex, http } from 'viem'; -import { balancerPoolDataQueriesAbi, tokenRatesFragmentAbi } from '../../abi/'; +import { + Address, + createPublicClient, + formatUnits, + http, + PublicClient, +} from 'viem'; import { GetPoolsResponse, PoolDataEnricher, @@ -8,17 +13,11 @@ import { RawWeightedPoolToken, } from '../types'; -import { - CHAINS, - getPoolAddress, - poolHasActualSupply, - poolHasPercentFee, - poolHasVirtualSupply, - poolIsLinearPool, -} from '../../utils'; +import { CHAINS } from '../../utils'; import { HumanAmount, SwapOptions } from '../../types'; +import { fetchAdditionalPoolData } from '../onChainPoolDataViaReadContract'; -interface OnChainPoolData { +export interface OnChainPoolData { id: string; balances: readonly bigint[]; totalSupply: bigint; @@ -34,212 +33,34 @@ interface OnChainPoolData { isPaused: boolean; inRecoveryMode: boolean; - - queryFailed: boolean; -} - -enum TotalSupplyType { - TOTAL_SUPPLY = 0, - VIRTUAL_SUPPLY = 1, - ACTUAL_SUPPLY = 2, -} - -enum SwapFeeType { - SWAP_FEE_PERCENTAGE = 0, - PERCENT_FEE = 1, -} - -interface OnChainPoolDataQueryConfig { - loadTokenBalances: 'all' | 'updates-after-block' | 'none'; - blockNumber: bigint; - loadTotalSupply: boolean; - loadSwapFees: boolean; - loadLinearWrappedTokenRates: boolean; - loadLinearTargets: boolean; - loadWeightsForPools: { - poolIds?: string[]; - poolTypes?: string[]; - }; - loadAmpForPools: { - poolIds?: string[]; - poolTypes?: string[]; - }; - loadScalingFactorForPools: { - poolIds?: string[]; - poolTypes?: string[]; - }; - loadTokenRatesForPools: { - poolIds?: string[]; - poolTypes?: string[]; - poolTypeVersions?: number[]; - }; - loadRatesForPools: { - poolIds?: string[]; - poolTypes?: string[]; - }; - - loadInRecoveryMode: boolean; - loadIsPaused: boolean; } export class OnChainPoolDataEnricher implements PoolDataEnricher { - private readonly config: OnChainPoolDataQueryConfig; + private readonly client: PublicClient; constructor( private readonly chainId: number, private readonly rpcUrl: string, - private readonly balancerPoolQueriesAddress: Address, - config?: Partial, + private readonly batchSize: number, + private readonly vault: Address, ) { - this.config = { - loadTokenBalances: 'updates-after-block', - blockNumber: 0n, - loadTotalSupply: true, - loadSwapFees: true, - loadLinearWrappedTokenRates: true, - loadLinearTargets: true, - loadWeightsForPools: {}, - loadAmpForPools: {}, - loadScalingFactorForPools: {}, - loadRatesForPools: {}, - loadTokenRatesForPools: {}, - loadInRecoveryMode: true, - loadIsPaused: true, - ...config, - }; + this.client = createPublicClient({ + transport: http(this.rpcUrl, { timeout: 60_000 }), + chain: CHAINS[this.chainId], + }); } public async fetchAdditionalPoolData( data: GetPoolsResponse, options: SwapOptions, ): Promise { - const rawPools = data.pools; - - if (rawPools.length === 0) { - return []; - } - - const { - poolIds, - weightedPoolIdxs, - ampPoolIdxs, - linearPoolIdxs, - totalSupplyTypes, - scalingFactorPoolIdxs, - ratePoolIdxs, - swapFeeTypes, - tokenRatesPoolIdxs, - } = this.getPoolDataQueryParams(data); - - const client = createPublicClient({ - transport: http(this.rpcUrl), - chain: CHAINS[this.chainId], - }); - - const [ - balances, - totalSupplies, - swapFees, - linearWrappedTokenRates, - linearTargets, - weights, - scalingFactors, - amps, - rates, - ignoreIdxs, - ] = await client.readContract({ - address: this.balancerPoolQueriesAddress, - abi: balancerPoolDataQueriesAbi, - functionName: 'getPoolData', - args: [ - poolIds, - { - loadTokenBalanceUpdatesAfterBlock: - this.config.loadTokenBalances !== 'none', - loadTotalSupply: this.config.loadTotalSupply, - loadSwapFees: this.config.loadSwapFees, - loadLinearWrappedTokenRates: linearPoolIdxs.length > 0, - loadLinearTargets: linearPoolIdxs.length > 0, - loadNormalizedWeights: weightedPoolIdxs.length > 0, - loadScalingFactors: scalingFactorPoolIdxs.length > 0, - loadAmps: ampPoolIdxs.length > 0, - loadRates: ratePoolIdxs.length > 0, - blockNumber: - data.syncedToBlockNumber && - this.config.loadTokenBalances === 'updates-after-block' - ? data.syncedToBlockNumber - : 0n, - totalSupplyTypes, - swapFeeTypes, - linearPoolIdxs, - weightedPoolIdxs, - scalingFactorPoolIdxs, - ampPoolIdxs, - ratePoolIdxs, - }, - ], - blockNumber: options.block, - }); - - const [isPaused, inRecoveryMode] = await client.readContract({ - address: this.balancerPoolQueriesAddress, - abi: balancerPoolDataQueriesAbi, - functionName: 'getPoolStatus', - args: [ - poolIds, - { - loadInRecoveryMode: this.config.loadInRecoveryMode, - loadIsPaused: this.config.loadIsPaused, - }, - ], - blockNumber: options.block, - }); - - let tokenRates: (readonly [bigint, bigint] | undefined)[] = []; - if (tokenRatesPoolIdxs.length > 0) { - const call = { - abi: tokenRatesFragmentAbi, - functionName: 'getTokenRates', - } as const; - const poolAddressesWithTokenRates: readonly Address[] = poolIds - .filter((_, i) => tokenRatesPoolIdxs.includes(BigInt(i))) - .map((id) => getPoolAddress(id) as Address); - const results = await client.multicall({ - contracts: poolAddressesWithTokenRates.map((a) => ({ - address: a, - ...call, - })), - blockNumber: options.block, - }); - tokenRates = results.map((r) => r.result); - } - - return poolIds.map((_poolId, i) => ({ - id: poolIds[i], - balances: balances[i], - totalSupply: totalSupplies[i], - weights: weightedPoolIdxs.includes(BigInt(i)) - ? weights[weightedPoolIdxs.indexOf(BigInt(i))] - : undefined, - amp: ampPoolIdxs.includes(BigInt(i)) - ? amps[ampPoolIdxs.indexOf(BigInt(i))] - : undefined, - wrappedTokenRate: linearPoolIdxs.includes(BigInt(i)) - ? linearWrappedTokenRates[linearPoolIdxs.indexOf(BigInt(i))] - : undefined, - linearTargets: linearPoolIdxs.includes(BigInt(i)) - ? linearTargets[linearPoolIdxs.indexOf(BigInt(i))] - : undefined, - scalingFactors: scalingFactors[i], - rate: rates[i], - swapFee: swapFees[i], - tokenRates: tokenRatesPoolIdxs.includes(BigInt(i)) - ? tokenRates[tokenRatesPoolIdxs.indexOf(BigInt(i))] - : undefined, - inRecoveryMode: inRecoveryMode[i], - isPaused: isPaused[i], - queryFailed: ignoreIdxs.includes(BigInt(i)), - })); + return fetchAdditionalPoolData( + this.vault, + data.pools, + this.client, + options, + this.batchSize, + ); } public enrichPoolsWithData( @@ -291,173 +112,22 @@ export class OnChainPoolDataEnricher implements PoolDataEnricher { formatUnits(tokenRate, 18) as HumanAmount, ) : undefined, + lowerTarget: data?.linearTargets + ? (formatUnits(data.linearTargets[0], 18) as HumanAmount) + : 'lowerTarget' in pool + ? pool.lowerTarget + : undefined, + upperTarget: data?.linearTargets + ? (formatUnits(data.linearTargets[1], 18) as HumanAmount) + : 'upperTarget' in pool + ? pool.upperTarget + : undefined, + inRecoveryMode: data?.inRecoveryMode || false, + isPaused: data?.isPaused || false, }; }); } - private getPoolDataQueryParams(data: GetPoolsResponse) { - const poolIds: Hex[] = []; - const totalSupplyTypes: TotalSupplyType[] = []; - const linearPoolIdxs: bigint[] = []; - const weightedPoolIdxs: bigint[] = []; - const ampPoolIdxs: bigint[] = []; - const scalingFactorPoolIdxs: bigint[] = []; - const ratePoolIdxs: bigint[] = []; - const swapFeeTypes: SwapFeeType[] = []; - const tokenRatesPoolIdxs: bigint[] = []; - - const { - loadScalingFactorForPoolTypes, - loadScalingFactorForPoolIds, - loadWeightsForPoolTypes, - loadAmpForPoolTypes, - loadAmpForPoolIds, - loadWeightsForPoolIds, - loadTokenRatesForPoolIds, - loadTokenRatesForPoolTypes, - loadTokenRatesForPoolTypeVersions, - loadRatesForPoolIds, - loadRatesForPoolTypes, - } = this.getMergedFilterConfig(data); - - for (let i = 0; i < data.pools.length; i++) { - const pool = data.pools[i]; - - poolIds.push(pool.id); - - totalSupplyTypes.push( - poolHasVirtualSupply(pool.poolType) - ? TotalSupplyType.VIRTUAL_SUPPLY - : poolHasActualSupply(pool.poolType) - ? TotalSupplyType.ACTUAL_SUPPLY - : TotalSupplyType.TOTAL_SUPPLY, - ); - - if (poolIsLinearPool(pool.poolType)) { - linearPoolIdxs.push(BigInt(i)); - } - - if ( - loadWeightsForPoolTypes.has(pool.poolType) || - loadWeightsForPoolIds.has(pool.id) - ) { - weightedPoolIdxs.push(BigInt(i)); - } - - if ( - loadAmpForPoolTypes.has(pool.poolType) || - loadAmpForPoolIds.has(pool.id) - ) { - ampPoolIdxs.push(BigInt(i)); - } - - if ( - loadScalingFactorForPoolIds.has(pool.id) || - loadScalingFactorForPoolTypes.has(pool.poolType) - ) { - scalingFactorPoolIdxs.push(BigInt(i)); - } - - if ( - loadTokenRatesForPoolIds.has(pool.id) || - (loadTokenRatesForPoolTypes.has(pool.poolType) && - loadTokenRatesForPoolTypeVersions.has(pool.poolTypeVersion)) - ) { - tokenRatesPoolIdxs.push(BigInt(i)); - } - - if ( - loadRatesForPoolIds.has(pool.id) || - loadRatesForPoolTypes.has(pool.poolType) - ) { - ratePoolIdxs.push(BigInt(i)); - } - - if (this.config.loadSwapFees) { - swapFeeTypes.push( - poolHasPercentFee(pool.poolType) - ? SwapFeeType.PERCENT_FEE - : SwapFeeType.SWAP_FEE_PERCENTAGE, - ); - } - } - - return { - poolIds, - totalSupplyTypes, - linearPoolIdxs, - weightedPoolIdxs, - ampPoolIdxs, - scalingFactorPoolIdxs, - ratePoolIdxs, - swapFeeTypes, - tokenRatesPoolIdxs, - }; - } - - private getMergedFilterConfig({ - poolsWithActiveWeightUpdates = [], - poolsWithActiveAmpUpdates = [], - }: { - poolsWithActiveWeightUpdates?: string[]; - poolsWithActiveAmpUpdates?: string[]; - }) { - const { - loadWeightsForPools, - loadScalingFactorForPools, - loadAmpForPools, - loadTokenRatesForPools, - loadRatesForPools, - } = this.config; - - const loadWeightsForPoolIds = new Set([ - ...poolsWithActiveWeightUpdates, - ...(loadWeightsForPools.poolIds || []), - ]); - const loadAmpForPoolIds = new Set([ - ...poolsWithActiveAmpUpdates, - ...(loadAmpForPools.poolIds || []), - ]); - const loadScalingFactorForPoolIds = new Set( - loadScalingFactorForPools.poolIds || [], - ); - const loadRatesForPoolIds = new Set(loadRatesForPools.poolIds || []); - - const loadWeightsForPoolTypes = new Set( - loadWeightsForPools.poolTypes || [], - ); - const loadAmpForPoolTypes = new Set(loadAmpForPools.poolTypes || []); - const loadScalingFactorForPoolTypes = new Set( - loadScalingFactorForPools.poolTypes || [], - ); - const loadTokenRatesForPoolIds = new Set( - loadTokenRatesForPools.poolIds || [], - ); - const loadTokenRatesForPoolTypes = new Set( - loadTokenRatesForPools.poolTypes || [], - ); - const loadTokenRatesForPoolTypeVersions = new Set( - loadTokenRatesForPools.poolTypeVersions || [], - ); - const loadRatesForPoolTypes = new Set( - loadRatesForPools.poolTypes || [], - ); - - return { - loadWeightsForPoolIds, - loadAmpForPoolIds, - loadScalingFactorForPoolIds, - loadRatesForPoolIds, - loadWeightsForPoolTypes, - loadAmpForPoolTypes, - loadScalingFactorForPoolTypes, - loadTokenRatesForPoolIds, - loadTokenRatesForPoolTypes, - loadTokenRatesForPoolTypeVersions, - loadRatesForPoolTypes, - }; - } - private getPoolTokenRate({ pool, token, @@ -478,7 +148,8 @@ export class OnChainPoolDataEnricher implements PoolDataEnricher { } if (data?.scalingFactors) { - return formatUnits(data.scalingFactors[index], 18); + const decimalsDiff = 18 - token.decimals; + return formatUnits(data.scalingFactors[index], 18 + decimalsDiff); } return token.priceRate; diff --git a/src/data/onChainPoolDataViaReadContract.ts b/src/data/onChainPoolDataViaReadContract.ts new file mode 100644 index 00000000..04dafceb --- /dev/null +++ b/src/data/onChainPoolDataViaReadContract.ts @@ -0,0 +1,403 @@ +import { PublicClient, parseAbi, Address, Abi, Hex } from 'viem'; +import { getPoolAddress } from '../utils'; +import { OnChainPoolData } from './enrichers/onChainPoolDataEnricher'; +import { SwapOptions } from '../types'; + +type Result = + | { + error: Error; + result?: undefined; + status: 'failure'; + } + | { + error?: undefined; + result: any; + status: 'success'; + }; +[]; + +type Results = Result[]; + +const abi = parseAbi([ + 'function getPoolTokens(bytes32 poolId) view returns (address[] tokens, uint256 lastChangeBlock)', + 'function getSwapFeePercentage() view returns (uint256)', + 'function percentFee() view returns (uint256)', + 'function protocolPercentFee() view returns (uint256)', + 'function getNormalizedWeights() view returns (uint256[])', + 'function totalSupply() view returns (uint256)', + 'function getVirtualSupply() view returns (uint256)', + 'function getActualSupply() view returns (uint256)', + 'function getTargets() view returns (uint256 lowerTarget, uint256 upperTarget)', + 'function getTokenRates() view returns (uint256, uint256)', + 'function getWrappedTokenRate() view returns (uint256)', + 'function getAmplificationParameter() view returns (uint256 value, bool isUpdating, uint256 precision)', + 'function getPausedState() view returns (bool)', + 'function inRecoveryMode() view returns (bool)', + 'function getRate() view returns (uint256)', + 'function getScalingFactors() view returns (uint256[] memory)', +]); + +// Extract the functionName property values into a union type +type FunctionNameUnion = typeof abi[number]['name']; + +type BuildReturn = { + address: Address; + abi: Abi; + functionName: FunctionNameUnion; + args?: readonly [`0x${string}`]; +}; + +const getTotalSupplyFn = (poolType: string) => { + if (poolType.includes('Linear') || ['StablePhantom'].includes(poolType)) { + return 'getVirtualSupply'; + } else if (poolType === 'ComposableStable') { + return 'getActualSupply'; + } else { + return 'totalSupply'; + } +}; + +const getSwapFeeFn = (poolType: string) => { + if (poolType === 'Element') { + return 'percentFee'; + } else if (poolType === 'FX') { + return 'protocolPercentFee'; + } else { + return 'getSwapFeePercentage'; + } +}; + +const defaultCalls = { + count: 4, + build: (id: string, poolType: string, vault: Address): BuildReturn[] => [ + { + address: vault, + abi, + functionName: 'getPoolTokens', + args: [id as Hex], + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: getTotalSupplyFn(poolType), + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: getSwapFeeFn(poolType), + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getPausedState', + }, + ], + parse: (results: Results, shift: number) => { + return { + balances: results[shift].result[1], + totalSupply: results[shift + 1].result, + swapFee: results[shift + 2].result, + isPaused: results[shift + 3].result, + }; + }, +}; + +// These don't exist on some earlier pool versions +const defaultCallsAux = { + count: 2 + defaultCalls.count, + build: (id: string, poolType: string, vault: Address): BuildReturn[] => [ + ...defaultCalls.build(id, poolType, vault), + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'inRecoveryMode', + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getScalingFactors', + }, + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...{ + inRecoveryMode: results[shift + defaultCalls.count].result, + scalingFactors: results[shift + defaultCalls.count + +1].result, + }, + }), +}; + +const weightedCalls = { + count: 1, + build: (id: string): BuildReturn[] => [ + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getNormalizedWeights', + }, + ], + parse: (results: Results, shift: number) => ({ + weights: results[shift].result, + }), +}; + +const linearCalls = { + count: 3, + build: (id: string): BuildReturn[] => [ + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getTargets', + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getWrappedTokenRate', + }, + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getRate', + }, + ], + parse: (results: Results, shift: number) => ({ + linearTargets: results[shift].result, + wrappedTokenRate: results[shift + 1].result, + poolRate: results[shift + 2].result, + }), +}; + +const stableCalls = { + count: 1, + build: (id: string): BuildReturn[] => [ + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getAmplificationParameter', + }, + ], + parse: (results: Results, shift: number) => { + return { + amp: results[shift].result[0], + }; + }, +}; + +const gyroECalls = { + count: 1, + build: (id: string): BuildReturn[] => [ + { + address: getPoolAddress(id) as `0x${string}`, + abi, + functionName: 'getTokenRates', + }, + ], + parse: (results: Results, shift: number) => ({ + tokenRates: results[shift].result, + }), +}; + +const poolTypeCalls = ( + poolType: string, + poolTypeVersion: number, + vault: Address, +) => { + const do_nothing = { + count: 0, + build: () => [], + parse: () => ({}), + }; + + switch (poolType) { + case 'Weighted': + case 'LiquidityBootstrapping': + case 'Investment': { + if (poolTypeVersion === 1) { + return { + count: defaultCalls.count + weightedCalls.count, + build: (id: string) => [ + ...defaultCalls.build(id, poolType, vault), + ...weightedCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...weightedCalls.parse( + results, + shift + defaultCalls.count, + ), + }), + }; + } else + return { + count: defaultCallsAux.count + weightedCalls.count, + build: (id: string) => [ + ...defaultCallsAux.build(id, poolType, vault), + ...weightedCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCallsAux.parse(results, shift), + ...weightedCalls.parse( + results, + shift + defaultCallsAux.count, + ), + }), + }; + } + case 'Stable': { + if (poolTypeVersion === 1) { + return { + count: defaultCalls.count + stableCalls.count, + build: (id: string) => [ + ...defaultCalls.build(id, poolType, vault), + ...stableCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...stableCalls.parse( + results, + shift + defaultCalls.count, + ), + }), + }; + } else { + return { + count: defaultCallsAux.count + stableCalls.count, + build: (id: string) => [ + ...defaultCallsAux.build(id, poolType, vault), + ...stableCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCallsAux.parse(results, shift), + ...stableCalls.parse( + results, + shift + defaultCallsAux.count, + ), + }), + }; + } + } + case 'StablePhantom': + case 'MetaStable': + return { + count: defaultCalls.count + stableCalls.count, + build: (id: string) => [ + ...defaultCalls.build(id, poolType, vault), + ...stableCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...stableCalls.parse(results, shift + defaultCalls.count), + }), + }; + case 'ComposableStable': { + return { + count: defaultCallsAux.count + stableCalls.count, + build: (id: string) => [ + ...defaultCallsAux.build(id, poolType, vault), + ...stableCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCallsAux.parse(results, shift), + ...stableCalls.parse( + results, + shift + defaultCallsAux.count, + ), + }), + }; + } + case 'GyroE': + if (poolTypeVersion === 1) { + return defaultCalls; + } else { + return { + count: defaultCalls.count + gyroECalls.count, + build: (id: string) => [ + ...defaultCalls.build(id, poolType, vault), + ...gyroECalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...gyroECalls.parse( + results, + shift + defaultCalls.count, + ), + }), + }; + } + case 'AaveLinear': + if (poolTypeVersion === 1) { + return { + count: defaultCalls.count + linearCalls.count, + build: (id: string) => [ + ...defaultCalls.build(id, poolType, vault), + ...linearCalls.build(id), + ], + parse: (results: Results, shift: number) => ({ + ...defaultCalls.parse(results, shift), + ...linearCalls.parse( + results, + shift + defaultCalls.count, + ), + }), + }; + } else { + return defaultCallsAux; + } + default: + return do_nothing; + } +}; + +export const fetchAdditionalPoolData = async ( + vault: Address, + pools: { + id: string; + poolType: string; + poolTypeVersion: number; + }[], + client: PublicClient, + options: SwapOptions, + batchSize: number, +): Promise => { + if (pools.length === 0) { + return []; + } + + const calls = pools.flatMap(({ id, poolType, poolTypeVersion }) => + poolTypeCalls(poolType, poolTypeVersion, vault).build( + id, + poolType, + vault, + ), + ); + + const results = await client.multicall({ + contracts: calls, + batchSize: batchSize, + blockNumber: options.block, + }); + + results.forEach((r, i) => { + if (r.status === 'failure') + console.error( + 'Failed request in multicall', + calls[i].address, + calls[i].functionName, + r.error, + ); + }); + + let shift = 0; + + return pools.map(({ id, poolType, poolTypeVersion }) => { + const result = { + id, + ...poolTypeCalls(poolType, poolTypeVersion, vault).parse( + results, + shift, + ), + } as OnChainPoolData; + shift += poolTypeCalls(poolType, poolTypeVersion, vault).count; + return result; + }); +}; diff --git a/src/data/providers/subgraphPoolProvider.ts b/src/data/providers/subgraphPoolProvider.ts index 2d8c1106..08b0f91c 100644 --- a/src/data/providers/subgraphPoolProvider.ts +++ b/src/data/providers/subgraphPoolProvider.ts @@ -5,7 +5,7 @@ import { RawPool, } from '../types'; import { fetchWithRetry } from '../../utils/fetch'; -import { SUBGRAPH_URLS } from '../../utils'; +import { SUBGRAPH_URLS, brickedPools } from '../../utils'; BigInt.prototype['toJSON'] = function () { return this.toString(); @@ -72,6 +72,7 @@ export class SubgraphPoolProvider implements PoolDataProvider { ? ['Investment', 'LiquidityBootstrapping'] : undefined, ...config, + poolIdNotIn: [...(config?.poolIdNotIn ?? []), ...brickedPools], }; } diff --git a/src/data/types.ts b/src/data/types.ts index 5cf7c36d..a913a37d 100644 --- a/src/data/types.ts +++ b/src/data/types.ts @@ -12,7 +12,6 @@ export type SupportedRawPoolTypes = | 'MetaStable' | 'ComposableStable' | 'StablePhantom' - | 'Element' | 'FX' | 'Gyro2' | 'Gyro3' diff --git a/src/entities/swap.ts b/src/entities/swap.ts index 10b37d28..733a0ef7 100644 --- a/src/entities/swap.ts +++ b/src/entities/swap.ts @@ -157,7 +157,7 @@ export class Swap { }); const queriesContract = getContract({ - address: BALANCER_QUERIES, + address: BALANCER_QUERIES[this.chainId], abi: balancerQueriesAbi, publicClient, }); diff --git a/src/sor.ts b/src/sor.ts index 7274077d..5a13785b 100644 --- a/src/sor.ts +++ b/src/sor.ts @@ -1,11 +1,6 @@ import { Router } from './router'; import { BasePool, Path, Token, TokenAmount, Swap } from './entities'; -import { - BALANCER_POOL_DATA_QUERIES_ADDRESSES, - ChainId, - checkInputs, - SUBGRAPH_URLS, -} from './utils'; +import { ChainId, checkInputs, SUBGRAPH_URLS, BATCHSIZE, VAULT } from './utils'; import { SorConfig, SwapInputRawAmount, SwapKind, SwapOptions } from './types'; import { PoolParser } from './entities/pools/parser'; import { OnChainPoolDataEnricher, SubgraphPoolProvider } from './data'; @@ -39,7 +34,8 @@ export class SmartOrderRouter { new OnChainPoolDataEnricher( chainId, rpcUrl, - BALANCER_POOL_DATA_QUERIES_ADDRESSES[chainId], + BATCHSIZE[chainId], + VAULT[chainId], ); this.poolDataService = new PoolDataService( Array.isArray(poolDataProviders) diff --git a/src/utils/constants.ts b/src/utils/constants.ts index 3ecee182..02bd4737 100644 --- a/src/utils/constants.ts +++ b/src/utils/constants.ts @@ -13,6 +13,7 @@ import { polygonZkEvm, zkSync, zkSyncTestnet, + fantom, } from 'viem/chains'; export const ZERO_ADDRESS: Address = @@ -60,6 +61,7 @@ export enum ChainId { ARBITRUM_ONE = 42161, AVALANCHE = 43114, BASE_GOERLI = 84531, + FANTOM = 250, } export const CHAINS: Record = { @@ -75,6 +77,18 @@ export const CHAINS: Record = { [ChainId.ARBITRUM_ONE]: arbitrum, [ChainId.AVALANCHE]: avalanche, [ChainId.BASE_GOERLI]: baseGoerli, + [ChainId.FANTOM]: fantom, +}; + +export const VAULT: Record = { + [ChainId.ARBITRUM_ONE]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.AVALANCHE]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.GNOSIS_CHAIN]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.MAINNET]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.OPTIMISM]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.POLYGON]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.ZKEVM]: '0xBA12222222228d8Ba445958a75a0704d566BF2C8', + [ChainId.FANTOM]: '0x20dd72Ed959b6147912C2e529F0a0C651c33c9ce', }; export const SUBGRAPH_URLS = { @@ -100,6 +114,8 @@ export const SUBGRAPH_URLS = { 'https://api.thegraph.com/subgraphs/name/balancer-labs/balancer-avalanche-v2', [ChainId.BASE_GOERLI]: 'https://api.thegraph.com/subgraphs/name/balancer-labs/balancer-base-goerli-v2', + [ChainId.FANTOM]: + 'https://api.thegraph.com/subgraphs/name/beethovenxfi/beethovenx-v2-fantom', }; export const STELLATE_URLS = { @@ -111,15 +127,26 @@ export const STELLATE_URLS = { [ChainId.ARBITRUM_ONE]: 'https://balancer-arbitrum-v2.stellate.balancer.fi', }; -export const BALANCER_QUERIES = '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5'; -export const BALANCER_POOL_DATA_QUERIES_ADDRESSES: Record = { - [ChainId.ARBITRUM_ONE]: '0x7Ba29fE8E83dd6097A7298075C4AFfdBda3121cC', - [ChainId.AVALANCHE]: '0x67af5D428d38C5176a286a2371Df691cDD914Fb8', - [ChainId.GNOSIS_CHAIN]: '0x3f170631ed9821Ca51A59D996aB095162438DC10', - [ChainId.MAINNET]: '0xf5CDdF6feD9C589f1Be04899F48f9738531daD59', - [ChainId.OPTIMISM]: '0x6B5dA774890Db7B7b96C6f44e6a4b0F657399E2e', - [ChainId.POLYGON]: '0x84813aA3e079A665C0B80F944427eE83cBA63617', - [ChainId.ZKEVM]: '0xF24917fB88261a37Cc57F686eBC831a5c0B9fD39', +export const BALANCER_QUERIES: Record = { + [ChainId.ARBITRUM_ONE]: '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5', + [ChainId.AVALANCHE]: '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5', + [ChainId.GNOSIS_CHAIN]: '0x0f3e0c4218b7b0108a3643cfe9d3ec0d4f57c54e', + [ChainId.MAINNET]: '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5', + [ChainId.OPTIMISM]: '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5', + [ChainId.POLYGON]: '0xE39B5e3B6D74016b2F6A9673D7d7493B6DF549d5', + [ChainId.ZKEVM]: '0x809b79b53f18e9bc08a961ed4678b901ac93213a', + [ChainId.FANTOM]: '0x1B0A42663DF1edeA171cD8732d288a81EFfF6d23', +}; + +export const BATCHSIZE: Record = { + [ChainId.ARBITRUM_ONE]: 800, + [ChainId.AVALANCHE]: 800, + [ChainId.GNOSIS_CHAIN]: 800, + [ChainId.MAINNET]: 800, + [ChainId.OPTIMISM]: 800, + [ChainId.POLYGON]: 800, + [ChainId.ZKEVM]: 128, + [ChainId.FANTOM]: 128, }; export const NATIVE_ASSETS = { @@ -163,6 +190,14 @@ export const NATIVE_ASSETS = { 'Ether', '0x82aF49447D8a07e3bd95BD0d56f35241523fBab1', ), + [ChainId.FANTOM]: new Token( + ChainId.FANTOM, + NATIVE_ADDRESS, + 18, + 'FANTOM', + 'Fantom', + '0x21be370D5312f44cB42ce377BC9b8a0cEF1A4C83', + ), }; export const ETH = NATIVE_ASSETS[ChainId.MAINNET]; @@ -175,3 +210,340 @@ export const DEFAULT_FUND_MANAGMENT = { }; export const DEFAULT_USERDATA = '0x'; + +// Pools from `23 Vulnerability that are bricked/broken +export const brickedPools = [ + '0x00c2a4be503869fa751c2dbcb7156cc970b5a8da000000000000000000000477', + '0x02d928e68d8f10c0358566152677db51e1e2dc8c00000000000000000000051e', + '0x04248aabca09e9a1a3d5129a7ba05b7f17de768400000000000000000000050e', + '0x05513ca725b6ce035ca2641075474eb469f05f4c00020000000000000000041f', + '0x0a0fb4ff697de5ac5b6770cd8ee1b72af80b57cf000000000000000000000496', + '0x0afbd58beca09545e4fb67772faf3858e610bcd00000000000000000000004b9', + '0x0d05aac44ac7dd3c7ba5d50be93eb884a057d23400000000000000000000051c', + '0x11839d635e2f0270da37e8ef4324d4d5d54329570002000000000000000004d8', + '0x126e7643235ec0ab9c103c507642dc3f4ca23c66000000000000000000000468', + '0x133d241f225750d2c92948e464a5a80111920331000000000000000000000476', + '0x159cb00338fb63f263fd6f621df619cef71da9540000000000000000000004d5', + '0x173063a30e095313eee39411f07e95a8a806014e0002000000000000000003ab', + '0x1bd2f176a812e312077bca87e37c08432bb09f3e0000000000000000000005a1', + '0x20b156776114e8a801e9767d90c6ccccc8adf398000000000000000000000499', + '0x246ffb4d928e394a02e45761fecdba6c2e79b8eb000000000000000000000541', + '0x25accb7943fd73dda5e23ba6329085a3c24bfb6a000200000000000000000387', + '0x26c2b83fc8535deead276f5cc3ad9c1a2192e02700020000000000000000056b', + '0x2b218683178d029bab6c9789b1073aa6c96e517600000000000000000000058c', + '0x2ba7aa2213fa2c909cd9e46fed5a0059542b36b00000000000000000000003a3', + '0x2bbf681cc4eb09218bee85ea2a5d3d13fa40fc0c0000000000000000000000fd', + '0x2e52c64fd319e380cdbcfc4577ea1fda558a32e40002000000000000000005ba', + '0x2f4eb100552ef93840d5adc30560e5513dfffacb000000000000000000000334', + '0x2ff1a9dbdacd55297452cfd8a4d94724bc22a5f7000000000000000000000484', + '0x3035917be42af437cbdd774be26b9ec90a2bd677000200000000000000000543', + '0x331d50e0b00fc1c32742f151e56b9b616227e23e00000000000000000000047c', + '0x334c96d792e4b26b841d28f53235281cec1be1f200020000000000000000038a', + '0x335d1709d4da9aca59d16328db5cd4ea66bfe06b0000000000000000000004d6', + '0x395d8a1d9ad82b5abe558f8abbfe183b27138af40000000000000000000004e5', + '0x3bb22fc9033b802f2ac47c18885f63476f158afc000000000000000000000483', + '0x3c640f0d3036ad85afa2d5a9e32be651657b874f00000000000000000000046b', + '0x3cdae4f12a67ba563499e102f309c73213cb241c000000000000000000000335', + '0x3dbb8d974b82e82ce79c20c0f5995f4f1f533ede000000000000000000000470', + '0x3f7a7fd7f214be45ec26820fd01ac3be4fc75aa70002000000000000000004c5', + '0x3fcb7085b8f2f473f80bf6d879cae99ea4de934400000000000000000000056d', + '0x41503c9d499ddbd1dcdf818a1b05e9774203bf46000000000000000000000594', + '0x4228290ee9cab692938ff0b4ba303fbcdb68e9f200020000000000000000057d', + '0x454ed96955d04d2f5cdd05e0fd1c77975bfe5307000000000000000000000410', + '0x481c5fc05d63a58aa2f0f2aa417c021b5d419cb200000000000000000000056a', + '0x483006684f422a9448023b2382615c57c5ecf18f000000000000000000000488', + '0x4a82b580365cff9b146281ab72500957a849abdc000000000000000000000494', + '0x4c81255cc9ed7062180ea99962fe05ac0d57350b0000000000000000000005a3', + '0x4c8d2e60863e8d7e1033eda2b3d84e92a641802000000000000000000000040f', + '0x4cbde5c4b4b53ebe4af4adb85404725985406163000000000000000000000595', + '0x4ce0bd7debf13434d3ae127430e9bd4291bfb61f00020000000000000000038b', + '0x4ce277df0feb5b4d07a0ca2adcf5326e4005239d000000000000000000000518', + '0x4fd4687ec38220f805b6363c3c1e52d0df3b5023000200000000000000000473', + '0x4fd63966879300cafafbb35d157dc5229278ed230000000000000000000000e9', + '0x50cf90b954958480b8df7958a9e965752f62712400000000000000000000046f', + '0x53bc3cba3832ebecbfa002c12023f8ab1aa3a3a0000000000000000000000411', + '0x5a6a8cffb4347ff7fc484bf5f0f8a2e234d34255000200000000000000000275', + '0x5b3240b6be3e7487d61cd1afdfc7fe4fa1d81e6400000000000000000000037b', + '0x60683b05e9a39e3509d8fdb9c959f23170f8a0fa000000000000000000000489', + '0x60d604890feaa0b5460b28a424407c24fe89374a0000000000000000000004fc', + '0x639883476960a23b38579acfd7d71561a0f408cf000200000000000000000505', + '0x652d486b80c461c397b0d95612a404da936f3db30000000000000000000000e7', + '0x6667c6fa9f2b3fc1cc8d85320b62703d938e43850000000000000000000004fb', + '0x6a1eb2e9b45e772f55bd9a34659a04b6f75da68700000000000000000000040d', + '0x6c56e72c551b5ac4bf54a620a76077ca768c8fe40002000000000000000004da', + '0x70b7d3b3209a59fb0400e17f67f3ee8c37363f4900020000000000000000018f', + '0x7337224d59cb16c2dc6938cd45a7b2c60c865d6a0000000000000000000004d4', + '0x74cbfaf94a3577c539a9dcee9870a6349a33b34f000000000000000000000534', + '0x779d01f939d78a918a3de18cc236ee89221dfd4e0000000000000000000004c7', + '0x7b50775383d3d6f0215a8f290f2c9e2eebbeceb20000000000000000000000fe', + '0x804cdb9116a10bb78768d3252355a1b18067bf8f0000000000000000000000fb', + '0x813e3fe1761f714c502d1d2d3a7cceb33f37f59d00000000000000000000040c', + '0x82698aecc9e28e9bb27608bd52cf57f704bd1b83000000000000000000000336', + '0x8a6b25e33b12d1bb6929a8793961076bd1f9d3eb0002000000000000000003e8', + '0x8e6ec57a822c2f527f2df7c7d7d361df3e7530a1000000000000000000000498', + '0x8f4063446f5011bc1c9f79a819efe87776f23704000000000000000000000197', + '0x9001cbbd96f54a658ff4e6e65ab564ded76a543100000000000000000000050a', + '0x9210f1204b5a24742eba12f710636d76240df3d00000000000000000000000fc', + '0x9516a2d25958edb8da246a320f2c7d94a0dbe25d000000000000000000000519', + '0x959216bb492b2efa72b15b7aacea5b5c984c3cca000200000000000000000472', + '0x968024662b9566b42d78af23a0f441bc8723fa83000200000000000000000418', + '0x99c88ad7dc566616548adde8ed3effa730eb6c3400000000000000000000049a', + '0x9b1c8407a360443a9e5eca004713e4088fab8ac0000000000000000000000497', + '0x9b692f571b256140a39a34676bffa30634c586e100000000000000000000059d', + '0x9d7f992c900fbea0ec314bdd71b7cc1becf76a33000200000000000000000573', + '0x9fb771d530b0ceba5160f7bfe2dd1e8b8aa1340300000000000000000000040e', + '0xa13a9247ea42d743238089903570127dda72fe4400000000000000000000035d', + '0xa1697f9af0875b63ddc472d6eebada8c1fab85680000000000000000000004f9', + '0xa3823e50f20982656557a4a6a9c06ba5467ae9080000000000000000000000e6', + '0xa718042e5622099e5f0ace4e7122058ab39e1bbe000200000000000000000475', + '0xa8b103a10a94f4f2d7ed2fdcd5545e807557330700000000000000000000048e', + '0xac5b4ef7ede2f2843a704e96dcaa637f4ba3dc3f00000000000000000000051d', + '0xac976bb42cb0c85635644e8c7c74d0e0286aa61c0000000000000000000003cb', + '0xae37d54ae477268b9997d4161b96b8200755935c000000000000000000000337', + '0xae8535c23afedda9304b03c68a3563b75fc8f92b0000000000000000000005a0', + '0xb0f75e97a114a4eb4a425edc48990e6760726709000000000000000000000198', + '0xb5e3de837f869b0248825e0175da73d4e8c3db6b000200000000000000000474', + '0xb841b062ea8ccf5c4cb78032e91de4ae875560420002000000000000000005b7', + '0xb9bd68a77ccf8314c0dfe51bc291c77590c4e9e6000200000000000000000385', + '0xbb6881874825e60e1160416d6c426eae65f2459e000000000000000000000592', + '0xbc0f2372008005471874e426e86ccfae7b4de79d000000000000000000000485', + '0xbf2ef8bdc2fc0f3203b3a01778e3ec5009aeef3300000000000000000000058d', + '0xbfa413a2ff0f20456d57b643746133f54bfe0cd20000000000000000000004c3', + '0xc2b021133d1b0cf07dba696fd5dd89338428225b000000000000000000000598', + '0xc443c15033fcb6cf72cc24f1bda0db070ddd9786000000000000000000000593', + '0xc50d4347209f285247bda8a09fc1c12ce42031c3000000000000000000000590', + '0xc5dc1316ab670a2eed5716d7f19ced321191f38200000000000000000000056e', + '0xc8c79fcd0e859e7ec81118e91ce8e4379a481ee6000000000000000000000196', + '0xcaa052584b462198a5a9356c28bce0634d65f65c0000000000000000000004db', + '0xcbfa4532d8b2ade2c261d3dd5ef2a2284f7926920000000000000000000004fa', + '0xcfae6e251369467f465f13836ac8135bd42f8a56000000000000000000000591', + '0xd4e7c1f3da1144c9e2cfd1b015eda7652b4a439900000000000000000000046a', + '0xd6e355036f41dc261b3f1ed3bbc6003e87aadb4f000000000000000000000495', + '0xd7edb56f63b2a0191742aea32df1f98ca81ed9c600000000000000000000058e', + '0xd997f35c9b1281b82c8928039d14cddab5e13c2000000000000000000000019c', + '0xdba274b4d04097b90a72b62467d828cefd708037000000000000000000000486', + '0xdc063deafce952160ec112fa382ac206305657e60000000000000000000004c4', + '0xdec02e6642e2c999af429f5ce944653cad15e093000000000000000000000469', + '0xe03af00fabe8401560c1ff7d242d622a5b601573000000000000000000000493', + '0xe0fcbf4d98f0ad982db260f86cf28b49845403c5000000000000000000000504', + '0xe2d16b0a39f3fbb4389a0e8f1efcbecfb3d1e6e10000000000000000000005a7', + '0xe4dc3c1998ac693d68f4c77476d7c815694c3e94000200000000000000000416', + '0xe6bcc79f328eec93d4ec8f7ed35534d9ab549faa0000000000000000000000e8', + '0xe8c56405bc405840154d9b572927f4197d110de10000000000000000000005a4', + '0xeb486af868aeb3b6e53066abc9623b1041b42bc000000000000000000000046c', + '0xeb567dde03f3da7fe185bdacd5ab495ab220769d000000000000000000000548', + '0xec3626fee40ef95e7c0cbb1d495c8b67b34d398300000000000000000000053d', + '0xf22ff21e17157340575158ad7394e068048dd98b0000000000000000000004b8', + '0xf57c794f42da72b38c8f610ff3b5e8502e48cbde00000000000000000000055c', + '0xf71d0774b214c4cf51e33eb3d30ef98132e4dbaa00000000000000000000046e', + '0xfa24a90a3f2bbe5feea92b95cd0d14ce709649f900000000000000000000058f', + '0xfd11ccdbdb7ab91cb9427a6d6bf570c95876d1950000000000000000000004c2', + '0xfebb0bbf162e64fb9d0dfe186e517d84c395f016000000000000000000000502', + '0xfef969638c52899f91781f1be594af6f40b99bad00000000000000000000047b', + '0x02e139d53ebf4033bf78ab66c6a1e7f1f204487f0002000000000000000009f9', + '0x03090a9811181a2afe830a3a0b467698ccf3a8b1000000000000000000000bf5', + '0x0320c1c5b6df19a194d48882aaec1c72940081d9000000000000000000000a7d', + '0x04b54ea92d73de2d62d651db7d9778f0c49157d8000200000000000000000ba2', + '0x0503dd6b2d3dd463c9bef67fb5156870af63393e00000000000000000000042e', + '0x0889b240a5876aae745ac19f1771853671dc5d36000000000000000000000b3f', + '0x0bc54e914f53f98d16035f4f0d948f3e09c2fac0000200000000000000000bac', + '0x0c06e87c7b88d998f645b91c1f53b51294b12bca000100000000000000000bb9', + '0x10b040038f87219d9b42e025e3bd9b8095c87dd9000000000000000000000b11', + '0x117a3d474976274b37b7b94af5dcade5c90c6e85000000000000000000000aca', + '0x11884da90fb4221b3aa288a7741c51ec4fc43b2f000000000000000000000a5f', + '0x1379b816b9be611431d693290289c204720ca56d000100000000000000000b6f', + '0x150e7b885bdfce974f2abe88a72fdbd692175c6f0002000000000000000009fd', + '0x178e029173417b1f9c8bc16dcec6f697bc323746000000000000000000000758', + '0x1aafc31091d93c3ff003cff5d2d8f7ba2e7284250000000000000000000003b3', + '0x216690738aac4aa0c4770253ca26a28f0115c595000000000000000000000b2c', + '0x216d6db0c28204014618482c369d7fbf0a8f3232000100000000000000000b60', + '0x230ecdb2a7cee56d6889965a023aa0473d6da507000000000000000000000bf3', + '0x252ff6a3a6fd7b5e8e999de8e3f5c3b306ed1401000200000000000000000bec', + '0x25e57f4612912614e6c99616bd2abb9b5ae71e99000000000000000000000bf0', + '0x2645b13fd2c5295296e94a76280b968bdcbbdfed000000000000000000000c11', + '0x284eb68520c8fa83361c1a3a5910aec7f873c18b000000000000000000000ac9', + '0x2c8dbe8eb86135d9f2f26d196748c088d47f73e7000200000000000000000a29', + '0x31bccf9e28b94e5dacebaa67fe8bc1603cecd904000000000000000000000a01', + '0x341068a547c3cde3c09e338714010dd01b32f93f000200000000000000000a34', + '0x3db543faf7a92052de7860c5c9debabee59ed5bd000000000000000000000a62', + '0x3dd0843a028c86e0b760b1a76929d1c5ef93a2dd00000000000000000000070d', + '0x3efb91c4f9b103ee45885695c67794591916f34e000200000000000000000b43', + '0x402cfdb7781fa85d52f425352661128250b79e12000000000000000000000be3', + '0x43894de14462b421372bcfe445fa51b1b4a0ff3d000000000000000000000b36', + '0x4739e50b59b552d490d3fdc60d200977a38510c0000000000000000000000b10', + '0x48e6b98ef6329f8f0a30ebb8c7c960330d64808500000000000000000000075b', + '0x4a0b73f0d13ff6d43e304a174697e3d5cfd310a400020000000000000000091c', + '0x4a77ef015ddcd972fd9ba2c7d5d658689d090f1a000000000000000000000b38', + '0x4ae3661afa119892f0cc8c43edaf6a94989ac171000000000000000000000c06', + '0x4ccb966d8246240afb7a1a24628efb930870b1c40002000000000000000009fc', + '0x52cc8389c6b93d740325729cc7c958066cee4262000000000000000000000b0f', + '0x5b77107fcdf2b41903bab2bc555d4fc14cf7667d000000000000000000000b32', + '0x5bae72b75caab1f260d21bc028c630140607d6e8000000000000000000000ac6', + '0x600bd01b6526611079e12e1ff93aba7a3e34226f0000000000000000000009e4', + '0x63ce19ccd39930725b8a3d2733627804718ab83d000000000000000000000bf2', + '0x64efad69f099813021b41f4cac6e749fd55e188f000000000000000000000b39', + '0x6933ec1ca55c06a894107860c92acdfd2dd8512f000000000000000000000428', + '0x6abe4e7a497b8293c258389c1b00d177e4f257ed00010000000000000000080d', + '0x6c8c7fc50247a47038015eb1fd5dc105d05dafba000200000000000000000ba0', + '0x7079a25dec33be61bbd81b2fb69b468e80d3e72c0000000000000000000009ff', + '0x71bd10c2a590b5858f5576550c163976a48af906000000000000000000000b27', + '0x7c82a23b4c48d796dee36a9ca215b641c6a8709d000000000000000000000acd', + '0x7f4f4942f2a14b6ab7b08b10ada1aacede4ee8d4000200000000000000000b44', + '0x86aef31951e0a3a54333bd9e72f9a95587d058c5000200000000000000000912', + '0x882c7a84231484b3e9f3fd45ac04b1eb5d35b076000200000000000000000a91', + '0x894c82800526e0391e709c0983a5aea3718b7f6d000000000000000000000ac5', + '0x89b28a9494589b09dbccb69911c189f74fdadc5a000000000000000000000b33', + '0x89bb15076c9f2d86aa98ec6cffc1a71e31c38953000000000000000000000bf1', + '0x89f1146fee52b5d9166e9c83cc388b6d8f69f1380001000000000000000009e7', + '0x8a819a4cabd6efcb4e5504fe8679a1abd831dd8f00000000000000000000042d', + '0x8b58a1e7fff52001c22386c2918d45938a6a9be30001000000000000000008d9', + '0x8b8225bfedebaf1708c55743acb4ad43fd4d0f21000200000000000000000918', + '0x8fbd0f8e490735cfc3abf4f29cbddd5c3289b9a7000000000000000000000b5b', + '0x8fd39252d683fdb60bddd4df4b53c9380b496d59000200000000000000000b45', + '0x9321e2250767d79bab5aa06daa8606a2b3b7b4c5000000000000000000000bf4', + '0x949a12b95ec5b80c375b98963a5d6b33b0d0efff0002000000000000000009fe', + '0x9a020bdc2faff5bd24c6acc2020d01ff9f2c627a000000000000000000000ae2', + '0x9cf9358300e34bf9373d30129a1e718d8d058b54000200000000000000000913', + '0x9e34631547adcf2f8cefa0f5f223955c7b137571000000000000000000000ad5', + '0xa5a935833f6a5312715f182733eab088452335d7000100000000000000000bee', + '0xa5fe91dde37d8bf2dacacc0168b115d28ed03f84000000000000000000000b35', + '0xa8bf1c584519be0184311c48adbdc4c15cb2e8c1000000000000000000000bf6', + '0xab269164a10fab22bc87c39946da06c870b172d6000000000000000000000bfc', + '0xac2cae8d2f78a4a8f92f20dbe74042cd0a8d5af3000000000000000000000be2', + '0xae646817e458c0be890b81e8d880206710e3c44e000000000000000000000acb', + '0xaef2c171dbe64b0c18977e16e70bfd29d4ee0256000000000000000000000ac8', + '0xb0c830dceb4ef55a60192472c20c8bf19df03488000000000000000000000be1', + '0xb266ac3b7c98d7bcb28731dac0ef42dba1b276be000000000000000000000be4', + '0xb371aa09f5a110ab69b39a84b5469d29f9b22b76000000000000000000000b37', + '0xb3d658d5b95bf04e2932370dd1ff976fe18dd66a000000000000000000000ace', + '0xb54b2125b711cd183edd3dd09433439d5396165200000000000000000000075e', + '0xb59be8f3c85a9dd6e2899103b6fbf6ea405b99a4000000000000000000000b34', + '0xb878ecce26838fbba4f78cb5b791a0e09152c067000000000000000000000427', + '0xb973ca96a3f0d61045f53255e319aedb6ed4924000000000000000000000042f', + '0xbd4e35784c832d0f9049b54cb3609e5907c5b495000100000000000000000b14', + '0xc55ec796a4debe625d95436a3531f4950b11bdcf000000000000000000000b3e', + '0xc7e6389e364f4275eb442ef215ed21877028e2af000000000000000000000ac7', + '0xc83b55bbd005f1f84906545fcdb145dee53523e0000200000000000000000b30', + '0xcb21a9e647c95127ed784626485b3104cb28d0e7000000000000000000000425', + '0xd00f9ca46ce0e4a63067c4657986f0167b0de1e5000000000000000000000b42', + '0xd2f3b9e67c69762dd1c88f1d3dadd1649a190761000200000000000000000bf7', + '0xd4accb350f9cf59fe3cf7a5ee6ed9ace6a568ea9000200000000000000000b75', + '0xda1cd1711743e57dd57102e9e61b75f3587703da000000000000000000000acc', + '0xdae301690004946424e41051ace1791083be42a1000000000000000000000b40', + '0xde0a77ab6689b980c30306b10f9131a007e1af81000200000000000000000ba1', + '0xe051605a83deae38d26a7346b100ef1ac2ef8a0b0000000000000000000003ce', + '0xe1fb90d0d3b47e551d494d7ebe8f209753526b01000000000000000000000ac4', + '0xe2272cddb2cc408e79e02a73d1db9acc24a843d5000200000000000000000ba7', + '0xe2dc0e0f2c358d6e31836dee69a558ab8d1390e70000000000000000000009fa', + '0xe4885ed2818cc9e840a25f94f9b2a28169d1aea7000000000000000000000b29', + '0xe6909c2f18a29d97217a6146f045e1780606991f000100000000000000000bfe', + '0xe78b25c06db117fdf8f98583cdaaa6c92b79e917000000000000000000000b2b', + '0xea11645ac7d8f2def94c9d8d86bd766296c9b6b6000000000000000000000b3a', + '0xeb480dbbdd921cd6c359e4cc4c65ddea6395e2a1000200000000000000000946', + '0xed35f28f837e96f81240ebb82e0e3f518c7e8a2f000100000000000000000bb5', + '0xf0211cceebe6fcc45052b4e57ee95d233f5669d2000100000000000000000c01', + '0xf22a66046b5307842f21b311ecb4c462c24c0635000000000000000000000b15', + '0xf28f17be00f8ca3c9b7f66a4aad5513757fb3341000200000000000000000b5a', + '0xf42ed61450458ee4620f5ef4f29adb25a6ef0fb6000000000000000000000bf8', + '0xf48f01dcb2cbb3ee1f6aab0e742c2d3941039d56000000000000000000000445', + '0xf93579002dbe8046c43fefe86ec78b1112247bb8000000000000000000000759', + '0xf984eb2b8a7ef780245a797a2fccd82f346409ca000000000000000000000a59', + '0xfa2c0bd8327c99db5bde4c9e9e5cbf30946351bb000000000000000000000948', + '0xff4ce5aaab5a627bf82f4a571ab1ce94aa365ea600000000000000000000075a', + '0x1ac55c31dac78ca943cb8ebfca5945ce09e036e2000000000000000000000024', + '0x225e0047671939a8d78e08ebd692788abe63f15c000000000000000000000009', + '0x41211bba6d37f5a74b22e667533f080c7c7f3f1300000000000000000000000b', + '0x4de21b365d6543661d0e105e579a34b963862497000200000000000000000045', + '0x581ec1f5e7ced12b186deae32256adb53bdd5b08000000000000000000000001', + '0x66f33ae36dd80327744207a48122f874634b3ada000100000000000000000013', + '0xa3ed6f78edc29f69df8a0d16b1d1ccf9871f918c000000000000000000000032', + '0xa611a551b95b205ccd9490657acf7899daee5db700000000000000000000002e', + '0xb95829adbacd8af89e291dee78bc09e24de51d6b000000000000000000000043', + '0xb973ca96a3f0d61045f53255e319aedb6ed49240000200000000000000000011', + '0xba1a5b19d09a79dada039b1f974015c5a989d5fd000100000000000000000046', + '0xbb9cd48d33033f5effbedec9dd700c7d7e1dcf5000000000000000000000000e', + '0xd16f72b02da5f51231fde542a8b9e2777a478c8800000000000000000000000f', + '0xd4015683b8153666190e0b2bec352580ebc4caca00000000000000000000000d', + '0xe15cac1df3621e001f76210ab12a7f1a1691481f000000000000000000000044', + '0xe7f88d7d4ef2eb18fcf9dd7216ba7da1c46f3dd600000000000000000000000a', + '0xf48f01dcb2cbb3ee1f6aab0e742c2d3941039d56000200000000000000000012', + '0xfedb19ec000d38d92af4b21436870f115db22725000000000000000000000010', + '0xffff76a3280e95dc855696111c2562da09db2ac000000000000000000000000c', + '0x00fcd3d55085e998e291a0005cedecf58ac14c4000020000000000000000047f', + '0x077794c30afeccdf5ad2abc0588e8cee7197b71a000000000000000000000352', + '0x117a3d474976274b37b7b94af5dcade5c90c6e85000000000000000000000381', + '0x11884da90fb4221b3aa288a7741c51ec4fc43b2f000000000000000000000353', + '0x19b1c92631405a0a9495ccba0becf4f2e8e908bd000000000000000000000410', + '0x1e550b7764da9638fdd32c8a701364de31f45ee800000000000000000000047c', + '0x1fa7f727934226aedab636d62a084931b97d366b000000000000000000000411', + '0x23ca0306b21ea71552b148cf3c4db4fc85ae19290000000000000000000000c9', + '0x284eb68520c8fa83361c1a3a5910aec7f873c18b000000000000000000000380', + '0x2a96254ca32020b20ed3506f8f75318da24709f9000200000000000000000456', + '0x36942963e3b6f37ecc45a4e72349558514233f0000000000000000000000048a', + '0x3f53a862919ccfa023cb6ace91378a79fb0f6bf500000000000000000000040f', + '0x40af308e3d07ec769d85eb80afb116525ff4ac99000000000000000000000485', + '0x418de00ae109e6f874d872658767866d680eaa1900000000000000000000047d', + '0x45c4d1376943ab28802b995acffc04903eb5223f000000000000000000000470', + '0x4689122d360c4725d244c5cfea22861333d862e6000100000000000000000468', + '0x4739e50b59b552d490d3fdc60d200977a38510c0000000000000000000000409', + '0x49a0e3334496442a9706e481617724e7e37eaa080000000000000000000003ff', + '0x519cce718fcd11ac09194cff4517f12d263be067000000000000000000000382', + '0x52cc8389c6b93d740325729cc7c958066cee4262000000000000000000000408', + '0x567ecfcb22205d279bb8eed3e066989902bf03d5000000000000000000000452', + '0x585d95df0231fa08aeee35ff0c16b92fd0ecdc3300020000000000000000045f', + '0x5a7f39435fd9c381e4932fa2047c9a5136a5e3e7000000000000000000000400', + '0x5bae72b75caab1f260d21bc028c630140607d6e8000000000000000000000350', + '0x6cb787a419c3e6ee2e9ff365856c29cd10659113000000000000000000000474', + '0x7c82a23b4c48d796dee36a9ca215b641c6a8709d000000000000000000000406', + '0x81fc12c60ee5b753cf5fd0adc342dfb5f3817e3200000000000000000000035d', + '0x894c82800526e0391e709c0983a5aea3718b7f6d00000000000000000000034f', + '0x970712708a08e8fb152be4d81b2dc586923f5369000200000000000000000479', + '0x9bf7c3b63c77b4b4f2717776f15a4bec1b532a280000000000000000000000c8', + '0x9cebf13bb702f253abf1579294694a1edad00eaa000000000000000000000486', + '0x9e34631547adcf2f8cefa0f5f223955c7b137571000000000000000000000407', + '0x9fb7d6dcac7b6aa20108bad226c35b85a9e31b63000200000000000000000412', + '0xa1ea76c42b2938cfa9abea12357881006c52851300000000000000000000048f', + '0xa50f89e9f439fde2a6fe05883721a00475da3c4500000000000000000000048b', + '0xa612b6aed2e7ca1a3a4f23fbca9128461bbb7718000000000000000000000274', + '0xa8af146d79ac0bb981e4e0d8b788ec5711b1d5d000000000000000000000047b', + '0xad28940024117b442a9efb6d0f25c8b59e1c950b00000000000000000000046f', + '0xae646817e458c0be890b81e8d880206710e3c44e00000000000000000000039d', + '0xaef2c171dbe64b0c18977e16e70bfd29d4ee0256000000000000000000000351', + '0xbbf9d705b75f408cfcaee91da32966124d2c6f7d00000000000000000000047e', + '0xbd724eb087d4cc0f61a5fed1fffaf937937e14de000000000000000000000473', + '0xbe0f30217be1e981add883848d0773a86d2d2cd4000000000000000000000471', + '0xc46be4b8bb6b5a3d3120660efae9c5416318ed40000000000000000000000472', + '0xc69771058481551261709d8db44977e9afde645000010000000000000000042a', + '0xc6eee8cb7643ec2f05f46d569e9ec8ef8b41b389000000000000000000000475', + '0xcba9ff45cfb9ce238afde32b0148eb82cbe635620000000000000000000003fd', + '0xcf8b555b7754556cf2ac2165e77ee23ed8517d7900020000000000000000045e', + '0xd0dc20e6342db2de82692b8dc842301ff9121805000200000000000000000454', + '0xd3d5d45f4edf82ba0dfaf061d230766032a10e07000200000000000000000413', + '0xd6d20527c7b0669989ee082b9d3a1c63af742290000000000000000000000483', + '0xda1cd1711743e57dd57102e9e61b75f3587703da0000000000000000000003fc', + '0xe1fb90d0d3b47e551d494d7ebe8f209753526b0100000000000000000000034e', + '0xee02583596aee94cccb7e8ccd3921d955f17982a00000000000000000000040a', + '0xf984eb2b8a7ef780245a797a2fccd82f346409ca00000000000000000000034d', + '0xff8f84e8c87532af96aef5582ee451572233678b000200000000000000000478', + '0x054e7b0c73e1ee5aed6864fa511658fc2b54bcaa000000000000000000000015', + '0x3f1a2c4a3a751f6626bd90ef16e104f0772d4d6b00020000000000000000001b', + '0x7275c131b1f67e8b53b4691f92b0e35a4c1c6e22000000000000000000000010', + '0xa154009870e9b6431305f19b09f9cfd7284d4e7a000000000000000000000013', + '0xa1d14d922a575232066520eda11e27760946c991000000000000000000000012', + '0xa826a114b0c7db4d1ff4a4be845a78998c64564c000000000000000000000008', + '0xea67626e1f0b59e0d172a04f5702ef90bcdf440c00000000000000000000000f', + '0xeb496161099d45b3ea4892408ef745c6182eb56e00000000000000000000000e', + '0xece571847897fd61e764d455dc15cf1cd9de8d6f000000000000000000000014', + '0xed3e2f496cbcd8e212192fb8d1499842f04a0d19000000000000000000000009', + '0x02c9dcb975262719a61f9b40bdf0987ead9add3a000000000000000000000006', + '0x16c9a4d841e88e52b51936106010f27085a529ec00000000000000000000000c', + '0x32be2d0ddeaf3333501b24a28668ce373ba8e763000200000000000000000014', + '0x32f03464fdf909fdf3798f87ff3712b10c59bd86000000000000000000000005', + '0x4b718e0e2fea1da68b763cd50c446fba03ceb2ea00000000000000000000000b', + '0x68a69c596b3839023c0e08d09682314f582314e5000200000000000000000011', + '0x6f34a44fce1506352a171232163e7716dd073ade000200000000000000000015', + '0x9e2d87f904862671eb49cb358e74284762cc9f42000200000000000000000013', + '0xac4b72c01072a52b73ca71105504f1372efcce0d000000000000000000000003', + '0xbfd65c6160cfd638a85c645e6e6d8acac5dac935000000000000000000000004', + '0xe274c9deb6ed34cfe4130f8d0a8a948dea5bb28600000000000000000000000d', +]; diff --git a/test/fxPool.integration.test.ts b/test/fxPool.integration.test.ts index 7c629836..a4ef6217 100644 --- a/test/fxPool.integration.test.ts +++ b/test/fxPool.integration.test.ts @@ -3,7 +3,7 @@ import { beforeAll, beforeEach, describe, expect, test } from 'vitest'; import dotenv from 'dotenv'; dotenv.config(); -import { BALANCER_POOL_DATA_QUERIES_ADDRESSES, ChainId } from '../src/utils'; +import { BATCHSIZE, ChainId, VAULT } from '../src/utils'; import { BasePool, OnChainPoolDataEnricher, @@ -18,7 +18,7 @@ import { describe('fx integration tests', () => { const chainId = ChainId.POLYGON; - const rpcUrl = process.env['POLYGON_RPC_URL'] || 'https://polygon-rpc.com'; + const rpcUrl = process.env.POLYGON_RPC_URL || 'https://polygon-rpc.com'; const USDC = new Token( chainId, @@ -49,10 +49,8 @@ describe('fx integration tests', () => { const onChainPoolDataEnricher = new OnChainPoolDataEnricher( chainId, rpcUrl, - BALANCER_POOL_DATA_QUERIES_ADDRESSES[chainId], - { - loadSwapFees: false, - }, + BATCHSIZE[chainId], + VAULT[chainId], ); sor = new SmartOrderRouter({ diff --git a/test/gyroEV2Pool.integration.test.ts b/test/gyroEV2Pool.integration.test.ts index 6c621222..fa07611e 100644 --- a/test/gyroEV2Pool.integration.test.ts +++ b/test/gyroEV2Pool.integration.test.ts @@ -4,7 +4,7 @@ import dotenv from 'dotenv'; dotenv.config(); import testPools from './lib/testData/gyroETestPool.json'; -import { BALANCER_POOL_DATA_QUERIES_ADDRESSES, ChainId } from '../src/utils'; +import { BATCHSIZE, ChainId, VAULT } from '../src/utils'; import { BasePool, OnChainPoolDataEnricher, @@ -21,7 +21,7 @@ import { MockPoolProvider } from './lib/utils/mockPoolProvider'; describe('gyroEV2: WMATIC-stMATIC integration tests', () => { const chainId = ChainId.POLYGON; - const rpcUrl = process.env['POLYGON_RPC_URL'] || 'https://polygon-rpc.com'; + const rpcUrl = process.env.POLYGON_RPC_URL || 'https://polygon-rpc.com'; const WMATIC = new Token( chainId, '0x0d500b1d8e8ef31e21c99d1db9a6444d3adf1270', @@ -45,13 +45,8 @@ describe('gyroEV2: WMATIC-stMATIC integration tests', () => { const onChainPoolDataEnricher = new OnChainPoolDataEnricher( chainId, rpcUrl, - BALANCER_POOL_DATA_QUERIES_ADDRESSES[chainId], - { - loadTokenRatesForPools: { - poolTypes: ['GyroE'], - poolTypeVersions: [2], - }, - }, + BATCHSIZE[chainId], + VAULT[chainId], ); sor = new SmartOrderRouter({ diff --git a/test/sor.test.ts b/test/sor.test.ts index 2ac4e9a5..c3ca0667 100644 --- a/test/sor.test.ts +++ b/test/sor.test.ts @@ -6,189 +6,282 @@ dotenv.config(); import { SmartOrderRouter } from '../src/sor'; import { sorGetSwapsWithPools } from '../src/static'; import { SubgraphPoolProvider } from '../src/data/providers/subgraphPoolProvider'; -import { - BALANCER_POOL_DATA_QUERIES_ADDRESSES, - ChainId, - ETH, -} from '../src/utils'; +import { ChainId, ETH, NATIVE_ASSETS, BATCHSIZE, VAULT } from '../src/utils'; import { Token, TokenAmount } from '../src/entities'; import { OnChainPoolDataEnricher } from '../src/data/enrichers/onChainPoolDataEnricher'; import { SwapKind, SwapOptions } from '../src/types'; import { BasePool } from '../src/entities/pools'; -describe('SmartOrderRouter', () => { - describe('Mainnet', () => { - const chainId = ChainId.MAINNET; - const rpcUrl = - process.env['ETHEREUM_RPC_URL'] || 'https://eth.llamarpc.com'; - const subgraphPoolDataService = new SubgraphPoolProvider(chainId); - const onChainPoolDataEnricher = new OnChainPoolDataEnricher( - chainId, - rpcUrl, - BALANCER_POOL_DATA_QUERIES_ADDRESSES[chainId], - { - loadAmpForPools: { - poolTypes: ['ComposableStable'], - }, - }, - ); - - const sor = new SmartOrderRouter({ - chainId, - poolDataProviders: subgraphPoolDataService, - poolDataEnrichers: onChainPoolDataEnricher, - rpcUrl: rpcUrl, - }); +describe( + 'SmartOrderRouter', + () => { + describe('Mainnet', () => { + const chainId = ChainId.MAINNET; + const rpcUrl = + process.env.ETHEREUM_RPC_URL || 'https://eth.llamarpc.com'; + const subgraphPoolDataService = new SubgraphPoolProvider(chainId); + const onChainPoolDataEnricher = new OnChainPoolDataEnricher( + chainId, + rpcUrl, + BATCHSIZE[chainId], + VAULT[chainId], + ); - const BAL = new Token( - chainId, - '0xba100000625a3754423978a60c9317c58a424e3D', - 18, - 'BAL', - ); - const USDC = new Token( - chainId, - '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', - 6, - 'USDC', - ); - const USDT = new Token( - chainId, - '0xdAC17F958D2ee523a2206206994597C13D831ec7', - 6, - 'USDT', - ); - const DAI = new Token( - chainId, - '0x6B175474E89094C44Da98b954EedeAC495271d0F', - 18, - 'DAI', - ); - - const swapOptions: SwapOptions = { - block: 17473810n, - }; - - let pools: BasePool[]; - // Since constructing a Swap mutates the pool balances, we refetch for each test - // May be a better way to deep clone a BasePool[] class instead - beforeEach(async () => { - pools = await sor.fetchAndCachePools(swapOptions.block); - }); + const sor = new SmartOrderRouter({ + chainId, + poolDataProviders: subgraphPoolDataService, + poolDataEnrichers: onChainPoolDataEnricher, + rpcUrl: rpcUrl, + }); + + const BAL = new Token( + chainId, + '0xba100000625a3754423978a60c9317c58a424e3D', + 18, + 'BAL', + ); + const USDC = new Token( + chainId, + '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48', + 6, + 'USDC', + ); + const USDT = new Token( + chainId, + '0xdAC17F958D2ee523a2206206994597C13D831ec7', + 6, + 'USDT', + ); + const DAI = new Token( + chainId, + '0x6B175474E89094C44Da98b954EedeAC495271d0F', + 18, + 'DAI', + ); - describe('Weighted Pools', () => { - // ETH -> BAL swapGivenIn single hop - // Weighted pool - // 0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014 - test('Native ETH -> Token givenIn single hop', async () => { - const inputAmount = TokenAmount.fromHumanAmount(ETH, '1'); - - const swap = await sorGetSwapsWithPools( - ETH, - BAL, - SwapKind.GivenIn, - inputAmount, - pools, - swapOptions, - ); - - if (!swap) throw new Error('Swap is undefined'); - - const onchain = await swap.query(rpcUrl, swapOptions.block); - - expect(swap.quote.amount).toEqual(onchain.amount); - expect(swap.inputAmount.amount).toEqual(inputAmount.amount); - expect(swap.outputAmount.amount).toEqual(swap.quote.amount); - expect(swap.paths.length).toEqual(1); - expect(swap.paths[0].pools.length).toEqual(1); - expect(swap.paths[0].pools[0].id).toEqual( - '0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014', - ); + const swapOptions: SwapOptions = { + block: 17473810n, + }; + + let pools: BasePool[]; + // Since constructing a Swap mutates the pool balances, we refetch for each test + // May be a better way to deep clone a BasePool[] class instead + beforeEach(async () => { + pools = await sor.fetchAndCachePools(swapOptions.block); }); - // ETH -> BAL swapGivenOut single hop - // Weighted pool - // 0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014 - test('Native ETH -> Token givenOut single hop', async () => { - const outputAmount = TokenAmount.fromHumanAmount(BAL, '100'); - - const swap = await sorGetSwapsWithPools( - ETH, - BAL, - SwapKind.GivenOut, - outputAmount, - pools, - swapOptions, - ); - - if (!swap) throw new Error('Swap is undefined'); - - const onchain = await swap.query(rpcUrl, swapOptions.block); - - expect(swap.quote.amount).toEqual(onchain.amount); - expect(swap.inputAmount.amount).toEqual(swap.quote.amount); - expect(swap.outputAmount.amount).toEqual(outputAmount.amount); - expect(swap.paths.length).toEqual(1); - expect(swap.paths[0].pools.length).toEqual(1); - expect(swap.paths[0].pools[0].id).toEqual( - '0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014', - ); + describe('Weighted Pools', () => { + // ETH -> BAL swapGivenIn single hop + // Weighted pool + // 0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014 + test('Native ETH -> Token givenIn single hop', async () => { + const inputAmount = TokenAmount.fromHumanAmount(ETH, '1'); + + const swap = await sorGetSwapsWithPools( + ETH, + BAL, + SwapKind.GivenIn, + inputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + + expect(swap.quote.amount).toEqual(onchain.amount); + expect(swap.inputAmount.amount).toEqual(inputAmount.amount); + expect(swap.outputAmount.amount).toEqual(swap.quote.amount); + expect(swap.paths.length).toEqual(1); + expect(swap.paths[0].pools.length).toEqual(1); + expect(swap.paths[0].pools[0].id).toEqual( + '0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014', + ); + }); + + // ETH -> BAL swapGivenOut single hop + // Weighted pool + // 0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014 + test('Native ETH -> Token givenOut single hop', async () => { + const outputAmount = TokenAmount.fromHumanAmount( + BAL, + '100', + ); + + const swap = await sorGetSwapsWithPools( + ETH, + BAL, + SwapKind.GivenOut, + outputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + + expect(swap.quote.amount).toEqual(onchain.amount); + expect(swap.inputAmount.amount).toEqual(swap.quote.amount); + expect(swap.outputAmount.amount).toEqual( + outputAmount.amount, + ); + expect(swap.paths.length).toEqual(1); + expect(swap.paths[0].pools.length).toEqual(1); + expect(swap.paths[0].pools[0].id).toEqual( + '0x5c6ee304399dbdb9c8ef030ab642b10820db8f56000200000000000000000014', + ); + }); + }); + + describe('Stable Pools', () => { + test('DAI -> USDT givenIn ComposableStable', async () => { + const inputAmount = TokenAmount.fromHumanAmount( + DAI, + '100000', + ); + + const swap = await sorGetSwapsWithPools( + DAI, + USDT, + SwapKind.GivenIn, + inputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + expect(swap.quote.amount).toEqual(onchain.amount); + expect(swap.inputAmount.amount).toEqual(inputAmount.amount); + expect(swap.outputAmount.amount).toEqual(swap.quote.amount); + expect(swap.paths.length).toEqual(1); + expect(swap.paths[0].pools.length).toEqual(1); + }); + + test('USDC -> DAI givenOut ComposableStable', async () => { + const outputAmount = TokenAmount.fromHumanAmount( + DAI, + '1000000', + ); + + const swap = await sorGetSwapsWithPools( + USDC, + DAI, + SwapKind.GivenOut, + outputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + expect(swap.quote.amount).toEqual(onchain.amount); + }); }); }); + describe('Fantom', () => { + const chainId = ChainId.FANTOM; + const inputToken = NATIVE_ASSETS[chainId]; + const rpcUrl = process.env.FANTOM_RPC_URL || ''; + const subgraphPoolDataService = new SubgraphPoolProvider( + chainId, + undefined, + { + poolIdIn: [ + '0x9e4341acef4147196e99d648c5e43b3fc9d026780002000000000000000005ec', + ], + }, + ); + const onChainPoolDataEnricher = new OnChainPoolDataEnricher( + chainId, + rpcUrl, + BATCHSIZE[chainId], + VAULT[chainId], + ); + + const sor = new SmartOrderRouter({ + chainId, + poolDataProviders: subgraphPoolDataService, + poolDataEnrichers: onChainPoolDataEnricher, + rpcUrl: rpcUrl, + }); + + const BEETS = new Token( + chainId, + '0xF24Bcf4d1e507740041C9cFd2DddB29585aDCe1e', + 18, + 'BEETS', + ); + + const swapOptions: SwapOptions = { + block: 65313450n, + }; - describe('Stable Pools', () => { - // DAI -> bb-a-DAI -> bb-a-USDT -> USDT swapGivenIn boosted - // Aave Linear + Boosted Pool - // 0x6667c6fa9f2b3fc1cc8d85320b62703d938e43850000000000000000000004fb - // 0xfebb0bbf162e64fb9d0dfe186e517d84c395f016000000000000000000000502 - // 0xa1697f9af0875b63ddc472d6eebada8c1fab85680000000000000000000004f9 - test('DAI -> USDT givenIn boosted', async () => { - const inputAmount = TokenAmount.fromHumanAmount(DAI, '100000'); - - const swap = await sorGetSwapsWithPools( - DAI, - USDT, - SwapKind.GivenIn, - inputAmount, - pools, - swapOptions, - ); - - if (!swap) throw new Error('Swap is undefined'); - - const onchain = await swap.query(rpcUrl, swapOptions.block); - expect(swap.quote.amount).toEqual(onchain.amount); - expect(swap.inputAmount.amount).toEqual(inputAmount.amount); - expect(swap.outputAmount.amount).toEqual(swap.quote.amount); - expect(swap.paths.length).toEqual(1); - expect(swap.paths[0].pools.length).toEqual(3); + let pools: BasePool[]; + // Since constructing a Swap mutates the pool balances, we refetch for each test + // May be a better way to deep clone a BasePool[] class instead + beforeEach(async () => { + pools = await sor.fetchAndCachePools(swapOptions.block); }); - // DAI -> bb-a-DAI -> bb-a-USDT -> USDT swapGivenOut boosted - // Aave Linear + Boosted Pool - // 0x6667c6fa9f2b3fc1cc8d85320b62703d938e43850000000000000000000004fb - // 0xfebb0bbf162e64fb9d0dfe186e517d84c395f016000000000000000000000502 - // 0xa1697f9af0875b63ddc472d6eebada8c1fab85680000000000000000000004f9 - test('USDC -> DAI givenOut boosted', async () => { - const outputAmount = TokenAmount.fromHumanAmount( - DAI, - '1000000', - ); - - const swap = await sorGetSwapsWithPools( - USDC, - DAI, - SwapKind.GivenOut, - outputAmount, - pools, - swapOptions, - ); - - if (!swap) throw new Error('Swap is undefined'); - - const onchain = await swap.query(rpcUrl, swapOptions.block); - expect(swap.quote.amount).toEqual(onchain.amount); + describe('Native Swaps', () => { + test('Native -> Token givenIn', async () => { + const inputAmount = TokenAmount.fromHumanAmount( + inputToken, + '100', + ); + + const swap = await sorGetSwapsWithPools( + inputToken, + BEETS, + SwapKind.GivenIn, + inputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + + expect(swap.quote.amount).toEqual(onchain.amount); + expect(swap.inputAmount.amount).toEqual(inputAmount.amount); + expect(swap.outputAmount.amount).toEqual(swap.quote.amount); + }); + + test('Native ETH -> Token givenOut', async () => { + const outputAmount = TokenAmount.fromHumanAmount( + BEETS, + '100000', + ); + + const swap = await sorGetSwapsWithPools( + inputToken, + BEETS, + SwapKind.GivenOut, + outputAmount, + pools, + swapOptions, + ); + + if (!swap) throw new Error('Swap is undefined'); + + const onchain = await swap.query(rpcUrl, swapOptions.block); + + expect(swap.quote.amount).toEqual(onchain.amount); + expect(swap.inputAmount.amount).toEqual(swap.quote.amount); + expect(swap.outputAmount.amount).toEqual( + outputAmount.amount, + ); + }); }); }); - }); -}); + }, + { + timeout: 120000, + }, +); diff --git a/test/subgraph.test.ts b/test/subgraph.test.ts index de56c4eb..470c4c8e 100644 --- a/test/subgraph.test.ts +++ b/test/subgraph.test.ts @@ -3,18 +3,37 @@ import { SubgraphPoolProvider } from '../src/data/providers/subgraphPoolProvider import { ChainId } from '../src/utils'; import { ProviderSwapOptions } from '../src/data/types'; -describe('SubgraphPoolProvider', () => { - test('getPools mainnet', async () => { - const chainId = ChainId.MAINNET; - const subgraphPoolDataService = new SubgraphPoolProvider(chainId); +describe( + 'SubgraphPoolProvider', + () => { + test('getPools mainnet', async () => { + const chainId = ChainId.MAINNET; + const subgraphPoolDataService = new SubgraphPoolProvider(chainId); - const providerOptions: ProviderSwapOptions = { - timestamp: BigInt(Math.floor(new Date().getTime() / 1000)), - }; + const providerOptions: ProviderSwapOptions = { + timestamp: BigInt(Math.floor(new Date().getTime() / 1000)), + }; - const { pools } = await subgraphPoolDataService.getPools( - providerOptions, - ); - expect(pools.length).toBeGreaterThan(0); - }); -}); + const { pools } = await subgraphPoolDataService.getPools( + providerOptions, + ); + expect(pools.length).toBeGreaterThan(0); + }); + test('getPools fantom', async () => { + const chainId = ChainId.FANTOM; + const subgraphPoolDataService = new SubgraphPoolProvider(chainId); + + const providerOptions: ProviderSwapOptions = { + timestamp: BigInt(Math.floor(new Date().getTime() / 1000)), + }; + + const { pools } = await subgraphPoolDataService.getPools( + providerOptions, + ); + expect(pools.length).toBeGreaterThan(0); + }); + }, + { + timeout: 60000, + }, +);