@avalabs/evm-module 3.0.7 → 3.1.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/module.ts","../src/handlers/get-tokens/get-tokens.ts","../src/utils/get-provider.ts","../../../packages-internal/utils/src/services/token-service/token-service.ts","../../../packages-internal/utils/src/utils/retry.ts","../../../packages-internal/utils/src/utils/coingecko-retry.ts","../../../packages-internal/utils/src/utils/charsum.ts","../../../packages-internal/utils/src/utils/array-hash.ts","../../../packages-internal/utils/src/services/token-service/coingecko-proxy-client.ts","../../../packages-internal/utils/src/utils/fetch-and-verify.ts","../../../packages-internal/utils/src/services/token-service/watchlist-proxy-client.ts","../../../packages-internal/utils/src/services/pricing-service/exchange-rates.ts","../../../packages-internal/utils/src/utils/detail-item.ts","../../../packages-internal/utils/src/utils/get-core-headers.ts","../../../packages-internal/utils/src/consts.ts","../../../packages-internal/utils/src/utils/get-glacier-api-key.ts","../../../packages-internal/utils/src/utils/glacier-fetch-http-request.ts","../src/handlers/get-network-fee/get-network-fee.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/convert-transaction-normal.ts","../src/handlers/get-transaction-history/utils/get-explorer-address-by-network.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/convert-transaction-erc20.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/get-transaction-from-etherscan.ts","../src/handlers/get-transaction-history/utils/is-ethereum-chain-id.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-tx-type.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-sender-info.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-tokens.ts","../src/handlers/get-transaction-history/utils/resolve.ts","../src/handlers/get-transaction-history/utils/ipfs-resolver-with-fallback.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-nft-metadata.ts","../src/handlers/get-transaction-history/utils/get-small-image-for-nft.ts","../src/types.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/convert-transaction.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-transactions-from-glacier.ts","../src/handlers/get-transaction-history/get-transaction-history.ts","../manifest.json","../src/handlers/eth-send-transaction/eth-send-transaction.ts","../src/utils/get-nonce.ts","../src/utils/evm-tx-updater.ts","../src/utils/parse-erc20-transaction-type.ts","../src/utils/encode-erc20-approval.ts","../src/utils/estimate-gas-limit.ts","../src/utils/build-tx-approval-request.ts","../src/utils/process-transaction-simulation.ts","../src/utils/scan-transaction.ts","../src/utils/parse-erc20-tx.ts","../src/utils/type-utils.ts","../src/utils/transaction-alerts.ts","../src/handlers/eth-send-transaction/schema.ts","../src/utils/transaction-schema.ts","../src/utils/wait-for-transaction-receipt.ts","../src/utils/get-tx-hash.ts","../src/handlers/get-balances/get-balances.ts","../src/utils/find-async.ts","../src/utils/id-promise.ts","../src/services/rpc-service/rpc-service.ts","../src/env.ts","../src/services/glacier-service/glacier-service.ts","../src/constants.ts","../src/utils/ipsf-resolver-with-fallback.ts","../src/utils/get-small-image-for-nft.ts","../src/handlers/eth-sign/eth-sign.ts","../src/handlers/eth-sign/utils/beautify-message/beautify-message.ts","../src/handlers/eth-sign/schemas/parse-request-params/parse-request-params.ts","../src/handlers/eth-sign/schemas/eth-sign.ts","../src/handlers/eth-sign/schemas/shared.ts","../src/handlers/eth-sign/schemas/eth-sign-typed-data.ts","../src/handlers/eth-sign/schemas/personal-sign.ts","../src/handlers/eth-sign/utils/typeguards.ts","../src/handlers/eth-sign/utils/is-typed-data-valid.ts","../src/handlers/forward-to-rpc-node/forward-to-rpc-node.ts","../src/handlers/get-address/get-address.ts","../src/handlers/derive-address/derive-address.ts","../../../packages-internal/utils/src/utils/address-derivation.ts","../src/handlers/build-derivation-path/build-derivation-path.ts","../src/services/debank-service/debank-service.ts","../src/services/debank-service/de-bank.ts","../src/handlers/eth-send-transaction-batch/eth-send-transaction-batch.ts","../src/handlers/eth-send-transaction-batch/schema.ts","../src/handlers/eth-send-transaction-batch/utils/ensure-proper-nonces.ts","../src/utils/evm-tx-batch-updater.ts","../src/handlers/eth-send-transaction-batch/utils/process-transaction-batch-simulation.ts"],"names":["RpcMethod","parseManifest","rpcErrors","getTokens","chainId","proxyApiUrl","response","JsonRpcBatchInternal","VsCurrencyType","getBasicCoingeckoHttp","simplePrice","simpleTokenPrice","retry","operation","isSuccess","maxRetries","backoffPolicy","RetryBackoffPolicy","backoffPeriodMillis","retries","lastError","delay","result","err","errorMessage","retryIndex","secondsToDelay","_","msToDelay","linearCount","linearStepMs","n","base","incSum","ms","r","coingeckoRetry","charsum","s","i","sum","arrayHash","array","cs","RawSimplePriceResponseSchema","fetchAndVerify","fetchOptions","schema","responseJson","CoingeckoProxyClient","params","queryParams","id","rawQueryParams","z","WatchlistTokenResponseSchema","WatchlistProxyClient","coingeckoBasicClient","_storage","_proxyApiUrl","TokenService","storage","__privateAdd","__publicField","data","currencies","formattedData","tokenData","currency","__privateSet","tokenDetails","tokenInfo","__privateGet","token","coinIds","cacheId","useCoingeckoProxy","tokenAddresses","assetPlatformId","rawData","marketCap","vol24","change24","lastUpdated","shouldThrow","number","object","record","string","CURRENCY_EXCHANGE_RATES_URL","CURRENCY_EXCHANGE_RATES_FALLBACK_URL","ExchangeRateSchema","getExchangeRates","DetailItemType","textItem","label","value","alignment","linkItem","addressItem","networkItem","AppName","getCoreHeaders","name","version","GLACIER_API_KEY","getGlacierApiKey","FetchHttpRequest","GLOBAL_QUERY_PARAMS","GlacierFetchHttpRequest","config","options","mergedQuery","modifiedOptions","EVMNetwork","getProvider","chainName","rpcUrl","multiContractAddress","pollingInterval","provider","addGlacierAPIKeyIfNeeded","knownHosts","url","urlObj","glacierApiKey","search_params","TokenUnit","ChainId","DEFAULT_PRESETS","BASE_PRIORITY_FEE_WEI","isCChain","getNetworkFee","caipId","suggestedFees","suggestPriceOptions","lastBlock","baseFeePerGas","gasMultiplier","getGasMultiplier","multiplier","baseFee","maxPriorityFeePerGas","maxFee","lowMaxTip","mediumMaxTip","highMaxTip","respond","multipliers","TokenType","TransactionType","getExplorerAddressByNetwork","explorerUrl","hash","hashType","convertTransactionNormal","tx","networkToken","address","isSender","timestamp","decimals","amountDisplayValue","txType","from","to","gasPrice","gasUsed","explorerLink","isContractCall","convertTransactionERC20","tokenDecimal","tokenName","tokenSymbol","contractAddress","getErc20Txs","getNormalTxs","getTransactionFromEtherscan","isTestnet","nextPageToken","offset","parsedPageToken","page","queries","normalHist","erc20Hist","erc20TxHashes","filteredNormalTxs","next","isEthereumChainId","startCase","getTxType","nativeTransaction","erc20Transfers","erc721Transfers","erc1155Transfers","userAddress","tokens","nativeOnly","method","parseRawMethod","isSwap","isNativeSend","isNativeReceive","isNFTPurchase","isApprove","isTransfer","isAirdrop","isUnwrap","isFillOrder","isNFTSend","isNFT","isNFTReceive","tokenType","getSenderInfo","isOutgoing","isIncoming","resolve","promise","res","ipfsResolver","CLOUDFLARE_IPFS_URL","ipfsResolverWithFallback","sourceUrl","desiredGatewayPrefix","fetchWithTimeout","uri","timeout","controller","getNftMetadata","tokenUri","json","COVALENT_IMG_SIZER","getSmallImageForNFT","imgUrl","imageSize","decimal","erc20Transfer","erc721Transfer","imageUri","getImageUri","erc1155Transfer","metadata","error","NonContractCallTypes","ERC20TransactionType","convertTransaction","transactions","blockTimestamp","getTransactionsFromGlacier","glacierService","tranasaction","transaction","getTransactionHistory","manifest_default","getNonce","Interface","ERC20","parseERC20TransactionType","description","functionName","isERC20TransactionType","ethers","encodeApprovalLimit","tokenAddress","spenderAddress","limit","requests","getTxUpdater","requestId","signingData","displayData","maxFeeRate","maxTipRate","approvalLimit","gasLimit","request","newSigningData","newDisplayData","tokenApprovals","ensureTokenApprovalCanBeEdited","ensureApproveWasCalled","approval","updatedRequest","txData","estimateGasLimit","accessList","nonce","buildTxApprovalRequest","network","isSimulationSuccessful","balanceChange","alert","dappInfo","transactionType","title","transactionDetails","AlertType","balanceToDisplayValue","numberToBN","isHexString","MaxUint256","scanTransactionBatch","domain","withGasEstimation","blockaid","scanTransaction","scanJsonRpc","accountAddress","xss","parseWithErc20Abi","contract","contractCalls","symbol","iface","calledFunction","decodeFunctionData","isNetworkToken","isERC20Token","hasToField","supportsBatchApprovals","isBalanceChange","input","isTokenApprovals","isEmpty","transactionAlerts","simulateTransaction","rpcMethod","dAppUrl","simulationResult","processTransactionSimulation","estimatedGasLimit","validation","simulation","gasEstimation","processTokenApprovals","processBalanceChange","erc20ParseResult","isExposureTrace","trace","normalizeAddresses","spender","assetAddress","mapExposureTracesToSpenderAsset","traces","accumulator","accountSummary","mappedExposureTraces","approvals","convertAssetToNetworkContractToken","tokenApproval","raw_value","usd_price","amount","assetDiffs","ins","processAssetDiffs","outs","type","assetDiff","a","b","asset","convertNativeAssetToToken","items","diff","displayValue","valueBN","x","processJsonRpcSimulation","transactionSchema","paramsSchema","parseRequestParams","waitForTransactionReceipt","txHash","onTransactionPending","onTransactionConfirmed","onTransactionReverted","success","getTxHash","ethSendTransaction","approvalController","scan","updateTx","cleanup","findAsync","asyncCallback","promises","item","index","found","addIdToPromise","reason","settleAllIdPromises","results","acc","_network","_customTokens","RpcService","customTokens","coingeckoTokenId","tokenService","simplePriceResponse","priceInCurrency","balance","balanceUnit","balanceInCurrency","coingeckoPlatformId","erc20TokenList","getTokenWithBalance","tokenBalancePromises","tokenBalancesResults","tokenIds","erc20TokenBalances","tokenId","tokenBalance","getBalances","addresses","tokenTypes","balanceServices","services","supportingService","balanceService","balances","nativeTokenPromises","erc20TokenPromises","nftTokenPromises","nativeBalance","avaxMarketData","nativeTokenBalances","nftTokenBalances","balanceOrError","balancesOrError","Environment","prodEnv","devEnv","getEnv","environment","Erc1155Token","Erc721Token","Glacier","BLOCKAID_API_KEY","IPFS_URL","GlacierUnhealthyError","CHAINS_TO_FILTER","EvmGlacierService","glacierApiUrl","headers","supportedChains","chain","coingeckoId","nativeTokenBalance","entries","tokenlist","tokensWithBalance","convertErc20TokenWithBalanceToTokenWithBalance","convertErc20TokenToTokenWithBalance","pageToken","pageSize","tokenBalances","balanceDisplayValue","balanceCurrencyDisplayValue","toUtf8String","beautifyComplexMessage","key","toUnicodeBold","subKey","beautifySimpleMessage","str","boldMap","char","messageSchema","addressSchema","ethSignSchema","messageTypeSchema","typedDataSchema","typedDataV1Schema","combinedTypedDataSchema","dataSchema","combinedDataSchema","ethSignTypedDataSchema","ethSignTypedDataV1Schema","ethSignTypedDataV3Schema","ethSignTypedDataV4Schema","personalSignSchema","ctx","parsed","isTypedDataV1","isTypedData","TypedDataEncoder","isTypedDataValid","EIP712Domain","types","ethSign","typedDataValidationResult","messageDetails","primaryType","messageToDisplay","forwardToRpcNode","message","NetworkVMType","WalletType","getAddressFromXPub","getEvmAddressFromPubKey","getAddress","accountIndex","xpub","walletType","pubKeyBuffer","computeAddress","hasDerivationDetails","buildDerivationPath","derivationPathType","deriveAddress","secretId","derivationPath","publicKeyHex","NftTokenMetadataStatus","_supportedChainIds","DeBank","baseUrl","tokenBalanceResponse","chainListResponse","DeBankNftTokenListSchema","BaseDeBankTokenSchema","DeBankNftTokenSchema","_deBank","_getChainInfo","getChainInfo_fn","_mapNftList","mapNftList_fn","DeBankService","chainInfo","chainIdString","__privateMethod","tokenUnit","usdToCurrencyRate","exchangeRates","nftList","deBankNftList","transactionArraySchema","areAllEqualByProp","areAllEmptyByProp","areAllDifferentByProp","batchOptionsSchema","transactionBatchSchema","elements","prop","el","ensureProperNonces","getTxBatchUpdater","signingRequests","txIndex","signingRequest","simulateTransactionBatch","populateMissingGas","simulationResults","scans","matchingTx","processedResult","getHighestAlert","aggregateBalanceChange","aggregateTokenApprovals","alertPrio","highest","aggregatedTokenApprovals","appr","aggregatedApproval","aggrApproval","clearSpentApprovals","aggregated","aggregatedBalanceChange","aggregatedDiff","findTokenDiff","diffs","lookupDiff","latestApprovalState","ethSendTransactionBatch","parsedParams","transactionRequests","onlyWaitForLastTx","allTransactionPayloadsHaveGas","txHashes","isLast","receiptPromise","Blockaid","_glacierService","_deBankService","_approvalController","_blockaid","EvmModule","appInfo","runtime","utilityAddresses","shouldForwardToRpcNode"],"mappings":"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","sourcesContent":["import {\n type Module,\n type Manifest,\n type NetworkFees,\n type GetTransactionHistory,\n type RpcRequest,\n type Network,\n type ApprovalController,\n type GetBalancesParams,\n type GetBalancesResponse,\n type GetAddressParams,\n type GetAddressResponse,\n RpcMethod,\n parseManifest,\n type ConstructorParams,\n type NetworkFeeParam,\n type DeriveAddressParams,\n type DeriveAddressResponse,\n type BuildDerivationPathParams,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getTokens } from './handlers/get-tokens/get-tokens';\nimport { getNetworkFee } from './handlers/get-network-fee/get-network-fee';\nimport { getTransactionHistory } from './handlers/get-transaction-history/get-transaction-history';\nimport ManifestJson from '../manifest.json';\nimport { ethSendTransaction } from './handlers/eth-send-transaction/eth-send-transaction';\nimport { getBalances } from './handlers/get-balances/get-balances';\nimport { getEnv } from './env';\nimport { EvmGlacierService } from './services/glacier-service/glacier-service';\nimport { ethSign } from './handlers/eth-sign/eth-sign';\nimport { forwardToRpcNode } from './handlers/forward-to-rpc-node/forward-to-rpc-node';\nimport { getAddress } from './handlers/get-address/get-address';\nimport { deriveAddress } from './handlers/derive-address/derive-address';\nimport { DeBankService } from './services/debank-service/debank-service';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getProvider } from './utils/get-provider';\nimport { getCoreHeaders, TokenService } from '@internal/utils';\nimport { ethSendTransactionBatch } from './handlers/eth-send-transaction-batch/eth-send-transaction-batch';\nimport { supportsBatchApprovals } from './utils/type-utils';\nimport { buildDerivationPath } from './handlers/build-derivation-path/build-derivation-path';\nimport Blockaid from '@blockaid/client';\nimport { BLOCKAID_API_KEY } from './constants';\n\nexport class EvmModule implements Module {\n #glacierService: EvmGlacierService;\n #deBankService: DeBankService;\n #proxyApiUrl: string;\n #approvalController: ApprovalController;\n #blockaid: Blockaid;\n\n constructor({ approvalController, environment, appInfo, runtime }: ConstructorParams) {\n const { glacierApiUrl, proxyApiUrl } = getEnv(environment);\n this.#glacierService = new EvmGlacierService({\n glacierApiUrl,\n headers: getCoreHeaders(appInfo),\n });\n this.#deBankService = new DeBankService({ proxyApiUrl });\n this.#proxyApiUrl = proxyApiUrl;\n this.#approvalController = approvalController;\n this.#blockaid = new Blockaid({\n baseURL: proxyApiUrl + '/proxy/blockaid/',\n apiKey: BLOCKAID_API_KEY,\n httpAgent: runtime?.httpAgent,\n fetch: runtime?.fetch,\n });\n }\n\n getProvider(network: Network): Promise<JsonRpcBatchInternal> {\n return getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n }\n\n getAddress({ accountIndex, xpub, walletType }: GetAddressParams): Promise<GetAddressResponse> {\n return getAddress({ accountIndex, xpub, walletType });\n }\n\n buildDerivationPath(params: BuildDerivationPathParams) {\n return buildDerivationPath(params);\n }\n\n deriveAddress(params: DeriveAddressParams): Promise<DeriveAddressResponse> {\n return deriveAddress({\n ...params,\n approvalController: this.#approvalController,\n });\n }\n\n getBalances({\n addresses,\n network,\n currency,\n customTokens,\n storage,\n tokenTypes,\n }: GetBalancesParams): Promise<GetBalancesResponse> {\n const tokenService = new TokenService({ storage, proxyApiUrl: this.#proxyApiUrl });\n return getBalances({\n addresses,\n currency,\n network,\n proxyApiUrl: this.#proxyApiUrl,\n customTokens,\n balanceServices: [this.#glacierService, this.#deBankService],\n storage,\n tokenTypes,\n tokenService,\n });\n }\n\n getManifest(): Manifest | undefined {\n const result = parseManifest(ManifestJson);\n return result.success ? result.data : undefined;\n }\n\n getNetworkFee(network: NetworkFeeParam): Promise<NetworkFees> {\n const { chainId, chainName, rpcUrl, utilityAddresses, caipId } = network;\n return getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress: utilityAddresses?.multicall,\n caipId,\n proxyApiUrl: this.#proxyApiUrl,\n });\n }\n\n getTransactionHistory(params: GetTransactionHistory) {\n const { network, address, nextPageToken, offset } = params;\n const { chainId, isTestnet, networkToken, explorerUrl = '' } = network;\n\n return getTransactionHistory({\n chainId,\n isTestnet,\n networkToken,\n explorerUrl,\n address,\n nextPageToken,\n offset,\n glacierService: this.#glacierService,\n });\n }\n\n getTokens(network: Network) {\n const { chainId } = network;\n return getTokens({ chainId, proxyApiUrl: this.#proxyApiUrl });\n }\n\n async onRpcRequest(request: RpcRequest, network: Network) {\n switch (request.method) {\n case RpcMethod.ETH_SEND_TRANSACTION:\n return ethSendTransaction({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n case RpcMethod.ETH_SEND_TRANSACTION_BATCH: {\n if (!supportsBatchApprovals(this.#approvalController)) {\n return {\n error: rpcErrors.methodNotSupported(`Method ${request.method} requires BatchApprovalController`),\n };\n }\n\n return ethSendTransactionBatch({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n }\n case RpcMethod.PERSONAL_SIGN:\n case RpcMethod.ETH_SIGN:\n case RpcMethod.SIGN_TYPED_DATA:\n case RpcMethod.SIGN_TYPED_DATA_V1:\n case RpcMethod.SIGN_TYPED_DATA_V3:\n case RpcMethod.SIGN_TYPED_DATA_V4:\n return ethSign({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n default:\n if (shouldForwardToRpcNode(request.method)) {\n return forwardToRpcNode(request, network);\n }\n\n return { error: rpcErrors.methodNotSupported(`Method ${request.method} not supported`) };\n }\n }\n}\n\nconst shouldForwardToRpcNode = (method: RpcMethod) => {\n return (\n method.startsWith('eth_') ||\n ['web3_clientVersion', 'web3_sha3', 'net_version', 'net_peerCount', 'net_listening'].includes(method)\n );\n};\n","import type { NetworkContractToken } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nexport async function getTokens({\n chainId,\n proxyApiUrl,\n}: {\n chainId: number;\n proxyApiUrl: string;\n}): Promise<NetworkContractToken[]> {\n const response = await fetch(`${proxyApiUrl}/tokens?evmChainId=${chainId}`);\n\n if (!response.ok) {\n throw rpcErrors.internal(`Failed to fetch tokens for chainId ${chainId}`);\n }\n\n return response.json();\n}\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getGlacierApiKey } from '@internal/utils';\nimport { Network as EVMNetwork } from 'ethers';\n\ntype ProviderParams = {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n multiContractAddress?: string;\n pollingInterval?: number;\n};\n\nexport const getProvider = async (params: ProviderParams): Promise<JsonRpcBatchInternal> => {\n const { chainId, chainName, rpcUrl, multiContractAddress, pollingInterval = 2000 } = params;\n\n const provider = new JsonRpcBatchInternal(\n { maxCalls: 40, multiContractAddress },\n addGlacierAPIKeyIfNeeded(rpcUrl),\n new EVMNetwork(chainName, chainId),\n );\n\n provider.pollingInterval = pollingInterval;\n\n return provider;\n};\n\n// RPC urls returned in the token list are always using the production URL\nconst knownHosts = ['glacier-api.avax.network', 'proxy-api.avax.network'];\n\n/**\n * Glacier needs an API key for development, this adds the key if needed.\n */\nexport function addGlacierAPIKeyIfNeeded(url: string): string {\n const urlObj = new URL(url);\n const glacierApiKey = getGlacierApiKey();\n\n if (glacierApiKey && knownHosts.includes(urlObj.hostname)) {\n const search_params = urlObj.searchParams; // copy, does not update the URL\n search_params.set('token', glacierApiKey);\n urlObj.search = search_params.toString();\n return urlObj.toString();\n }\n return url;\n}\n","import {\n VsCurrencyType,\n getBasicCoingeckoHttp,\n simplePrice,\n simpleTokenPrice,\n type SimplePriceParams,\n} from '@avalabs/core-coingecko-sdk';\nimport type { Storage, RawSimplePriceResponse, SimplePriceResponse } from '@avalabs/vm-module-types';\nimport { coingeckoRetry } from '../../utils/coingecko-retry';\nimport { arrayHash } from '../../utils/array-hash';\nimport { CoingeckoProxyClient } from './coingecko-proxy-client';\nimport { WatchlistProxyClient } from './watchlist-proxy-client';\n\nconst coingeckoBasicClient = getBasicCoingeckoHttp();\n\nexport class TokenService {\n #storage?: Storage;\n #proxyApiUrl: string;\n\n constructor({ storage, proxyApiUrl }: { proxyApiUrl: string; storage?: Storage }) {\n this.#storage = storage;\n this.#proxyApiUrl = proxyApiUrl;\n }\n\n async getWatchlistDataForToken({\n tokenDetails,\n currency = VsCurrencyType.USD,\n }: {\n tokenDetails: {\n symbol: string;\n isNative: boolean;\n caip2Id: string;\n address?: string;\n };\n currency: VsCurrencyType;\n }): Promise<{\n priceInCurrency: number;\n change24: number;\n marketCap: number;\n vol24: number;\n }> {\n const data = (\n await new WatchlistProxyClient(this.#proxyApiUrl).watchlistToken({\n tokens: tokenDetails.symbol,\n currency: currency,\n })\n ).filter((token) => {\n return tokenDetails.isNative\n ? token.internalId === `NATIVE-${tokenDetails.symbol.toLowerCase()}`\n : token.platforms[tokenDetails.caip2Id] === tokenDetails.address;\n });\n\n const tokenInfo = data[0];\n\n if (!tokenInfo) {\n return {\n priceInCurrency: 0,\n change24: 0,\n marketCap: 0,\n vol24: 0,\n };\n }\n\n return {\n priceInCurrency: tokenInfo.current_price ?? 0,\n change24: tokenInfo.price_change_percentage_24h ?? 0,\n marketCap: tokenInfo.market_cap ?? 0,\n vol24: tokenInfo.total_volume ?? 0,\n };\n }\n\n /**\n * Get token price with market data first on coingecko (free tier) directly,\n * if we get 429 error, retry it on coingecko proxy (paid service)\n * @returns token price with market data\n */\n async getSimplePrice({\n coinIds = [],\n currencies = [VsCurrencyType.USD],\n }: SimplePriceParams): Promise<SimplePriceResponse | undefined> {\n let data: SimplePriceResponse | undefined;\n\n const key = coinIds ? `${arrayHash(coinIds)}-${currencies.toString()}` : `${currencies.toString()}`;\n\n const cacheId = `getSimplePrice-${key}`;\n\n data = this.#storage?.get?.<SimplePriceResponse>(cacheId);\n\n if (data) return data;\n\n try {\n data = await coingeckoRetry((useCoingeckoProxy) =>\n this.simplePrice({\n coinIds,\n currencies,\n marketCap: true,\n vol24: true,\n change24: true,\n useCoingeckoProxy,\n }),\n );\n } catch {\n data = undefined;\n }\n this.#storage?.set?.(cacheId, data);\n return data;\n }\n\n /**\n * Get token price with market data for a list of addresses\n * @param tokenAddresses the token addresses\n * @param assetPlatformId The platform id for all the tokens in the list\n * @param currency the currency to be used\n * @returns a list of token price with market data\n */\n async getPricesByAddresses(\n tokenAddresses: string[],\n assetPlatformId: string,\n currency: VsCurrencyType = VsCurrencyType.USD,\n ): Promise<SimplePriceResponse | undefined> {\n let data: SimplePriceResponse | undefined;\n\n const key = `${arrayHash(tokenAddresses)}-${assetPlatformId}-${currency}`;\n\n const cacheId = `getPricesWithMarketDataByAddresses-${key}`;\n data = this.#storage?.get?.<SimplePriceResponse>(cacheId);\n\n if (data) return data;\n\n try {\n data = await coingeckoRetry((useCoingeckoProxy) =>\n this.fetchPricesByAddresses({\n assetPlatformId,\n tokenAddresses,\n currency,\n useCoingeckoProxy,\n }),\n );\n } catch (err) {\n console.error(err);\n data = undefined;\n }\n this.#storage?.set?.(cacheId, data);\n return data;\n }\n\n private async fetchPricesByAddresses({\n assetPlatformId,\n tokenAddresses,\n currency = VsCurrencyType.USD,\n useCoingeckoProxy = false,\n }: {\n assetPlatformId: string;\n tokenAddresses: string[];\n currency: VsCurrencyType;\n useCoingeckoProxy?: boolean;\n }): Promise<SimplePriceResponse> {\n if (useCoingeckoProxy) {\n const rawData = await new CoingeckoProxyClient(this.#proxyApiUrl).simplePriceByContractAddresses({\n id: assetPlatformId,\n contract_addresses: tokenAddresses,\n vs_currencies: [currency],\n include_market_cap: true,\n include_24hr_vol: true,\n include_24hr_change: true,\n });\n return this.transformSimplePriceResponse(rawData, [currency]);\n }\n\n return simpleTokenPrice(coingeckoBasicClient, {\n assetPlatformId,\n tokenAddresses,\n currencies: [currency],\n marketCap: true,\n vol24: true,\n change24: true,\n });\n }\n\n private async simplePrice({\n coinIds = [],\n currencies = [VsCurrencyType.USD],\n marketCap = false,\n vol24 = false,\n change24 = false,\n lastUpdated = false,\n useCoingeckoProxy = false,\n shouldThrow = true,\n }: SimplePriceParams & { useCoingeckoProxy?: boolean }): Promise<SimplePriceResponse> {\n if (useCoingeckoProxy) {\n const rawData = await new CoingeckoProxyClient(this.#proxyApiUrl).simplePrice({\n ids: coinIds,\n vs_currencies: currencies,\n include_market_cap: marketCap,\n include_24hr_vol: vol24,\n include_24hr_change: change24,\n include_last_updated_at: lastUpdated,\n });\n return this.transformSimplePriceResponse(rawData, currencies);\n }\n return simplePrice(coingeckoBasicClient, {\n coinIds,\n currencies,\n marketCap,\n vol24,\n change24,\n lastUpdated,\n shouldThrow,\n });\n }\n\n private transformSimplePriceResponse = (\n data: RawSimplePriceResponse,\n currencies = [VsCurrencyType.USD],\n ): SimplePriceResponse => {\n const formattedData: SimplePriceResponse = {};\n Object.keys(data).forEach((id) => {\n const tokenData = data[id];\n formattedData[id] = {};\n currencies.forEach((currency: VsCurrencyType) => {\n formattedData[id] = {\n [currency]: {\n price: tokenData?.[currency],\n change24: tokenData?.[`${currency}_24h_change`],\n vol24: tokenData?.[`${currency}_24h_vol`],\n marketCap: tokenData?.[`${currency}_market_cap`],\n },\n };\n });\n });\n return formattedData;\n };\n}\n","const DEFAULT_MAX_RETRIES = 10;\n\ntype RetryParams<T> = {\n operation: (retryIndex: number) => Promise<T>;\n isSuccess: (result: T) => boolean;\n maxRetries?: number;\n backoffPolicy?: RetryBackoffPolicyInterface;\n};\n/*\n * Retries an operation with defined backoff policy.\n *\n * @param operation - The operation to retry.\n * @param isSuccess - The predicate to check if the operation succeeded.\n * @param maxRetries - The maximum number of retries.\n * @param backoffPolicy - Function to generate delay time based on current retry count.\n *\n * @returns The result of the operation.\n * @throws An error if the operation fails after the maximum number of retries.\n *\n * @example\n * const result = await retry(\n * async () => {\n * const response = await fetch('https://example.com')\n * return response.json()\n * },\n * result => result.status === 200\n * )\n */\nexport const retry = async <T>({\n operation,\n isSuccess,\n maxRetries = DEFAULT_MAX_RETRIES,\n backoffPolicy = RetryBackoffPolicy.exponential(),\n}: RetryParams<T>): Promise<T> => {\n let backoffPeriodMillis = 0;\n let retries = 0;\n let lastError: unknown;\n\n while (retries < maxRetries) {\n if (retries > 0) {\n await delay(backoffPeriodMillis);\n }\n\n try {\n const result = await operation(retries);\n\n if (isSuccess(result)) {\n return result;\n }\n } catch (err) {\n // when the operation throws an error, we still retry\n lastError = err;\n }\n\n backoffPeriodMillis = backoffPolicy(retries);\n retries++;\n }\n\n const errorMessage = lastError ? `Max retry exceeded. ${lastError}` : 'Max retry exceeded.';\n\n throw new Error(errorMessage);\n};\n\ntype RetryBackoffPolicyInterface = (retryIndex: number) => number;\n\nexport class RetryBackoffPolicy {\n static exponential(): RetryBackoffPolicyInterface {\n return (retryIndex: number): number => {\n return Math.pow(2, retryIndex) * 1000;\n };\n }\n\n static constant(secondsToDelay: number): RetryBackoffPolicyInterface {\n return (_: number): number => {\n return secondsToDelay * 1000;\n };\n }\n\n static constantMs(msToDelay: number): RetryBackoffPolicyInterface {\n return (_: number): number => {\n return msToDelay;\n };\n }\n\n /**\n * linearThenExponential backoff:\n * - First `linearCount` retries: linear increase by `linearStepMs`\n * - After that: increment grows exponentially based on `linearStepMs`\n * Example (linearCount=4, linearStepMs=1000):\n * 1s, 2s, 3s, 4s, 6s, 10s, 18s, 34s...\n */\n static linearThenExponential(linearCount: number, linearStepMs: number): RetryBackoffPolicyInterface {\n return (retryIndex: number): number => {\n if (retryIndex < linearCount) {\n // Linear phase: (i+1) * step\n return (retryIndex + 1) * linearStepMs;\n }\n // Exponential-increment phase (closed form):\n // n = number of exponential increments applied\n // base = linearCount * step\n // increment sum = 2*step * (2^n - 1)\n const n = retryIndex - linearCount + 1;\n const base = linearCount * linearStepMs;\n const incSum = 2 * linearStepMs * (Math.pow(2, n) - 1);\n return base + incSum;\n };\n }\n}\n\nfunction delay(ms: number) {\n return new Promise((r) => setTimeout(r, ms));\n}\n","import { RetryBackoffPolicy, retry } from './retry';\n\ntype Error = {\n status: {\n error_code: number;\n error_message: string;\n };\n};\n\nexport const coingeckoRetry = <T>(\n operation: (useCoingeckoProxy: boolean) => Promise<T | Error>,\n): Promise<T | undefined> => {\n return retry({\n operation: (retryIndex: number) => operation(retryIndex > 0),\n maxRetries: 2,\n backoffPolicy: RetryBackoffPolicy.constant(1),\n isSuccess: (response: T | Error) => {\n const errorStatus = (response as Error)?.status;\n return errorStatus?.error_code !== 429;\n },\n }) as Promise<T | undefined>;\n};\n","export function charsum(s: string): number {\n let i,\n sum = 0;\n for (i = 0; i < s.length; i++) {\n sum += s.charCodeAt(i) * (i + 1);\n }\n return sum;\n}\n","import { charsum } from './charsum';\n\n// from https://stackoverflow.com/a/25105589\nexport function arrayHash(array: string[]): string {\n let i,\n sum = 0;\n for (i = 0; i < array.length; i++) {\n const cs = charsum(array[i] ?? '');\n sum = sum + 65027 / cs;\n }\n return ('' + sum).slice(0, 16);\n}\n","import { RawSimplePriceResponseSchema } from '@avalabs/vm-module-types';\nimport { fetchAndVerify } from '../../utils/fetch-and-verify';\n\nexport class CoingeckoProxyClient {\n constructor(private proxyApiUrl: string) {}\n\n simplePrice(params: {\n ids: string[];\n vs_currencies: string[];\n include_market_cap?: boolean;\n include_24hr_vol?: boolean;\n include_24hr_change?: boolean;\n include_last_updated_at?: boolean;\n }) {\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(params as any);\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/proxy/coingecko/simple/price?${queryParams}`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n RawSimplePriceResponseSchema,\n );\n }\n\n simplePriceByContractAddresses(params: {\n id: string;\n contract_addresses: string[];\n vs_currencies?: string[];\n include_market_cap?: boolean;\n include_24hr_vol?: boolean;\n include_24hr_change?: boolean;\n }) {\n const { id, ...rawQueryParams } = params;\n\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(rawQueryParams as any);\n\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/proxy/coingecko/simple/token_price/${id}?${queryParams}`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n RawSimplePriceResponseSchema,\n );\n }\n}\n","import z from 'zod';\nimport type { ZodSchema } from 'zod';\n\nexport async function fetchAndVerify<T extends ZodSchema>(\n fetchOptions: Parameters<typeof fetch>,\n schema: T,\n): Promise<z.infer<T>> {\n const response = await fetch(...fetchOptions);\n\n if (!response.ok) {\n throw new Error(`Request failed with status ${response.status}`);\n }\n\n const responseJson = await response.json();\n return schema.parse(responseJson);\n}\n","import { fetchAndVerify } from '../../utils/fetch-and-verify';\nimport { z } from 'zod';\n\nconst WatchlistTokenResponseSchema = z.array(\n z.object({\n // the object has more properties than the ones listed here, but we only need these at the moment\n internalId: z.string(),\n id: z.string(),\n symbol: z.string(),\n name: z.string(),\n image: z.string().optional().nullable(),\n current_price: z.number().optional().nullable(),\n price_change_percentage_24h: z.number().optional().nullable(),\n market_cap: z.number().optional().nullable(),\n total_volume: z.number().optional().nullable(),\n platforms: z.record(z.string(), z.string()),\n }),\n);\n\nexport class WatchlistProxyClient {\n constructor(private proxyApiUrl: string) {}\n\n watchlistToken(params: { tokens: string; currency: string }) {\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(params as any);\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/watchlist/tokens?${queryParams}`,\n {\n method: 'GET',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n WatchlistTokenResponseSchema,\n );\n }\n}\n","import z, { number, object, record, string } from 'zod';\nimport { fetchAndVerify } from '../../utils/fetch-and-verify';\n\nconst CURRENCY_EXCHANGE_RATES_URL =\n 'https://cdn.jsdelivr.net/npm/@fawazahmed0/currency-api@latest/v1/currencies/usd.min.json';\n\nconst CURRENCY_EXCHANGE_RATES_FALLBACK_URL = 'https://latest.currency-api.pages.dev/v1/currencies/usd.min.json';\n\nconst ExchangeRateSchema = object({\n date: string(),\n usd: record(number()),\n});\n\ntype ExchangeRate = z.infer<typeof ExchangeRateSchema>;\n\nexport const getExchangeRates = async (): Promise<ExchangeRate> => {\n try {\n return await fetchAndVerify([CURRENCY_EXCHANGE_RATES_URL], ExchangeRateSchema);\n } catch {\n return await fetchAndVerify([CURRENCY_EXCHANGE_RATES_FALLBACK_URL], ExchangeRateSchema);\n }\n};\n","import {\n type AddressItem,\n type CurrencyItem,\n type NodeIDItem,\n type TextItem,\n type DataItem,\n type DateItem,\n type LinkItemValue,\n DetailItemType,\n type LinkItem,\n type FundsRecipientItem,\n type AddressListItem,\n type NetworkItemValue,\n type NetworkItem,\n} from '@avalabs/vm-module-types';\n\nexport const fundsRecipientItem = (\n address: string,\n amount: bigint,\n maxDecimals: number,\n symbol: string,\n): FundsRecipientItem => ({\n type: DetailItemType.FUNDS_RECIPIENT,\n label: address,\n amount,\n maxDecimals,\n symbol,\n});\n\nexport const currencyItem = (label: string, value: bigint, maxDecimals: number, symbol: string): CurrencyItem => ({\n label,\n type: DetailItemType.CURRENCY,\n value,\n maxDecimals,\n symbol,\n});\n\nexport const textItem = (\n label: string,\n value: string,\n alignment: 'horizontal' | 'vertical' = 'horizontal',\n): TextItem => ({\n label,\n alignment,\n type: DetailItemType.TEXT,\n value,\n});\n\nexport const linkItem = (label: string, value: LinkItemValue): LinkItem => ({\n label,\n value,\n type: DetailItemType.LINK,\n});\n\nexport const addressItem = (label: string, value: string): AddressItem => ({\n label,\n type: DetailItemType.ADDRESS,\n value,\n});\n\nexport const addressListItem = (label: string, value: string[]): AddressListItem => ({\n label,\n type: DetailItemType.ADDRESS_LIST,\n value,\n});\n\nexport const nodeIDItem = (label: string, value: string): NodeIDItem => ({\n label,\n type: DetailItemType.NODE_ID,\n value,\n});\n\nexport const dataItem = (label: string, value: string): DataItem => ({\n label,\n type: DetailItemType.DATA,\n value,\n});\n\nexport const dateItem = (label: string, value: string): DateItem => ({\n label,\n type: DetailItemType.DATE,\n value,\n});\n\nexport const networkItem = (label: string, value: NetworkItemValue): NetworkItem => ({\n label,\n type: DetailItemType.NETWORK,\n value,\n});\n","import { AppName, type AppInfo } from '@avalabs/vm-module-types';\n\nexport const getCoreHeaders = ({ name, version }: AppInfo): Record<string, string> | undefined => {\n switch (name) {\n case AppName.CORE_MOBILE_IOS:\n case AppName.CORE_MOBILE_ANDROID:\n case AppName.CORE_WEB:\n case AppName.CORE_EXTENSION:\n case AppName.EXPLORER:\n return {\n 'x-application-name': name,\n 'x-application-version': version,\n };\n case AppName.OTHER:\n return undefined;\n }\n};\n","export const GLACIER_API_KEY = process.env.GLACIER_API_KEY;\n","import { GLACIER_API_KEY } from '../consts';\n\n// this key is only needed in development to bypass rate limit\n// it should never be used in production\nexport const getGlacierApiKey = (): string | undefined => {\n return GLACIER_API_KEY;\n};\n","import { FetchHttpRequest, type OpenAPIConfig, type ApiRequestOptions, CancelablePromise } from '@avalabs/glacier-sdk';\nimport { getGlacierApiKey } from './get-glacier-api-key';\n\nconst GLOBAL_QUERY_PARAMS: Record<string, string | undefined> = {\n rltoken: getGlacierApiKey(),\n};\n\n/**\n * Custom HTTP request handler that automatically appends the Glacier API key (if present)\n * to bypass rate limits in development environments.\n */\nexport class GlacierFetchHttpRequest extends FetchHttpRequest {\n constructor(config: OpenAPIConfig) {\n super(config);\n }\n\n public override request<T>(options: ApiRequestOptions): CancelablePromise<T> {\n // Merge global query parameters with request-specific ones\n const mergedQuery = {\n ...GLOBAL_QUERY_PARAMS,\n ...(options.query || {}), // Request-specific params (override globals if same key)\n };\n\n // Create modified options with merged query\n const modifiedOptions: ApiRequestOptions = {\n ...options,\n query: Object.keys(mergedQuery).length > 0 ? mergedQuery : undefined,\n };\n\n // Call the base class's request method\n return super.request<T>(modifiedOptions);\n }\n}\n","import type { NetworkFees, SuggestGasPriceOptionsResponse } from '@avalabs/vm-module-types';\nimport { getProvider } from '../../utils/get-provider';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { ChainId } from '@avalabs/core-chains-sdk';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nconst DEFAULT_PRESETS = {\n LOW: 1n,\n MEDIUM: 4n,\n HIGH: 6n,\n};\n\nconst BASE_PRIORITY_FEE_WEI = 500000000n; //0.5 GWei\n\nconst isCChain = (chainId: number) =>\n chainId === ChainId.AVALANCHE_TESTNET_ID || chainId === ChainId.AVALANCHE_MAINNET_ID;\n\n/**\n * Returns {@link NetworkFees} based on {@link DEFAULT_PRESETS} multipliers.\n * @throws Error if provider does not support eip-1559\n */\nexport async function getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n caipId,\n proxyApiUrl,\n}: {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n proxyApiUrl?: string;\n multiContractAddress?: string;\n caipId?: string;\n}): Promise<NetworkFees> {\n const provider = await getProvider({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n });\n\n if (!proxyApiUrl) {\n throw rpcErrors.internal('Proxy API URL is needed');\n }\n\n if (isCChain(chainId)) {\n try {\n const suggestedFees = await suggestPriceOptions(provider);\n\n return {\n ...suggestedFees,\n baseFee: suggestedFees.medium.maxFeePerGas,\n displayDecimals: 9,\n isFixedFee: false,\n };\n } catch (err) {\n console.error('eth_suggestPriceOptions call failed, falling back to legacy fee fetching');\n }\n }\n\n const lastBlock = await provider.getBlock('latest', false);\n\n if (!lastBlock) {\n throw rpcErrors.internal('There is no block');\n }\n const baseFeePerGas = lastBlock.baseFeePerGas;\n\n if (!baseFeePerGas) {\n throw rpcErrors.internal('Pre-EIP-1559 networks are not supported');\n }\n\n const gasMultiplier = await getGasMultiplier(proxyApiUrl, caipId);\n\n const multiplier = new TokenUnit(gasMultiplier, 0, '');\n\n const baseFee = new TokenUnit(baseFeePerGas, 0, '');\n const maxPriorityFeePerGas = BigInt('1000000000');\n\n const maxFee = baseFee.mul(multiplier).add(maxPriorityFeePerGas).toSubUnit();\n\n const lowMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.LOW;\n const mediumMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.MEDIUM;\n const highMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.HIGH;\n return {\n baseFee: maxFee,\n low: {\n maxFeePerGas: maxFee + lowMaxTip,\n maxPriorityFeePerGas: lowMaxTip,\n },\n medium: {\n maxFeePerGas: maxFee + mediumMaxTip,\n maxPriorityFeePerGas: mediumMaxTip,\n },\n high: {\n maxFeePerGas: maxFee + highMaxTip,\n maxPriorityFeePerGas: highMaxTip,\n },\n isFixedFee: false,\n displayDecimals: 9,\n };\n}\n\nasync function suggestPriceOptions(\n provider: JsonRpcBatchInternal,\n): Promise<Pick<NetworkFees, 'low' | 'medium' | 'high'>> {\n const options: SuggestGasPriceOptionsResponse = await provider.send('eth_suggestPriceOptions', []);\n\n return {\n low: {\n maxFeePerGas: BigInt(options.slow.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.slow.maxPriorityFeePerGas),\n },\n medium: {\n maxFeePerGas: BigInt(options.normal.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.normal.maxPriorityFeePerGas),\n },\n high: {\n maxFeePerGas: BigInt(options.fast.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.fast.maxPriorityFeePerGas),\n },\n };\n}\n\nasync function getGasMultiplier(proxyApiUrl: string, caipId?: string) {\n const defaultMultiplier = 1.5;\n if (!caipId) {\n return defaultMultiplier;\n }\n\n try {\n const respond = await fetch(`${proxyApiUrl}/gas/multiplier`);\n\n if (!respond.ok) {\n throw new Error(respond.statusText);\n }\n const multipliers = await respond.json();\n\n return multipliers[caipId] ?? multipliers.default;\n } catch (e) {\n return defaultMultiplier;\n }\n}\n","import type { NormalTx } from '@avalabs/core-etherscan-sdk';\nimport { TokenType, TransactionType, type NetworkToken, type Transaction } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\n\nexport const convertTransactionNormal = ({\n tx,\n networkToken,\n chainId,\n explorerUrl,\n address,\n}: {\n tx: NormalTx;\n networkToken: NetworkToken;\n chainId: number;\n explorerUrl: string;\n address: string;\n}): Transaction => {\n const isSender = tx.from.toLowerCase() === address.toLowerCase();\n const timestamp = parseInt(tx.timeStamp) * 1000;\n const decimals = networkToken.decimals;\n const amount = new TokenUnit(tx.value, networkToken.decimals, networkToken.symbol);\n const amountDisplayValue = amount.toDisplay();\n const txType = isSender ? TransactionType.SEND : TransactionType.RECEIVE;\n\n const { from, to, gasPrice, gasUsed, hash } = tx;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n\n return {\n isIncoming: !isSender,\n isOutgoing: isSender,\n isContractCall: isContractCall(tx),\n timestamp,\n hash,\n isSender,\n from,\n to,\n tokens: [\n {\n decimal: decimals.toString(),\n name: networkToken.name,\n symbol: networkToken.symbol,\n amount: amountDisplayValue,\n type: TokenType.NATIVE,\n },\n ],\n gasUsed,\n gasPrice,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n};\n\nfunction isContractCall(tx: NormalTx): boolean {\n return tx.input !== '0x';\n}\n","export function getExplorerAddressByNetwork(\n explorerUrl: string,\n hash: string,\n hashType: 'address' | 'tx' = 'tx',\n): string {\n return `${explorerUrl}/${hashType}/${hash}`;\n}\n","import type { Erc20Tx } from '@avalabs/core-etherscan-sdk';\nimport { TokenType, TransactionType, type Transaction } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\n\nexport function convertTransactionERC20({\n tx,\n address,\n explorerUrl,\n chainId,\n}: {\n tx: Erc20Tx;\n address: string;\n chainId: number;\n explorerUrl: string;\n}): Transaction {\n const isSender = tx.from.toLowerCase() === address.toLowerCase();\n const timestamp = parseInt(tx.timeStamp) * 1000;\n const amount = new TokenUnit(tx.value, Number(tx.tokenDecimal), tx.tokenSymbol);\n const amountDisplayValue = amount.toDisplay();\n const { from, to, gasPrice, gasUsed, hash, tokenDecimal, tokenName, tokenSymbol, contractAddress } = tx;\n const txType = isSender ? TransactionType.SEND : TransactionType.RECEIVE;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n\n return {\n isIncoming: !isSender,\n isOutgoing: isSender,\n isContractCall: false,\n timestamp,\n hash,\n isSender,\n from,\n to,\n tokens: [\n {\n decimal: tokenDecimal,\n name: tokenName,\n symbol: tokenSymbol,\n type: TokenType.ERC20,\n amount: amountDisplayValue,\n address: contractAddress,\n },\n ],\n gasUsed,\n gasPrice,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n}\n","import { convertTransactionNormal } from './convert-transaction-normal';\nimport { convertTransactionERC20 } from './convert-transaction-erc20';\nimport type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { getErc20Txs, getNormalTxs } from '@avalabs/core-etherscan-sdk';\n\ninterface EtherscanPagination {\n queries: ('normal' | 'erc20')[];\n page?: number;\n}\n\nexport const getTransactionFromEtherscan = async ({\n isTestnet,\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n}: {\n isTestnet?: boolean;\n networkToken: NetworkToken;\n explorerUrl: string;\n chainId: number;\n address: string;\n nextPageToken?: string;\n offset?: number;\n}): Promise<TransactionHistoryResponse> => {\n /*\n Using JSON for nextPageToken because this function is managing both the Normal\n and ERC20 queries. It encodes the current page and the queries that should be\n run. For example, if 'normal' has no more records to fetch then it will be\n excluded from the list and the JSON will be something like:\n { page: 3, queries: ['erc20'] }\n */\n const parsedPageToken = nextPageToken ? (JSON.parse(nextPageToken) as EtherscanPagination) : undefined;\n const page = parsedPageToken?.page || 1;\n const queries = parsedPageToken?.queries || ['normal', 'erc20'];\n\n const normalHist = (queries.includes('normal') ? await getNormalTxs(address, !isTestnet, { page, offset }) : []).map(\n (tx) => convertTransactionNormal({ tx, chainId, networkToken, explorerUrl, address }),\n );\n\n const erc20Hist = (\n queries.includes('erc20')\n ? await getErc20Txs(address, !isTestnet, undefined, {\n page,\n offset,\n })\n : []\n ).map((tx) =>\n convertTransactionERC20({\n tx,\n address,\n explorerUrl,\n chainId,\n }),\n );\n\n // Filter erc20 transactions from normal tx list\n const erc20TxHashes = erc20Hist.map((tx) => tx.hash);\n const filteredNormalTxs = normalHist.filter((tx) => {\n return !erc20TxHashes.includes(tx.hash);\n });\n\n const next: EtherscanPagination = { queries: [], page: page + 1 };\n if (normalHist.length) next.queries.push('normal');\n if (erc20Hist.length) next.queries.push('erc20');\n\n return {\n transactions: [...filteredNormalTxs, ...erc20Hist],\n nextPageToken: next.queries.length ? JSON.stringify(next) : '', // stop pagination\n };\n};\n","enum ChainId {\n ETHEREUM_HOMESTEAD = 1,\n ETHEREUM_TEST_RINKEBY = 4,\n ETHEREUM_TEST_GOERLY = 5,\n ETHEREUM_TEST_SEPOLIA = 11155111,\n}\n\nexport const isEthereumChainId = (chainId: number): boolean => {\n return (\n ChainId.ETHEREUM_HOMESTEAD === chainId ||\n ChainId.ETHEREUM_TEST_GOERLY === chainId ||\n ChainId.ETHEREUM_TEST_RINKEBY === chainId ||\n ChainId.ETHEREUM_TEST_SEPOLIA === chainId\n );\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TokenType, TransactionType, type TxToken } from '@avalabs/vm-module-types';\nimport startCase from 'lodash.startcase';\n\nexport const getTxType = (\n { nativeTransaction, erc20Transfers, erc721Transfers, erc1155Transfers }: TransactionDetails,\n userAddress: string,\n tokens: TxToken[],\n): TransactionType => {\n const nativeOnly = !erc20Transfers && !erc721Transfers && !erc1155Transfers;\n const method = parseRawMethod(nativeTransaction.method?.methodName);\n\n const address = userAddress.toLowerCase();\n\n const isSwap = method.toLowerCase().includes('swap');\n const isNativeSend = nativeOnly && nativeTransaction.from.address.toLowerCase() === address;\n const isNativeReceive = nativeOnly && nativeTransaction.to.address.toLowerCase() === address;\n const isNFTPurchase = method === 'Market Buy Orders With Eth' || method === 'Buy NFT';\n const isApprove = method === 'Approve';\n const isTransfer = method.toLowerCase().includes('transfer');\n const isAirdrop = method.toLowerCase().includes('airdrop');\n const isUnwrap = method.toLowerCase().includes('unwrap');\n const isFillOrder = method === 'Fill Order';\n const isNFTSend = !!tokens[0] && isNFT(tokens[0].type) && tokens[0].from?.address.toLowerCase() === address;\n const isNFTReceive = !!tokens[0] && isNFT(tokens[0].type) && tokens[0].to?.address.toLowerCase() === address;\n\n if (isSwap) return TransactionType.SWAP;\n if (isNativeSend) return TransactionType.SEND;\n if (isNativeReceive) return TransactionType.RECEIVE;\n if (isNFTPurchase) return TransactionType.NFT_BUY;\n if (isApprove) return TransactionType.APPROVE;\n if (isNFTSend) return TransactionType.NFT_SEND;\n if (isNFTReceive) return TransactionType.NFT_RECEIVE;\n if (isTransfer) return TransactionType.TRANSFER;\n if (isAirdrop) return TransactionType.AIRDROP;\n if (isUnwrap) return TransactionType.UNWRAP;\n if (isFillOrder) return TransactionType.FILL_ORDER;\n return TransactionType.UNKNOWN;\n};\n\nfunction isNFT(tokenType: TokenType) {\n return tokenType === TokenType.ERC721 || tokenType === TokenType.ERC1155;\n}\n\nconst parseRawMethod = (method = ''): string => {\n if (method.includes('(')) {\n return startCase(method.split('(', 1)[0]);\n }\n return method;\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TransactionType } from '@avalabs/vm-module-types';\n\nexport const getSenderInfo = (\n txType: TransactionType,\n { nativeTransaction, erc20Transfers, erc721Transfers }: TransactionDetails,\n address: string,\n): { isOutgoing: boolean; isIncoming: boolean; isSender: boolean; from: string; to: string } => {\n const isTransfer = txType === TransactionType.TRANSFER;\n const isNativeSend = txType === TransactionType.SEND;\n const isNativeReceive = txType === TransactionType.RECEIVE;\n let from = nativeTransaction?.from?.address;\n let to = nativeTransaction?.to?.address;\n\n // Until multi tokens transaction is supported in UI, using from and to of the only token is helpful for UI\n if (isTransfer && erc20Transfers && erc20Transfers[0]) {\n from = erc20Transfers[0].from.address;\n to = erc20Transfers[0].to.address;\n }\n\n if (isTransfer && erc721Transfers && erc721Transfers[0]) {\n from = erc721Transfers[0].from.address;\n to = erc721Transfers[0].to.address;\n }\n\n const isOutgoing = isNativeSend || (isTransfer && from.toLowerCase() === address.toLowerCase());\n const isIncoming = isNativeReceive || (isTransfer && to.toLowerCase() === address.toLowerCase());\n\n const isSender = from === address;\n\n return {\n isOutgoing,\n isIncoming,\n isSender,\n from,\n to,\n };\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport type { TxToken, NetworkToken } from '@avalabs/vm-module-types';\nimport { TokenType } from '@avalabs/vm-module-types';\nimport { resolve } from '../../utils/resolve';\nimport { getNftMetadata } from './get-nft-metadata';\nimport { getSmallImageForNFT } from '../../utils/get-small-image-for-nft';\nimport { ipfsResolverWithFallback } from '../../utils/ipfs-resolver-with-fallback';\n\nexport const getTokens = async (\n { nativeTransaction, erc20Transfers, erc721Transfers, erc1155Transfers }: TransactionDetails,\n networkToken: NetworkToken,\n): Promise<TxToken[]> => {\n const result: TxToken[] = [];\n\n if (nativeTransaction.value !== '0') {\n const decimal = networkToken.decimals;\n const amount = new TokenUnit(nativeTransaction.value, networkToken.decimals, networkToken.symbol);\n const amountDisplayValue = amount.toDisplay();\n result.push({\n decimal: decimal.toString(),\n name: networkToken.name,\n symbol: networkToken.symbol,\n amount: amountDisplayValue,\n from: nativeTransaction.from,\n to: nativeTransaction.to,\n type: TokenType.NATIVE,\n });\n }\n\n erc20Transfers?.forEach((erc20Transfer) => {\n const decimals = erc20Transfer.erc20Token.decimals;\n const amount = new TokenUnit(erc20Transfer.value, decimals, erc20Transfer.erc20Token.symbol);\n const amountDisplayValue = amount.toDisplay();\n\n result.push({\n decimal: decimals.toString(),\n address: erc20Transfer.erc20Token.address,\n name: erc20Transfer.erc20Token.name,\n symbol: erc20Transfer.erc20Token.symbol,\n amount: amountDisplayValue,\n from: erc20Transfer.from,\n to: erc20Transfer.to,\n imageUri: erc20Transfer.erc20Token.logoUri,\n type: TokenType.ERC20,\n });\n });\n\n if (erc721Transfers) {\n await Promise.allSettled(\n erc721Transfers.map(async (erc721Transfer) => {\n const token = erc721Transfer.erc721Token;\n const imageUri = await getImageUri(token.tokenUri, token.metadata.imageUri);\n\n result.push({\n name: erc721Transfer.erc721Token.name,\n symbol: erc721Transfer.erc721Token.symbol,\n address: erc721Transfer.erc721Token.address,\n amount: '1',\n imageUri,\n from: erc721Transfer.from,\n to: erc721Transfer.to,\n collectableTokenId: erc721Transfer.erc721Token.tokenId,\n type: TokenType.ERC721,\n });\n }),\n );\n }\n\n if (erc1155Transfers) {\n await Promise.allSettled(\n erc1155Transfers.map(async (erc1155Transfer) => {\n const token = erc1155Transfer.erc1155Token;\n const imageUri = await getImageUri(token.tokenUri, token.metadata.imageUri);\n\n result.push({\n name: erc1155Transfer.erc1155Token.metadata.name ?? '',\n symbol: erc1155Transfer.erc1155Token.metadata.symbol ?? '',\n address: erc1155Transfer.erc1155Token.address,\n amount: erc1155Transfer.value,\n imageUri,\n from: erc1155Transfer.from,\n to: erc1155Transfer.to,\n collectableTokenId: erc1155Transfer.erc1155Token.tokenId,\n type: TokenType.ERC1155,\n });\n }),\n );\n }\n\n return result;\n};\n\nconst getImageUri = async (tokenUri: string, imageUri?: string): Promise<string> => {\n if (imageUri) {\n if (imageUri.startsWith('ipfs://')) {\n return ipfsResolverWithFallback(imageUri);\n } else {\n return imageUri;\n }\n } else {\n const [metadata, error] = await resolve(getNftMetadata(tokenUri));\n if (error) {\n return '';\n } else {\n return metadata.image ? getSmallImageForNFT(metadata.image) : '';\n }\n }\n};\n","export async function resolve<T = unknown>(promise: Promise<T>) {\n try {\n return promise.then((res) => [res, null]).catch((err) => [null, err]);\n } catch (err) {\n return Promise.resolve([null, err]);\n }\n}\n","import { ipfsResolver } from '@avalabs/core-utils-sdk';\n\nexport const CLOUDFLARE_IPFS_URL = 'https://ipfs.io';\n\nexport function ipfsResolverWithFallback(\n sourceUrl: string | undefined,\n desiredGatewayPrefix: string = CLOUDFLARE_IPFS_URL,\n) {\n if (!sourceUrl) {\n return '';\n }\n\n try {\n return ipfsResolver(sourceUrl, desiredGatewayPrefix);\n } catch {\n return sourceUrl;\n }\n}\n","import { ipfsResolverWithFallback } from '../../utils/ipfs-resolver-with-fallback';\n\ninterface NftMetadata {\n attributes?: string;\n name?: string;\n image?: string;\n description?: string;\n}\n\nasync function fetchWithTimeout(uri: string, timeout = 5000) {\n const controller = new AbortController();\n setTimeout(() => controller.abort(), timeout);\n\n return fetch(uri, { signal: controller.signal });\n}\n\nexport async function getNftMetadata(tokenUri: string) {\n let data: NftMetadata = {};\n if (!tokenUri) {\n return {};\n } else if (tokenUri.startsWith('data:application/json;base64,')) {\n const value = tokenUri.substring(29);\n try {\n const json = Buffer.from(value, 'base64').toString();\n data = JSON.parse(json);\n } catch {\n data = {};\n }\n } else {\n data = await fetchWithTimeout(ipfsResolverWithFallback(tokenUri))\n .then((r) => r.json())\n .catch(() => ({}));\n }\n return data;\n}\n","import { ipfsResolverWithFallback } from './ipfs-resolver-with-fallback';\n\nconst COVALENT_IMG_SIZER = 'https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image';\n\n/**\n * Covalent has an on the fly image resizer, it resolves image urls then resizes the image.\n *\n * This allows us to request smaller images depending on the UI needs\n *\n * @param imgUrl the url of the image to convert to size\n * @returns The url to the image which is sized at the time of request\n */\nexport function getSmallImageForNFT(imgUrl: string, imageSize: '256' | '512' | '1024' = '256') {\n const url = ipfsResolverWithFallback(imgUrl);\n return `${COVALENT_IMG_SIZER}/width=${imageSize},fit/${url}`;\n}\n","import type { z } from 'zod';\nimport { TransactionType } from '@avalabs/vm-module-types';\n\nimport type { transactionSchema } from './utils/transaction-schema';\nimport type { transactionArraySchema } from './handlers/eth-send-transaction-batch/schema';\n\nexport const NonContractCallTypes = [TransactionType.SEND, TransactionType.RECEIVE, TransactionType.TRANSFER];\n\nexport type TransactionParams = z.infer<typeof transactionSchema>;\n\nexport type RequiredBy<T, K extends keyof T> = T & Required<Pick<T, K>>;\n\nexport type TransactionBatchParams = z.infer<typeof transactionArraySchema>;\n\nexport enum ERC20TransactionType {\n TOTAL_SUPPLY = 'totalSupply',\n BALANCE_OF = 'balanceOf',\n TRANSFER = 'transfer',\n TRANSFER_FROM = 'transferFrom',\n APPROVE = 'approve',\n ALLOWANCE = 'allowance',\n}\n","import type { Transaction, NetworkToken } from '@avalabs/vm-module-types';\nimport { getTxType } from './get-tx-type';\nimport { getSenderInfo } from './get-sender-info';\nimport { getTokens } from './get-tokens';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\nimport type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { NonContractCallTypes } from '../../../../types';\n\ntype ConvertTransactionParams = {\n transactions: TransactionDetails;\n explorerUrl: string;\n networkToken: NetworkToken;\n chainId: number;\n address: string;\n};\n\nexport const convertTransaction = async ({\n transactions,\n explorerUrl,\n networkToken,\n chainId,\n address,\n}: ConvertTransactionParams): Promise<Transaction> => {\n const tokens = await getTokens(transactions, networkToken);\n const txType = getTxType(transactions, address, tokens);\n const { isOutgoing, isIncoming, isSender, from, to } = getSenderInfo(txType, transactions, address);\n const { blockTimestamp, txHash: hash, gasPrice, gasUsed } = transactions.nativeTransaction;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n const isContractCall = !NonContractCallTypes.includes(txType);\n\n return {\n isContractCall,\n isIncoming,\n isOutgoing,\n isSender,\n timestamp: blockTimestamp * 1000, // s to ms\n hash,\n from,\n to,\n tokens,\n gasPrice,\n gasUsed,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n};\n","import type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { convertTransaction } from './convert-transaction';\nimport type { EvmGlacierService } from '../../../../services/glacier-service/glacier-service';\n\nexport const getTransactionsFromGlacier = async ({\n chainId,\n explorerUrl,\n networkToken,\n address,\n nextPageToken,\n offset,\n glacierService,\n}: {\n chainId: number;\n explorerUrl: string;\n networkToken: NetworkToken;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierService: EvmGlacierService;\n}): Promise<TransactionHistoryResponse> => {\n try {\n const response = await glacierService.listTransactions({\n chainId: chainId.toString(),\n address,\n pageToken: nextPageToken,\n pageSize: offset,\n });\n\n const convertedTxs = await Promise.all(\n response.transactions\n .filter(\n // Currently not showing failed tx\n (tranasaction) => tranasaction.nativeTransaction.txStatus === '1',\n )\n .map((transactions) =>\n convertTransaction({\n transactions,\n explorerUrl,\n networkToken,\n chainId,\n address,\n }).then((tx) => tx),\n ),\n );\n\n const transactions = convertedTxs.filter(\n // Filtering txs with 0 value since there is no change in balance\n (transaction) => transaction.tokens.find((token) => Number(token.amount) !== 0),\n );\n\n return {\n transactions,\n nextPageToken: response.nextPageToken,\n };\n } catch {\n return {\n transactions: [],\n nextPageToken: '',\n };\n }\n};\n","import { getTransactionFromEtherscan } from './converters/etherscan-transaction-converter/get-transaction-from-etherscan';\nimport { isEthereumChainId } from './utils/is-ethereum-chain-id';\nimport type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { getTransactionsFromGlacier } from './converters/evm-transaction-converter/get-transactions-from-glacier';\nimport type { EvmGlacierService } from '../../services/glacier-service/glacier-service';\n\nexport const getTransactionHistory = async ({\n chainId,\n isTestnet,\n networkToken,\n explorerUrl,\n address,\n nextPageToken,\n offset,\n glacierService,\n}: {\n chainId: number;\n isTestnet?: boolean;\n networkToken: NetworkToken;\n explorerUrl: string;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierService: EvmGlacierService;\n}): Promise<TransactionHistoryResponse> => {\n if (isEthereumChainId(chainId)) {\n return getTransactionFromEtherscan({\n isTestnet,\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n });\n }\n\n const isHealthy = glacierService.isHealthy();\n if (!isHealthy) {\n return {\n transactions: [],\n nextPageToken: '',\n };\n }\n\n return getTransactionsFromGlacier({\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n glacierService,\n });\n};\n","{\n \"name\": \"EVM\",\n \"description\": \"\",\n \"version\": \"0.0.1\",\n \"sources\": {\n \"module\": {\n \"checksum\": \"\",\n \"location\": {\n \"npm\": {\n \"filePath\": \"dist/index.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n },\n \"provider\": {\n \"checksum\": \"\",\n \"location\": {\n \"npm\": {\n \"filePath\": \"dist/provider/index.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n }\n },\n \"network\": {\n \"chainIds\": [],\n \"namespaces\": [\"eip155\"]\n },\n \"cointype\": \"60\",\n \"permissions\": {\n \"rpc\": {\n \"dapps\": true,\n \"methods\": [\n \"personal_sign\",\n \"eth_sendTransaction\",\n \"eth_sendTransactionBatch\",\n \"eth_*\",\n \"web3_clientVersion\",\n \"web3_sha3\",\n \"net_version\",\n \"net_peerCount\",\n \"net_listening\"\n ],\n \"nonRestrictedMethods\": [\n \"eth_requestAccounts\",\n \"eth_decrypt\",\n \"eth_getEncryptionPublicKey\",\n \"eth_subscribe\",\n \"eth_unsubscribe\",\n \"eth_blockNumber\",\n \"eth_call\",\n \"eth_chainId\",\n \"eth_coinbase\",\n \"eth_estimateGas\",\n \"eth_feeHistory\",\n \"eth_gasPrice\",\n \"eth_getBalance\",\n \"eth_getBlockByHash\",\n \"eth_getBlockByNumber\",\n \"eth_getBlockTransactionCountByHash\",\n \"eth_getBlockTransactionCountByNumber\",\n \"eth_getCode\",\n \"eth_getFilterChanges\",\n \"eth_getFilterLogs\",\n \"eth_getLogs\",\n \"eth_getProof\",\n \"eth_getStorageAt\",\n \"eth_getTransactionByBlockHashAndIndex\",\n \"eth_getTransactionByBlockNumberAndIndex\",\n \"eth_getTransactionByHash\",\n \"eth_getTransactionCount\",\n \"eth_getTransactionReceipt\",\n \"eth_getUncleCountByBlockHash\",\n \"eth_getUncleCountByBlockNumber\",\n \"eth_newBlockFilter\",\n \"eth_newFilter\",\n \"eth_newPendingTransactionFilter\",\n \"eth_syncing\",\n \"eth_uninstallFilter\",\n \"net_version\",\n \"net_peerCount\",\n \"net_listening\",\n \"web3_sha3\",\n \"web3_clientVersion\"\n ]\n }\n },\n \"manifestVersion\": \"0.1\"\n}\n","import { type Network, type RpcRequest, type ApprovalController } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nimport { getNonce } from '../../utils/get-nonce';\nimport { getProvider } from '../../utils/get-provider';\nimport { getTxUpdater } from '../../utils/evm-tx-updater';\nimport { estimateGasLimit } from '../../utils/estimate-gas-limit';\nimport { buildTxApprovalRequest } from '../../utils/build-tx-approval-request';\nimport { simulateTransaction } from '../../utils/process-transaction-simulation';\n\nimport { parseRequestParams } from './schema';\nimport { waitForTransactionReceipt } from '../../utils/wait-for-transaction-receipt';\nimport { getTxHash } from '../../utils/get-tx-hash';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSendTransaction = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n blockaid: Blockaid;\n}) => {\n const { params } = request;\n\n // validate params\n const { data, error, success } = parseRequestParams(params);\n\n if (!success) {\n console.error('invalid params', error);\n return {\n error: rpcErrors.invalidParams({ message: 'Transaction params are invalid', data: { cause: error } }),\n };\n }\n\n const [transaction] = data;\n\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n // calculate gas limit if not provided/invalid\n if (!transaction.gas || Number(transaction.gas) < 0) {\n try {\n const gasLimit = await estimateGasLimit({\n transactionParams: {\n from: transaction.from,\n to: transaction.to,\n data: transaction.data,\n value: transaction.value,\n accessList: transaction.accessList,\n },\n provider,\n });\n\n transaction.gas = '0x' + gasLimit.toString(16);\n } catch (error) {\n return {\n error: rpcErrors.internal('Unable to calculate gas limit'),\n };\n }\n }\n\n // calculate nonce if not provided\n if (!transaction.nonce) {\n try {\n const nonce = await getNonce({\n from: transaction.from,\n provider,\n });\n transaction.nonce = String(nonce);\n } catch (error) {\n return {\n error: rpcErrors.internal('Unable to calculate nonce'),\n };\n }\n }\n\n const scan = await simulateTransaction({\n rpcMethod: request.method,\n chainId: network.chainId,\n params: transaction,\n dAppUrl: request.dappInfo.url,\n provider,\n blockaid,\n });\n\n const { displayData, signingData } = buildTxApprovalRequest(request, network, transaction, scan);\n\n const { updateTx, cleanup } = getTxUpdater(request.requestId, signingData, displayData);\n // prompt user for approval\n const response = await approvalController.requestApproval({ request, displayData, signingData, updateTx });\n\n cleanup();\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n let txHash;\n\n try {\n txHash = await getTxHash(provider, response);\n } catch (error) {\n return {\n error: rpcErrors.internal({ message: 'Unable to get transaction hash', data: { cause: error } }),\n };\n }\n\n waitForTransactionReceipt({\n explorerUrl: network.explorerUrl ?? '',\n provider,\n txHash,\n onTransactionPending: approvalController.onTransactionPending,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n request,\n });\n\n return { result: txHash };\n};\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nexport const getNonce = async ({\n from,\n provider,\n}: {\n from: string;\n provider: JsonRpcBatchInternal;\n}): Promise<number> => {\n return provider.getTransactionCount(from);\n};\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport type { DisplayData, EvmTxUpdateFn, SigningData_EthSendTx, TokenApprovals } from '@avalabs/vm-module-types';\n\nimport { ERC20TransactionType } from '../types';\nimport { parseERC20TransactionType } from './parse-erc20-transaction-type';\nimport { encodeApprovalLimit } from './encode-erc20-approval';\n\nconst requests = new Map<string, { displayData: DisplayData; signingData: SigningData_EthSendTx }>();\n\nexport const getTxUpdater = (\n requestId: string,\n signingData: SigningData_EthSendTx,\n displayData: DisplayData,\n): { updateTx: EvmTxUpdateFn; cleanup: () => void } => {\n requests.set(requestId, { signingData, displayData });\n\n return {\n updateTx: ({ maxFeeRate, maxTipRate, approvalLimit, gasLimit }) => {\n const request = requests.get(requestId);\n\n if (!request) {\n throw rpcErrors.resourceNotFound();\n }\n\n const { signingData, displayData } = request;\n\n const newSigningData = {\n ...signingData,\n data: {\n ...signingData.data,\n gasLimit: gasLimit ?? signingData.data.gasLimit,\n maxFeePerGas: maxFeeRate ?? signingData.data.maxFeePerGas,\n maxPriorityFeePerGas: maxTipRate ?? signingData.data.maxPriorityFeePerGas,\n },\n };\n\n const newDisplayData = { ...displayData };\n\n if (typeof approvalLimit === 'string') {\n if (!approvalLimit.startsWith('0x')) {\n throw rpcErrors.invalidInput('Expected approvalLimit to be a hexadecimal number (0x-prefixed)');\n }\n\n const tokenApprovals = displayData.tokenApprovals;\n\n ensureTokenApprovalCanBeEdited(tokenApprovals);\n ensureApproveWasCalled(signingData.data.data);\n\n const approval = tokenApprovals.approvals[0]!;\n\n newSigningData.data.data = encodeApprovalLimit(approval.token.address, approval.spenderAddress, approvalLimit);\n newDisplayData.tokenApprovals = {\n approvals: [\n {\n ...approval,\n value: approvalLimit,\n },\n ],\n isEditable: true,\n };\n }\n\n const updatedRequest = { signingData: newSigningData, displayData: newDisplayData };\n\n requests.set(requestId, updatedRequest);\n\n return updatedRequest;\n },\n cleanup: () => requests.delete(requestId),\n };\n};\n\nfunction ensureTokenApprovalCanBeEdited(\n tokenApprovals?: TokenApprovals,\n): asserts tokenApprovals is TokenApprovals & { isEditable: true } {\n if (!tokenApprovals || !tokenApprovals.isEditable || tokenApprovals.approvals.length !== 1) {\n throw rpcErrors.invalidInput(\n 'Cannot edit the token approval for this request. Please start a new request instead.',\n );\n }\n}\n\nfunction ensureApproveWasCalled(txData?: string | null) {\n const previousMethod = !txData || txData === '0x' ? null : parseERC20TransactionType({ data: txData ?? undefined });\n\n // If original tx was not setting any approvals, do not allow it to be set later on.\n if (previousMethod !== ERC20TransactionType.APPROVE) {\n throw rpcErrors.invalidInput(\n 'Cannot change invoked method for requests in progress. Please start a new request instead.',\n );\n }\n}\n","import { Interface } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { ERC20TransactionType } from '../types';\n\nexport const parseERC20TransactionType = (transaction: {\n data?: string;\n value?: string;\n}): ERC20TransactionType | undefined => {\n if (!transaction.data) {\n return undefined;\n }\n\n try {\n const contractInterface = new Interface(ERC20.abi);\n\n const description = contractInterface.parseTransaction({\n data: transaction.data,\n value: transaction.value,\n });\n\n const functionName = description?.name ?? description?.fragment?.name;\n\n if (functionName && isERC20TransactionType(functionName)) {\n return functionName;\n }\n\n return undefined;\n } catch (e) {\n return undefined;\n }\n};\n\nfunction isERC20TransactionType(value: string): value is ERC20TransactionType {\n return Object.values(ERC20TransactionType).includes(value as ERC20TransactionType);\n}\n","import { ethers } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { ERC20TransactionType } from '../types';\n\nexport function encodeApprovalLimit(tokenAddress: string, spenderAddress: string, limit: string) {\n const contract = new ethers.Contract(tokenAddress, ERC20.abi);\n\n return contract.interface.encodeFunctionData(ERC20TransactionType.APPROVE, [spenderAddress, limit]);\n}\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport type { AccessList, BigNumberish } from 'ethers';\n\nexport const estimateGasLimit = async ({\n transactionParams: { from, to, data, value, accessList },\n provider,\n}: {\n transactionParams: {\n from: string;\n to?: string;\n data?: string;\n value?: BigNumberish;\n accessList?: AccessList;\n };\n provider: JsonRpcBatchInternal;\n}): Promise<number> => {\n const nonce = await provider.getTransactionCount(from);\n\n return Number(\n await provider.estimateGas({\n from,\n to,\n nonce,\n data,\n value,\n accessList,\n }),\n );\n};\n","import {\n RpcMethod,\n type DetailItem,\n type DisplayData,\n type Network,\n type RpcRequest,\n type SigningData,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\n\nimport { addressItem, linkItem, networkItem } from '@internal/utils/src/utils/detail-item';\n\nimport { ERC20TransactionType } from '../types';\nimport type { TransactionParams } from './transaction-schema';\nimport { parseERC20TransactionType } from './parse-erc20-transaction-type';\n\nexport const buildTxApprovalRequest = (\n request: RpcRequest,\n network: Network,\n transaction: TransactionParams,\n { isSimulationSuccessful, balanceChange, tokenApprovals, alert }: TransactionSimulationResult,\n) => {\n const { dappInfo } = request;\n const transactionType = parseERC20TransactionType(transaction);\n\n // generate display and signing data\n let title = 'Do you approve this transaction?';\n if (transactionType === ERC20TransactionType.APPROVE) {\n title = 'Do you approve this spend limit?';\n }\n\n const transactionDetails: DetailItem[] = [\n addressItem('Account', transaction.from),\n networkItem('Network', {\n name: network.chainName,\n logoUri: network.logoUri,\n }),\n linkItem('Website', dappInfo),\n ];\n\n if (transaction.to) {\n transactionDetails.push(addressItem('Contract', transaction.to));\n }\n\n const displayData: DisplayData = {\n title,\n details: [\n {\n title: 'Transaction Details',\n items: transactionDetails,\n },\n ],\n networkFeeSelector: true,\n alert,\n balanceChange,\n tokenApprovals,\n isSimulationSuccessful,\n };\n\n const signingData: SigningData = {\n type: RpcMethod.ETH_SEND_TRANSACTION,\n account: transaction.from,\n data: {\n type: 2, // hardcoding to 2 for now as we only support EIP-1559\n nonce: Number(transaction.nonce),\n gasLimit: Number(transaction.gas),\n maxFeePerGas: transaction.maxFeePerGas,\n maxPriorityFeePerGas: transaction.maxPriorityFeePerGas,\n to: transaction.to,\n from: transaction.from,\n data: transaction.data,\n value: transaction.value,\n chainId: transaction.chainId ?? network.chainId,\n accessList: transaction.accessList,\n },\n };\n\n return {\n displayData,\n signingData,\n };\n};\n","import Blockaid from '@blockaid/client';\nimport type { TransactionParams } from '../types';\nimport {\n type NetworkContractToken,\n type NetworkToken,\n TokenType,\n AlertType,\n type Alert,\n type BalanceChange,\n type TokenApproval,\n type TokenDiff,\n type TokenDiffItem,\n type TokenApprovals,\n type RpcRequest,\n RpcMethod,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue, numberToBN } from '@avalabs/core-utils-sdk';\nimport { isHexString, MaxUint256 } from 'ethers';\nimport { scanJsonRpc, scanTransaction } from './scan-transaction';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { parseWithErc20Abi } from './parse-erc20-tx';\nimport { hasToField } from './type-utils';\nimport { transactionAlerts } from './transaction-alerts';\n\ntype Erc20ExposureTrace = Blockaid.Evm.AccountSummary.Erc20ExposureTrace;\ntype Erc721ExposureTrace = Blockaid.Evm.AccountSummary.Erc721ExposureTrace;\ntype Erc1155ExposureTrace = Blockaid.Evm.AccountSummary.Erc1155ExposureTrace;\n\n/*\n * Although in the type definition they don't specify it, but for traces they are returning the asset as well:\n * https://docs.blockaid.io/changelog/november-12-2024#account-traces\n * */\ntype ExposureTrace =\n | (Erc20ExposureTrace & { asset: Blockaid.Evm.Erc20TokenDetails })\n | (Erc721ExposureTrace & { asset: Blockaid.Evm.Erc721TokenDetails })\n | (Erc1155ExposureTrace & { asset: Blockaid.Evm.Erc1155TokenDetails });\ntype Trace = Blockaid.Evm.AccountSummary['traces']['0'];\nexport type AssetDiffs = Blockaid.Evm.AccountSummary['assets_diffs'];\n\nexport const simulateTransaction = async ({\n rpcMethod,\n dAppUrl,\n params,\n chainId,\n provider,\n blockaid,\n}: {\n rpcMethod: RpcMethod;\n dAppUrl?: string;\n params: TransactionParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n blockaid: Blockaid;\n}) => {\n let simulationResult: Pick<Blockaid.TransactionScanResponse, 'simulation' | 'validation'> | undefined;\n\n try {\n simulationResult = await scanTransaction({\n chainId,\n params,\n domain: dAppUrl,\n blockaid,\n });\n } catch (error) {\n console.error('simulateTransaction error', error);\n }\n\n return processTransactionSimulation({ rpcMethod, params, chainId, provider, simulationResult });\n};\n\nexport const processTransactionSimulation = async ({\n rpcMethod,\n params,\n chainId,\n provider,\n simulationResult,\n}: {\n rpcMethod: RpcMethod;\n params: TransactionParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n simulationResult?: Pick<Blockaid.TransactionScanResponse, 'simulation' | 'validation' | 'gas_estimation'>;\n}): Promise<TransactionSimulationResult> => {\n let alert: Alert | undefined;\n let balanceChange: BalanceChange | undefined;\n let tokenApprovals: TokenApprovals | undefined;\n let isSimulationSuccessful = false;\n let estimatedGasLimit: number | undefined;\n\n if (simulationResult) {\n const { validation, simulation, gas_estimation: gasEstimation } = simulationResult;\n\n if (!validation || validation.result_type === 'Error' || validation.result_type === 'Warning') {\n alert = transactionAlerts[AlertType.WARNING];\n } else if (validation.result_type === 'Malicious') {\n alert = transactionAlerts[AlertType.DANGER];\n }\n\n if (simulation?.status === 'Success') {\n isSimulationSuccessful = true;\n tokenApprovals = processTokenApprovals(rpcMethod, simulation.account_summary);\n balanceChange = processBalanceChange(simulation.account_summary.assets_diffs);\n }\n\n if (gasEstimation?.status === 'Success') {\n estimatedGasLimit = Number(gasEstimation.estimate);\n }\n }\n\n // If debank parsing failed, check if toAddress is a known ERC20\n if (!isSimulationSuccessful && hasToField(params)) {\n const erc20ParseResult = await parseWithErc20Abi(params, chainId, provider);\n balanceChange = erc20ParseResult.balanceChange;\n tokenApprovals = erc20ParseResult.tokenApprovals;\n }\n\n return { alert, balanceChange, tokenApprovals, isSimulationSuccessful, estimatedGasLimit };\n};\n\nconst isExposureTrace = (trace: Trace): trace is ExposureTrace => {\n return (trace as ExposureTrace).trace_type === 'ExposureTrace';\n};\n\nconst normalizeAddresses = (spender: string, assetAddress: string): string =>\n `${spender.toLowerCase()}.${assetAddress.toLowerCase()}`;\n\nconst mapExposureTracesToSpenderAsset = (traces: Trace[]): Record<string, ExposureTrace> => {\n if (traces === undefined || traces.length === 0) {\n return {};\n }\n\n return traces.reduce(\n (accumulator, trace) => {\n if (!isExposureTrace(trace)) {\n return accumulator;\n }\n return {\n [normalizeAddresses(trace.spender, trace.asset.address)]: trace,\n };\n },\n {} as Record<string, ExposureTrace>,\n );\n};\n\nconst processTokenApprovals = (\n rpcMethod: RpcMethod,\n accountSummary: Blockaid.Evm.AccountSummary,\n): TokenApprovals | undefined => {\n const { traces } = accountSummary;\n\n const mappedExposureTraces = mapExposureTracesToSpenderAsset(traces);\n\n const approvals = Object.entries(mappedExposureTraces)\n .map(([_, trace]) => {\n const token = convertAssetToNetworkContractToken(trace.asset);\n if (!token) {\n return;\n }\n\n const tokenApproval: TokenApproval = {\n token,\n spenderAddress: trace.spender,\n logoUri: token.logoUri,\n };\n\n if (trace.type === 'ERC20ExposureTrace') {\n const {\n exposed: { raw_value, usd_price },\n } = trace as Erc20ExposureTrace;\n\n tokenApproval.value = raw_value;\n tokenApproval.usdPrice = `${usd_price}`;\n }\n\n if (trace.type === 'ERC721ExposureTrace') {\n // When dApp attempts to call setApprovalForAll(), the \"exposed\" field is not present.\n // However, the result of such transaction is that the spender gets approval for all\n // tokens in the NFT collection, so we treat it the same as \"Unlimited\" approval for ERC-20.\n if (!trace.exposed) {\n tokenApproval.value = `0x${MaxUint256.toString(16)}`;\n } else {\n const { usd_price, amount } = trace.exposed;\n\n tokenApproval.value = `${amount}`;\n tokenApproval.usdPrice = `${usd_price}`;\n }\n }\n\n return tokenApproval;\n })\n .filter(Boolean) as TokenApproval[];\n\n if (approvals.length === 0) {\n return undefined;\n }\n\n const isEditable =\n approvals.length === 1 &&\n approvals[0]?.token.type === TokenType.ERC20 &&\n rpcMethod === RpcMethod.ETH_SEND_TRANSACTION;\n\n return { isEditable, approvals };\n};\n\nexport const processBalanceChange = (assetDiffs: AssetDiffs): BalanceChange | undefined => {\n const ins = processAssetDiffs(assetDiffs, 'in');\n const outs = processAssetDiffs(assetDiffs, 'out');\n\n if (ins.length === 0 && outs.length === 0) {\n return undefined;\n }\n\n return { ins, outs };\n};\n\nconst processAssetDiffs = (assetDiffs: AssetDiffs, type: 'in' | 'out'): TokenDiff[] => {\n return (\n assetDiffs\n .filter((assetDiff) => assetDiff[type].length > 0)\n // sort asset diffs by length of in/out array\n // this is done to ensure that the token with multiple in/out values are displayed last,\n // to put them in groups with appropriate UI(i.e. accordion), after single in/out tokens\n .sort((a, b) => a[type].length - b[type].length)\n .map((assetDiff) => {\n const asset = assetDiff.asset;\n // convert blockaid asset to network token\n const token: NetworkToken | NetworkContractToken | undefined =\n 'address' in asset ? convertAssetToNetworkContractToken(asset) : convertNativeAssetToToken(asset);\n if (!token) {\n return undefined;\n }\n\n const items = assetDiff[type]\n .map((diff) => {\n let displayValue;\n if ('value' in diff && diff.value) {\n if ('decimals' in token) {\n const valueBN = numberToBN(diff.value, token.decimals);\n displayValue = balanceToDisplayValue(valueBN, token.decimals);\n } else if (isHexString(diff.value)) {\n // for some token (like ERC1155) blockaid returns value in hex format\n displayValue = parseInt(diff.value, 16).toString();\n }\n } else if ('type' in token && token.type === TokenType.ERC721) {\n // for ERC721 type token, we just display 1 to indicate that a single NFT will be transferred\n displayValue = '1';\n }\n\n return displayValue ? { displayValue, usdPrice: diff.usd_price } : undefined;\n })\n .filter((x): x is TokenDiffItem => x !== undefined);\n\n return { token, items };\n })\n .filter((x): x is TokenDiff => x !== undefined)\n );\n};\n\nconst convertAssetToNetworkContractToken = (\n asset:\n | Blockaid.Erc20TokenDetails\n | Blockaid.Erc1155TokenDetails\n | Blockaid.Erc721TokenDetails\n | Blockaid.NonercTokenDetails,\n): NetworkContractToken | undefined => {\n let token: NetworkContractToken | undefined;\n if (asset.type === 'ERC20') {\n token = {\n type: TokenType.ERC20,\n address: asset.address,\n decimals: asset.decimals,\n name: asset.name ?? asset.symbol ?? '',\n symbol: asset.symbol ?? '',\n logoUri: asset.logo_url,\n };\n } else if (asset.type === 'ERC1155') {\n token = {\n type: TokenType.ERC1155,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n } else if (asset.type === 'ERC721') {\n token = {\n type: TokenType.ERC721,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n } else if (asset.type === 'NONERC') {\n token = {\n type: TokenType.NONERC,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n }\n\n return token;\n};\n\nconst convertNativeAssetToToken = (asset: Blockaid.NativeAssetDetails): NetworkToken => {\n return {\n name: asset.name ?? '',\n symbol: asset.symbol ?? '',\n decimals: asset.decimals,\n description: '',\n logoUri: asset.logo_url,\n };\n};\n\nexport const processJsonRpcSimulation = async ({\n request,\n dAppUrl,\n accountAddress,\n chainId,\n data,\n blockaid,\n}: {\n request: RpcRequest;\n dAppUrl?: string;\n accountAddress: string;\n data: { method: string; params: unknown };\n chainId: number;\n blockaid: Blockaid;\n}) => {\n let alert: Alert | undefined;\n let balanceChange: BalanceChange | undefined;\n let tokenApprovals: TokenApprovals | undefined;\n\n try {\n const { validation, simulation } = await scanJsonRpc({\n chainId,\n accountAddress,\n data: data as Blockaid.Evm.JsonRpcScanParams.Data,\n domain: dAppUrl,\n blockaid,\n });\n\n if (!validation || validation.result_type === 'Error' || validation.result_type === 'Warning') {\n alert = transactionAlerts[AlertType.WARNING];\n } else if (validation.result_type === 'Malicious') {\n alert = transactionAlerts[AlertType.DANGER];\n }\n\n if (simulation?.status === 'Success') {\n tokenApprovals = processTokenApprovals(request.method, simulation.account_summary);\n balanceChange = processBalanceChange(simulation.account_summary.assets_diffs);\n }\n } catch (error) {\n console.error('processJsonRpcSimulation error', error);\n alert = transactionAlerts[AlertType.WARNING];\n }\n\n return { alert, balanceChange, tokenApprovals };\n};\n","import Blockaid from '@blockaid/client';\nimport type { TransactionBulkScanParams } from '@blockaid/client/resources/evm/transaction-bulk';\n\nimport type { TransactionBatchParams, TransactionParams } from '../types';\n\nexport const scanTransactionBatch = async ({\n chainId,\n params,\n domain,\n withGasEstimation,\n blockaid,\n}: {\n chainId: number;\n params: TransactionBatchParams;\n domain?: string;\n withGasEstimation?: boolean;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse[]> => {\n const options: TransactionBulkScanParams['options'] = ['validation', 'simulation'];\n\n if (withGasEstimation) {\n options.push('gas_estimation');\n }\n\n return blockaid.evm.transactionBulk.scan({\n chain: chainId.toString(),\n options,\n data: params,\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n\nexport const scanTransaction = async ({\n chainId,\n params,\n domain,\n blockaid,\n}: {\n chainId: number;\n params: TransactionParams;\n domain?: string;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse> => {\n return blockaid.evm.transaction.scan({\n account_address: params.from,\n chain: chainId.toString(),\n options: ['validation', 'simulation'],\n data: {\n from: params.from,\n to: params.to,\n data: params.data,\n value: params.value,\n gas: params.gas,\n gas_price: params.gasPrice,\n // TODO: provide accessList once Blockaid supports it\n // access_list: params.accessList\n },\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n\nexport const scanJsonRpc = async ({\n chainId,\n accountAddress,\n data,\n domain,\n blockaid,\n}: {\n chainId: number;\n accountAddress: string;\n data: Blockaid.Evm.JsonRpcScanParams.Data;\n domain?: string;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse> => {\n return blockaid.evm.jsonRpc.scan({\n chain: chainId.toString(),\n options: ['validation', 'simulation'],\n account_address: accountAddress,\n data,\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n","import xss from 'xss';\nimport { ethers } from 'ethers';\nimport { TokenType } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport { ERC20TransactionType, type RequiredBy, type TransactionParams } from '../types';\n\nexport const parseWithErc20Abi = async (\n params: RequiredBy<TransactionParams, 'to'>,\n chainId: number,\n provider: JsonRpcBatchInternal,\n) => {\n if (!params.data) {\n return {\n tokenApprovals: undefined,\n balanceChange: undefined,\n };\n }\n\n try {\n const contract = new ethers.Contract(params.to, ERC20.abi, provider);\n const contractCalls = await Promise.all([contract.name?.(), contract.symbol?.(), contract.decimals?.()]);\n // Purify the values for XSS protection\n const name = xss(contractCalls[0]);\n const symbol = xss(contractCalls[1]);\n const decimals = parseInt(contractCalls[2]);\n const token = {\n type: TokenType.ERC20,\n name,\n chainId,\n symbol,\n decimals,\n address: params.to,\n contractType: 'ERC-20',\n } as const;\n\n const iface = new ethers.Interface(ERC20.abi);\n const calledFunction = iface.getFunction(params.data.slice(0, 10));\n\n const decodeFunctionData = iface.decodeFunctionData(params.data.slice(0, 10), params.data);\n if (calledFunction?.name === ERC20TransactionType.TRANSFER) {\n return {\n balanceChange: {\n outs: [\n {\n items: [\n {\n displayValue: new TokenUnit(decodeFunctionData['amount'], token.decimals, token.symbol).toDisplay(),\n usdPrice: undefined,\n },\n ],\n token,\n },\n ],\n ins: [],\n },\n };\n } else if (calledFunction?.name === ERC20TransactionType.APPROVE) {\n return {\n tokenApprovals: {\n isEditable: true,\n approvals: [\n {\n token,\n spenderAddress: decodeFunctionData['spender'],\n value: decodeFunctionData['amount'],\n },\n ],\n },\n };\n }\n } catch (error) {\n console.error('parseErc20Tx error', error);\n }\n\n return {\n tokenApprovals: undefined,\n balanceChange: undefined,\n };\n};\n","import {\n TokenType,\n type ApprovalController,\n type BalanceChange,\n type BatchApprovalController,\n type ERC1155Token,\n type ERC20Token,\n type ERC721Token,\n type NetworkContractToken,\n type NetworkToken,\n type TokenApprovals,\n} from '@avalabs/vm-module-types';\nimport type { RequiredBy, TransactionParams } from '../types';\n\nexport function isNetworkToken(token: NetworkContractToken | NetworkToken): token is NetworkToken {\n return !('type' in token);\n}\n\nexport function isERC20Token(token: NetworkContractToken): token is ERC20Token {\n return token.type === TokenType.ERC20;\n}\n\nexport function isNftToken(token: NetworkContractToken): token is ERC1155Token | ERC721Token {\n return token.type === TokenType.ERC1155 || token.type === TokenType.ERC721;\n}\n\nexport function hasToField(params: TransactionParams): params is RequiredBy<TransactionParams, 'to'> {\n return typeof params.to === 'string';\n}\n\nexport function supportsBatchApprovals(\n controller: ApprovalController | BatchApprovalController,\n): controller is BatchApprovalController {\n return 'requestBatchApproval' in controller && typeof controller.requestBatchApproval === 'function';\n}\n\nexport function isBalanceChange(input: unknown): input is BalanceChange {\n return (\n typeof input === 'object' &&\n input !== null &&\n 'ins' in input &&\n 'outs' in input &&\n Array.isArray(input.ins) &&\n Array.isArray(input.outs)\n );\n}\n\nexport function isTokenApprovals(input: unknown): input is TokenApprovals {\n return (\n typeof input === 'object' &&\n input !== null &&\n 'approvals' in input &&\n 'isEditable' in input &&\n Array.isArray(input.approvals)\n );\n}\n\nexport function isEmpty(input: unknown): boolean {\n if (isBalanceChange(input)) {\n return input.ins.length === 0 && input.outs.length === 0;\n }\n\n if (isTokenApprovals(input)) {\n return input.approvals.length === 0;\n }\n\n return false;\n}\n","import { AlertType } from '@avalabs/vm-module-types';\n\nexport const transactionAlerts = {\n [AlertType.WARNING]: {\n type: AlertType.WARNING,\n details: {\n title: 'Suspicious transaction',\n description: 'Use caution, this transaction might be malicious.',\n },\n },\n [AlertType.DANGER]: {\n type: AlertType.DANGER,\n details: {\n title: 'Scam transaction',\n description: 'This transaction has been flagged as malicious, I understand the risk.',\n body: ['This transaction is malicious', 'do not proceed'],\n actionTitles: {\n reject: 'Reject Transaction',\n proceed: 'Proceed Anyway',\n },\n },\n },\n};\n","import { z } from 'zod';\n\nimport { transactionSchema } from '../../utils/transaction-schema';\n\nconst paramsSchema = z.tuple([transactionSchema]);\n\nexport const parseRequestParams = (params: unknown) => {\n return paramsSchema.safeParse(params);\n};\n","import { z } from 'zod';\n\nexport const transactionSchema = z.object({\n from: z.string().length(42),\n to: z.string().length(42).optional(),\n data: z.string().optional(),\n value: z.string().startsWith('0x').optional(),\n gas: z.string().startsWith('0x').optional(),\n gasPrice: z.string().startsWith('0x').optional(),\n maxFeePerGas: z.string().startsWith('0x').optional(),\n maxPriorityFeePerGas: z.string().startsWith('0x').optional(),\n nonce: z.string().optional(),\n chainId: z.string().optional().or(z.number().optional()),\n accessList: z\n .array(\n z.object({\n address: z.string().startsWith('0x'),\n storageKeys: z.array(z.string()),\n }),\n )\n .optional(),\n});\n\nexport type TransactionParams = z.infer<typeof transactionSchema>;\n","import type { Hex, RpcRequest } from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getExplorerAddressByNetwork } from '../handlers/get-transaction-history/utils/get-explorer-address-by-network';\nimport type { TransactionReceipt } from 'ethers';\nimport { retry, RetryBackoffPolicy } from '@internal/utils';\n\nexport const waitForTransactionReceipt = async ({\n explorerUrl,\n provider,\n txHash,\n onTransactionPending,\n onTransactionConfirmed,\n onTransactionReverted,\n request,\n}: {\n explorerUrl: string;\n provider: JsonRpcBatchInternal;\n txHash: Hex;\n onTransactionPending: ({ txHash, request }: { txHash: Hex; request: RpcRequest }) => void;\n onTransactionConfirmed: ({\n txHash,\n explorerLink,\n request,\n }: {\n txHash: Hex;\n explorerLink: string;\n request: RpcRequest;\n }) => void;\n onTransactionReverted: ({ txHash, request }: { txHash: Hex; request: RpcRequest }) => void;\n request: RpcRequest;\n}) => {\n try {\n onTransactionPending({ txHash, request });\n\n const receipt = await retry<TransactionReceipt | null>({\n operation: async () => provider.getTransactionReceipt(txHash),\n isSuccess: (r): r is TransactionReceipt => !!r, // success when receipt is present (>= 1 confirmation)\n backoffPolicy: RetryBackoffPolicy.linearThenExponential(4, 1000),\n maxRetries: 20,\n });\n\n const success = receipt?.status === 1; // 1 indicates success, 0 indicates revert\n\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, txHash);\n\n if (success) {\n onTransactionConfirmed({ txHash, explorerLink, request });\n return true;\n }\n } catch (error) {\n console.error(error);\n }\n\n onTransactionReverted({ txHash, request });\n\n return false;\n};\n","import type { SigningResult } from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nexport const getTxHash = async (provider: JsonRpcBatchInternal, response: SigningResult) => {\n if ('txHash' in response) {\n return response.txHash;\n }\n\n // broadcast the signed transaction\n const txHash = await provider.send('eth_sendRawTransaction', [response.signedData]);\n return txHash;\n};\n","import {\n type GetBalancesParams,\n type NetworkTokenWithBalance,\n type Storage,\n type Error,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n TokenType,\n} from '@avalabs/vm-module-types';\nimport type { TokenService } from '@internal/utils';\nimport { findAsync } from '../../utils/find-async';\nimport type { BalanceServiceInterface } from './balance-service-interface';\nimport type { CurrencyCode } from '@avalabs/glacier-sdk';\nimport { addIdToPromise, type IdPromise, settleAllIdPromises } from '../../utils/id-promise';\nimport { RpcService } from '../../services/rpc-service/rpc-service';\nimport { isERC20Token } from '../../utils/type-utils';\nimport type { VsCurrencyType } from '@avalabs/core-coingecko-sdk';\n\ntype AccountAddress = string;\ntype TokenSymbol = string;\ntype GetEvmBalancesResponse = Record<AccountAddress, Record<TokenSymbol, TokenWithBalanceEVM | Error> | Error>;\n\nexport const getBalances = async ({\n addresses,\n currency,\n network,\n proxyApiUrl,\n tokenTypes = [TokenType.NATIVE, TokenType.ERC20, TokenType.ERC721, TokenType.ERC1155],\n customTokens = [],\n storage,\n balanceServices = [],\n tokenService,\n}: GetBalancesParams & {\n proxyApiUrl: string;\n balanceServices: BalanceServiceInterface[];\n storage?: Storage;\n tokenService: TokenService;\n}): Promise<GetEvmBalancesResponse> => {\n const chainId = network.chainId;\n const services: BalanceServiceInterface[] = [\n ...balanceServices,\n new RpcService({ network, storage, proxyApiUrl, customTokens }),\n ];\n\n const supportingService: BalanceServiceInterface | undefined = await findAsync(\n services,\n (balanceService: BalanceServiceInterface) => balanceService.isNetworkSupported(network.chainId),\n );\n\n const balances: GetEvmBalancesResponse = {};\n if (supportingService) {\n const nativeTokenPromises: Promise<IdPromise<NetworkTokenWithBalance>>[] = [];\n const erc20TokenPromises: Promise<IdPromise<Record<string, TokenWithBalanceEVM | Error>>>[] = [];\n const nftTokenPromises: Promise<IdPromise<Record<string, NftTokenWithBalance | Error>>>[] = [];\n addresses.forEach((address) => {\n if (tokenTypes.includes(TokenType.NATIVE)) {\n nativeTokenPromises.push(\n addIdToPromise(\n supportingService\n .getNativeBalance({\n address,\n currency: currency.toUpperCase() as CurrencyCode,\n chainId,\n })\n .then(async (nativeBalance) => {\n if (nativeBalance.symbol === 'AVAX') {\n // we want to standardise AVAX price across the chains\n const avaxMarketData = await tokenService.getWatchlistDataForToken({\n tokenDetails: {\n symbol: network.networkToken.symbol,\n isNative: true,\n caip2Id: network.caipId ?? '',\n },\n currency: currency.toUpperCase() as VsCurrencyType,\n });\n\n return {\n ...nativeBalance,\n priceInCurrency: avaxMarketData.priceInCurrency,\n marketCap: avaxMarketData.marketCap,\n vol24: avaxMarketData.vol24,\n change24: avaxMarketData.change24,\n };\n }\n\n return nativeBalance;\n }),\n address,\n ),\n );\n }\n\n if (tokenTypes.includes(TokenType.ERC20)) {\n erc20TokenPromises.push(\n addIdToPromise(\n supportingService.listErc20Balances({\n customTokens: customTokens.filter(isERC20Token),\n currency: currency.toUpperCase() as CurrencyCode,\n chainId,\n address,\n pageSize: 100,\n }),\n address,\n ),\n );\n }\n\n if (tokenTypes.includes(TokenType.ERC721) || tokenTypes.includes(TokenType.ERC1155)) {\n nftTokenPromises.push(\n addIdToPromise(\n supportingService.listNftBalances({\n chainId,\n address,\n }),\n address,\n ),\n );\n }\n });\n const nativeTokenBalances = await settleAllIdPromises(nativeTokenPromises);\n const erc20TokenBalances = await settleAllIdPromises(erc20TokenPromises);\n const nftTokenBalances = await settleAllIdPromises(nftTokenPromises);\n Object.keys(nativeTokenBalances).forEach((address) => {\n const balanceOrError = nativeTokenBalances[address];\n if (!balanceOrError || 'error' in balanceOrError) {\n balances[address] = {\n error: `getNativeBalance failed: ${balanceOrError?.error ?? 'unknown error'}`,\n } as Error;\n return;\n }\n const tokenSymbol = balanceOrError.symbol as string;\n balances[address] = {\n [tokenSymbol]: balanceOrError,\n };\n });\n Object.keys(erc20TokenBalances).forEach((address) => {\n const balancesOrError = erc20TokenBalances[address];\n if (!balancesOrError || ('error' in balancesOrError && typeof balancesOrError.error !== 'string')) {\n balances[address] = {\n ...balances[address],\n error: `listErc20Balances failed: unknown error`,\n };\n } else if ('error' in balancesOrError && typeof balancesOrError.error === 'string') {\n balances[address] = {\n ...balances[address],\n error: `listErc20Balances failed: ${balancesOrError.error}`,\n };\n } else {\n balances[address] = {\n ...balances[address],\n ...balancesOrError,\n };\n }\n });\n Object.keys(nftTokenBalances).forEach((address) => {\n const balancesOrError = nftTokenBalances[address];\n if (!balancesOrError || ('error' in balancesOrError && typeof balancesOrError.error !== 'string')) {\n balances[address] = {\n ...balances[address],\n error: `listNftBalances failed: unknown error`,\n };\n } else if ('error' in balancesOrError && typeof balancesOrError.error === 'string') {\n balances[address] = {\n ...balances[address],\n error: `listNftBalances failed: ${balancesOrError.error}`,\n };\n } else {\n balances[address] = {\n ...balances[address],\n ...balancesOrError,\n };\n }\n });\n } else {\n addresses.forEach((address) => {\n balances[address] = {\n error: 'unsupported network',\n };\n });\n }\n\n return balances;\n};\n","/**\n * Asynchronously searches through an array to find the first element that satisfies the provided async callback function.\n *\n * @param {T[]} array - The array to search through.\n * @param {(item: T) => Promise<boolean>} asyncCallback - An async function that takes an element of the array and returns a Promise resolving to a boolean.\n * @returns {Promise<T | undefined>} - A Promise that resolves to the first element in the array that satisfies the asyncCallback function, or undefined if no such element is found.\n *\n * @example\n * ```ts\n * const array = [1, 2, 3, 4];\n * const asyncCallback = async (num) => {\n * return new Promise((resolve) => {\n * setTimeout(() => {\n * resolve(num % 2 === 0);\n * }, 100);\n * });\n * };\n *\n * findAsync(array, asyncCallback).then((result) => {\n * console.log(result); // Output: 2 (the first even number)\n * });\n * ```\n */\nexport async function findAsync<T>(array: T[], asyncCallback: (item: T) => Promise<boolean>): Promise<T | undefined> {\n const promises = array.map(async (item, index) => ({\n index,\n result: await asyncCallback(item),\n }));\n\n const results = await Promise.allSettled(promises);\n\n const found = results\n .filter((item) => item.status === 'fulfilled')\n .map((item) => item as PromiseFulfilledResult<{ index: number; result: boolean }>)\n .find((item) => {\n return item.value.result;\n });\n return found ? array[found.value.index] : undefined;\n}\n","import { type Error } from '@avalabs/vm-module-types';\n\ntype Id = string;\nexport type IdPromise<T> =\n | { id: string; status: 'fulfilled'; value: T }\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n | { id: string; status: 'rejected'; reason: any };\n\n/**\n * The addIdToPromise function takes an existing promise and an id string as input and returns a new promise that\n * resolves to an object containing the id, the status of the original promise (either 'fulfilled' or 'rejected'),\n * and the result or error of the original promise.\n *\n * Usage:\n * This function is useful for tracking the status and outcome of multiple promises when used with Promise.allSettled\n *\n * @param promise The original promise that resolves to a value of type T.\n * @param id A unique identifier associated with the promise.\n */\nexport function addIdToPromise<T>(promise: Promise<T>, id: string): Promise<IdPromise<T>> {\n return promise\n .then((value) => ({ id, status: 'fulfilled', value }) as IdPromise<T>)\n .catch((reason) => ({ id, status: 'rejected', reason }) as IdPromise<T>);\n}\n\n/**\n * The settleAllIdPromises function processes an array of promises that return IdPromise objects,\n * resolving all of them and organizing the results into a record.\n *\n * Usage:\n * Use {@link addIdToPromise} to get array of promises\n *\n * @param promises An array of promises that each resolve to an IdPromise<T> object.\n */\nexport async function settleAllIdPromises<T>(promises: Promise<IdPromise<T>>[]): Promise<Record<Id, T | Error>> {\n return await Promise.allSettled(promises).then((results) => {\n return results.reduce(\n (acc, result) => {\n if (result.status === 'fulfilled') {\n const value = result.value;\n if (value.status === 'fulfilled') {\n acc[value.id] = value.value;\n } else {\n acc[value.id] = { error: value.reason };\n }\n }\n return acc;\n },\n {} as Record<Id, T | Error>,\n );\n });\n}\n","import { CurrencyCode } from '@avalabs/glacier-sdk';\nimport type { BalanceServiceInterface, TokenId } from '../../handlers/get-balances/balance-service-interface';\nimport {\n type ERC20Token,\n type Network,\n type NetworkContractToken,\n type NetworkTokenWithBalance,\n type Storage,\n type Error,\n TokenType,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n} from '@avalabs/vm-module-types';\nimport { addIdToPromise, settleAllIdPromises } from '../../utils/id-promise';\nimport type { VsCurrencyType } from '@avalabs/core-coingecko-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { ethers } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { getProvider } from '../../utils/get-provider';\nimport { TokenService } from '@internal/utils';\nimport { getTokens } from '../../handlers/get-tokens/get-tokens';\nimport { isERC20Token } from '../../utils/type-utils';\n\nexport class RpcService implements BalanceServiceInterface {\n #network: Network;\n #storage: Storage | undefined;\n #proxyApiUrl: string;\n #customTokens: NetworkContractToken[];\n\n constructor({\n network,\n storage,\n proxyApiUrl,\n customTokens,\n }: {\n network: Network;\n storage?: Storage;\n proxyApiUrl: string;\n customTokens: NetworkContractToken[];\n }) {\n this.#network = network;\n this.#storage = storage;\n this.#proxyApiUrl = proxyApiUrl;\n this.#customTokens = customTokens;\n }\n\n async isNetworkSupported(): Promise<boolean> {\n return true;\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n }): Promise<NetworkTokenWithBalance> {\n const provider = await getProvider({\n chainId,\n chainName: this.#network.chainName,\n rpcUrl: this.#network.rpcUrl,\n multiContractAddress: this.#network.utilityAddresses?.multicall,\n });\n\n const coingeckoTokenId = this.#network.pricingProviders?.coingecko.nativeTokenId;\n const networkToken = this.#network.networkToken;\n const tokenService = new TokenService({ storage: this.#storage, proxyApiUrl: this.#proxyApiUrl });\n const simplePriceResponse = coingeckoTokenId\n ? await tokenService.getSimplePrice({\n coinIds: [coingeckoTokenId],\n currencies: [currency] as unknown as VsCurrencyType[],\n })\n : {};\n\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? undefined;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? undefined;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? undefined;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? undefined;\n\n const balance = await provider.getBalance(address);\n const balanceUnit = new TokenUnit(balance, networkToken.decimals, networkToken.symbol);\n const balanceInCurrency = priceInCurrency !== undefined ? balanceUnit.mul(priceInCurrency) : undefined;\n\n return {\n ...networkToken,\n coingeckoId: coingeckoTokenId ?? '',\n type: TokenType.NATIVE,\n balance,\n balanceDisplayValue: balanceUnit.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n marketCap,\n vol24,\n change24,\n };\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n pageSize: number;\n pageToken?: string;\n }): Promise<Record<TokenId, TokenWithBalanceEVM | Error>> {\n const provider = await getProvider({\n chainId,\n chainName: this.#network.chainName,\n rpcUrl: this.#network.rpcUrl,\n multiContractAddress: this.#network.utilityAddresses?.multicall,\n });\n\n const coingeckoPlatformId = this.#network.pricingProviders?.coingecko.assetPlatformId;\n const coingeckoTokenId = this.#network.pricingProviders?.coingecko.nativeTokenId;\n const tokens = await getTokens({ chainId: Number(chainId), proxyApiUrl: this.#proxyApiUrl });\n const tokenAddresses = tokens.map((token) => token.address);\n const erc20TokenList = [...tokens, ...this.#customTokens].filter(isERC20Token);\n\n const getTokenWithBalance = async (token: ERC20Token) => {\n const contract = new ethers.Contract(token.address, ERC20.abi, provider);\n const balanceBig: bigint = await contract.balanceOf?.(address);\n const balance = balanceBig || 0n;\n\n return {\n ...token,\n balance,\n };\n };\n\n const tokenBalancePromises = erc20TokenList.map((token) => {\n return addIdToPromise(getTokenWithBalance(token), token.address);\n });\n const tokenBalancesResults = await settleAllIdPromises(tokenBalancePromises);\n const tokenService = new TokenService({ storage: this.#storage, proxyApiUrl: this.#proxyApiUrl });\n const simplePriceResponse =\n (coingeckoPlatformId &&\n (await tokenService.getPricesByAddresses(\n tokenAddresses,\n coingeckoPlatformId,\n currency as unknown as VsCurrencyType,\n ))) ||\n {};\n\n const tokenIds = Object.keys(tokenBalancesResults);\n const erc20TokenBalances: Record<TokenId, TokenWithBalanceEVM | Error> = {};\n for (let i = 0; i < tokenIds.length; i++) {\n const tokenId = tokenIds[i];\n if (tokenId === undefined) continue;\n const tokenBalance = tokenBalancesResults[tokenId];\n if (tokenBalance === undefined || 'error' in tokenBalance) {\n erc20TokenBalances[tokenId] = {\n error: `rpcService:getTokenWithBalance failed: ${tokenBalance?.error ?? 'unknown error'}`,\n };\n continue;\n }\n\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? undefined;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? undefined;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? undefined;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? undefined;\n\n const balance = new TokenUnit(tokenBalance.balance, tokenBalance.decimals, tokenBalance.symbol);\n const balanceInCurrency = priceInCurrency !== undefined ? balance.mul(priceInCurrency) : undefined;\n\n erc20TokenBalances[tokenBalance.address.toLowerCase()] = {\n ...tokenBalance,\n type: TokenType.ERC20,\n balance: tokenBalance.balance,\n balanceDisplayValue: balance.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n marketCap,\n change24,\n vol24,\n reputation: null,\n };\n }\n return erc20TokenBalances;\n }\n\n async listNftBalances(): Promise<Record<string, NftTokenWithBalance | Error>> {\n // the token list does not maintain a list of NFTs\n return {};\n }\n}\n","import { Environment } from '@avalabs/vm-module-types';\n\ntype Env = {\n glacierApiUrl: string;\n proxyApiUrl: string;\n};\n\nexport const prodEnv: Env = {\n glacierApiUrl: 'https://glacier-api.avax.network',\n proxyApiUrl: 'https://proxy-api.avax.network',\n};\n\nexport const devEnv: Env = {\n glacierApiUrl: 'https://glacier-api-dev.avax.network',\n proxyApiUrl: 'https://proxy-api-dev.avax.network',\n};\n\nexport const getEnv = (environment: Environment): Env => {\n switch (environment) {\n case Environment.PRODUCTION:\n return prodEnv;\n case Environment.DEV:\n return devEnv;\n }\n};\n","import {\n CurrencyCode,\n Erc1155Token,\n Erc1155TokenBalance,\n Erc20TokenBalance,\n Erc721Token,\n Erc721TokenBalance,\n Glacier,\n} from '@avalabs/glacier-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport {\n type Error,\n type ERC20Token,\n type NetworkTokenWithBalance,\n TokenType,\n type TokenWithBalanceERC20,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n} from '@avalabs/vm-module-types';\n\nimport type { BalanceServiceInterface } from '@src/handlers/get-balances/balance-service-interface';\nimport { DEFAULT_DECIMALS } from '@src/constants';\nimport { ChainId } from '@avalabs/core-chains-sdk';\nimport { getSmallImageForNFT } from '@src/utils/get-small-image-for-nft';\nimport { GlacierFetchHttpRequest } from '@internal/utils';\n\nclass GlacierUnhealthyError extends Error {\n override message = 'Glacier is unhealthy. Try again later.';\n}\n\nconst CHAINS_TO_FILTER = [ChainId.ETHEREUM_HOMESTEAD];\n\nexport class EvmGlacierService implements BalanceServiceInterface {\n glacierSdk: Glacier;\n isGlacierHealthy = true;\n supportedChainIds: string[] = [];\n\n constructor({ glacierApiUrl, headers }: { glacierApiUrl: string; headers?: Record<string, string> }) {\n this.glacierSdk = new Glacier({ BASE: glacierApiUrl, HEADERS: headers }, GlacierFetchHttpRequest);\n /**\n * This is for performance, basically we just cache the health of glacier every 5 seconds and\n * go off of that instead of every request\n */\n this.getSupportedChainIds().catch(() => {\n // Noop. It will be retried by .isSupportedNetwork calls upon unlocking if necessary.\n });\n }\n\n isHealthy = (): boolean => this.isGlacierHealthy;\n\n setGlacierToUnhealthy(): void {\n this.isGlacierHealthy = false;\n setTimeout(\n () => {\n this.isGlacierHealthy = true;\n },\n 5 * 60 * 1000,\n ); // 5 minutes\n }\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n const chainIds = await this.getSupportedChainIds();\n return chainIds.some((id) => id === chainId.toString());\n }\n\n async getSupportedChainIds(): Promise<string[]> {\n if (this.supportedChainIds.length) {\n return this.supportedChainIds;\n }\n\n try {\n const supportedChains = await this.glacierSdk.evmChains.supportedChains({});\n /*\n * https://ava-labs.atlassian.net/browse/CP-9855\n * We are removing the support for Ethereum chain, so it is queried from DeBank instead\n */\n this.supportedChainIds = supportedChains.chains\n .map((chain) => chain.chainId)\n .filter((chainId) => !CHAINS_TO_FILTER.includes(Number(chainId)));\n return this.supportedChainIds;\n } catch {\n return [];\n }\n }\n\n async reindexNft({\n address,\n chainId,\n tokenId,\n }: {\n address: string;\n chainId: string;\n tokenId: string;\n }): Promise<void> {\n try {\n await this.glacierSdk.nfTs.reindexNft({\n address,\n chainId,\n tokenId,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async getTokenDetails({\n address,\n chainId,\n tokenId,\n }: {\n address: string;\n chainId: string;\n tokenId: string;\n }): Promise<Erc721Token | Erc1155Token> {\n try {\n return this.glacierSdk.nfTs.getTokenDetails({\n address,\n chainId,\n tokenId,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n coingeckoId,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n coingeckoId?: string;\n }): Promise<NetworkTokenWithBalance> {\n try {\n const nativeBalance = await this.glacierSdk.evmBalances.getNativeBalance({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n });\n\n const nativeTokenBalance = nativeBalance.nativeTokenBalance;\n const balance = new TokenUnit(nativeTokenBalance.balance, nativeTokenBalance.decimals, nativeTokenBalance.symbol);\n const priceInCurrency = nativeTokenBalance.price?.value;\n const balanceInCurrency = priceInCurrency !== undefined ? balance.mul(priceInCurrency) : undefined;\n\n return {\n name: nativeTokenBalance.name,\n symbol: nativeTokenBalance.symbol,\n decimals: nativeTokenBalance.decimals,\n type: TokenType.NATIVE,\n logoUri: nativeTokenBalance.logoUri,\n balance: balance.toSubUnit(),\n balanceDisplayValue: balance.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n coingeckoId: coingeckoId ?? '',\n };\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listErc721Balances({ chainId, address }: { chainId: string; address: string }): Promise<Erc721TokenBalance[]> {\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n const tokens: Erc721TokenBalance[] = [];\n do {\n try {\n const response = await this.glacierSdk.evmBalances.listErc721Balances({\n chainId,\n address,\n pageSize: 100,\n pageToken: nextPageToken,\n });\n\n tokens.push(...response.erc721TokenBalances);\n\n nextPageToken = response.nextPageToken;\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n } while (nextPageToken);\n\n return tokens;\n }\n\n async listErc1155Balances({\n chainId,\n address,\n }: {\n chainId: string;\n address: string;\n }): Promise<Erc1155TokenBalance[]> {\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n const tokens: Erc1155TokenBalance[] = [];\n do {\n try {\n const response = await this.glacierSdk.evmBalances.listErc1155Balances({\n chainId,\n address,\n pageSize: 100,\n pageToken: nextPageToken,\n });\n\n tokens.push(...response.erc1155TokenBalances);\n\n nextPageToken = response.nextPageToken;\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n } while (nextPageToken);\n\n return tokens;\n }\n\n async listNftBalances({\n chainId,\n address,\n }: {\n chainId: number;\n address: string;\n }): Promise<Record<string, NftTokenWithBalance | Error>> {\n const balances = await Promise.allSettled([\n this.listErc721Balances({ chainId: chainId.toString(), address }),\n this.listErc1155Balances({ chainId: chainId.toString(), address }),\n ]);\n\n const entries = balances\n .filter(\n (\n tokenlist,\n ): tokenlist is PromiseFulfilledResult<Erc721TokenBalance[]> | PromiseFulfilledResult<Erc1155TokenBalance[]> =>\n tokenlist.status === 'fulfilled',\n )\n .flatMap((tokenlist) => {\n return (\n tokenlist.value\n // We filter out erc1155s with 0 balance, which Glacier returns for some reason\n .filter((token) => token.ercType === Erc721Token.ercType.ERC_721 || BigInt(token.balance) > 0n)\n .map((token) => {\n return [\n `${token.address}-${token.tokenId}`,\n {\n address: token.address,\n description: token.metadata.description ?? '',\n logoUri: token.metadata.imageUri ?? '',\n logoSmall: getSmallImageForNFT(token.metadata.imageUri ?? ''),\n name: token.metadata.name ?? '',\n symbol: token.metadata.symbol ?? '',\n tokenId: token.tokenId,\n tokenUri: token.tokenUri,\n chainId,\n // glacier does not provide the collection name information\n collectionName: 'Unknown',\n balance: token.ercType === Erc1155Token.ercType.ERC_1155 ? BigInt(token.balance) : 1,\n balanceDisplayValue: token.ercType === Erc1155Token.ercType.ERC_1155 ? token.balance : '1',\n type: token.ercType === Erc721Token.ercType.ERC_721 ? TokenType.ERC721 : TokenType.ERC1155,\n metadata: {\n indexStatus: token.metadata.indexStatus,\n description: token.metadata.description,\n lastUpdatedTimestamp: token.metadata.metadataLastUpdatedTimestamp,\n properties:\n token.ercType === Erc721Token.ercType.ERC_721\n ? token.metadata.attributes\n : token.metadata.properties,\n },\n },\n ];\n })\n );\n });\n\n return Object.fromEntries(entries);\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n customTokens,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n customTokens: ERC20Token[];\n }): Promise<Record<string, TokenWithBalanceEVM | Error>> {\n try {\n const tokensWithBalance: TokenWithBalanceERC20[] = [];\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n do {\n const response = await this.glacierSdk.evmBalances.listErc20Balances({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n pageSize: 100, // glacier has a cap on page size of 100\n pageToken: nextPageToken,\n });\n\n tokensWithBalance.push(\n ...convertErc20TokenWithBalanceToTokenWithBalance(response.erc20TokenBalances, Number(chainId)),\n );\n nextPageToken = response.nextPageToken;\n } while (nextPageToken);\n\n /**\n * Glacier doesnt return tokens without balances so we need to polyfill that list\n * from our own list of tokens. We just set the balance to 0, these zero balance\n * tokens are only used for swap, bridge and tx parsing.\n */\n return [\n ...convertErc20TokenToTokenWithBalance(customTokens),\n ...tokensWithBalance, // this needs to be second in the list so it overwrites its zero balance counterpart if there is one\n ].reduce(\n (acc, token) => {\n return { ...acc, [token.address.toLowerCase()]: token };\n },\n {} as Record<string, TokenWithBalanceEVM>,\n );\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listTransactions({\n chainId,\n address,\n pageToken,\n pageSize,\n }: {\n chainId: string;\n address: string;\n pageToken?: string;\n pageSize?: number;\n }) {\n try {\n return this.glacierSdk.evmTransactions.listTransactions({\n chainId,\n address,\n pageToken,\n pageSize,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n}\n\nconst convertErc20TokenToTokenWithBalance = (tokens: ERC20Token[]): TokenWithBalanceERC20[] => {\n return tokens.map((token) => {\n return {\n ...token,\n decimals: token.decimals ?? DEFAULT_DECIMALS,\n type: TokenType.ERC20,\n balance: 0n,\n balanceInCurrency: 0,\n balanceDisplayValue: '0',\n balanceCurrencyDisplayValue: '0',\n priceInCurrency: 0,\n marketCap: 0,\n change24: 0,\n vol24: 0,\n reputation: null,\n };\n });\n};\n\nconst convertErc20TokenWithBalanceToTokenWithBalance = (\n tokenBalances: Erc20TokenBalance[],\n chainId: number,\n): TokenWithBalanceERC20[] => {\n return tokenBalances.map((token: Erc20TokenBalance): TokenWithBalanceERC20 => {\n const balance = new TokenUnit(token.balance, token.decimals, token.symbol);\n const balanceDisplayValue = balance.toDisplay();\n const balanceCurrencyDisplayValue = token.balanceValue?.value.toString();\n const priceInCurrency = token.price?.value;\n const balanceInCurrency =\n priceInCurrency !== undefined\n ? balance.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n return {\n chainId,\n address: token.address,\n name: token.name,\n symbol: token.symbol,\n decimals: token.decimals,\n logoUri: token.logoUri,\n balance: balance.toSubUnit(),\n balanceCurrencyDisplayValue,\n balanceDisplayValue,\n balanceInCurrency,\n priceInCurrency,\n reputation: token.tokenReputation,\n type: TokenType.ERC20,\n };\n });\n};\n","export const DEFAULT_DECIMALS = 18;\n\nexport const BLOCKAID_API_KEY = 'DUMMY_API_KEY'; // since we're using our own proxy and api key is handled there, we can use a dummy key here\n","import { ipfsResolver } from '@avalabs/core-utils-sdk';\n\nexport const IPFS_URL = 'https://ipfs.io';\n\nexport function ipfsResolverWithFallback(sourceUrl: string | undefined, desiredGatewayPrefix: string = IPFS_URL) {\n if (!sourceUrl) {\n return '';\n }\n\n try {\n return ipfsResolver(sourceUrl, desiredGatewayPrefix);\n } catch {\n return sourceUrl;\n }\n}\n","import { ipfsResolverWithFallback } from './ipsf-resolver-with-fallback';\n\nconst COVALENT_IMG_SIZER = 'https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image';\n\n/**\n * Covalent has an on the fly image resizer, it resolves image urls then resizes the image.\n *\n * This allows us to request smaller images depending on the UI needs\n *\n * @param imgUrl the url of the image to convert to size\n * @param imageSize the resize dimension\n * @returns The url to the image which is sized at the time of request\n */\nexport function getSmallImageForNFT(imgUrl: string, imageSize: '256' | '512' | '1024' = '256') {\n const url = ipfsResolverWithFallback(imgUrl);\n return `${COVALENT_IMG_SIZER}/width=${imageSize},fit/${url}`;\n}\n","import {\n type SigningData,\n type Network,\n type ApprovalController,\n type DisplayData,\n type RpcRequest,\n RpcMethod,\n type Alert,\n AlertType,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { toUtf8String } from 'ethers';\nimport { beautifySimpleMessage, beautifyComplexMessage } from './utils/beautify-message/beautify-message';\nimport { parseRequestParams } from './schemas/parse-request-params/parse-request-params';\nimport { isTypedDataV1 } from './utils/typeguards';\nimport { isTypedDataValid } from './utils/is-typed-data-valid';\nimport { processJsonRpcSimulation } from '../../utils/process-transaction-simulation';\nimport { textItem } from '@internal/utils/src/utils/detail-item';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSign = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n blockaid: Blockaid;\n}) => {\n const result = parseRequestParams({ method: request.method, params: request.params });\n\n if (!result.success) {\n console.error('invalid params', result.error);\n\n return {\n success: false,\n error: rpcErrors.invalidParams({ message: 'Params are invalid', data: { cause: result.error } }),\n };\n }\n const { method, data, address } = result.data;\n\n // validate typed data\n let typedDataValidationResult: ReturnType<typeof isTypedDataValid> | undefined;\n\n if (method === RpcMethod.SIGN_TYPED_DATA_V3 || method === RpcMethod.SIGN_TYPED_DATA_V4) {\n typedDataValidationResult = isTypedDataValid(data);\n }\n\n // generate display data and signing data\n let signingData: SigningData | undefined;\n let messageDetails: string | undefined;\n let alert: Alert | undefined;\n\n if (typedDataValidationResult && !typedDataValidationResult.isValid) {\n alert = {\n type: AlertType.INFO,\n details: {\n title: 'Warning: Verify Message Content',\n description: 'This message contains non-standard elements. Please verify message content!',\n detailedDescription: (typedDataValidationResult.error as Error).toString(),\n },\n };\n }\n\n if (method === RpcMethod.ETH_SIGN) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = data;\n } else if (method === RpcMethod.PERSONAL_SIGN) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = toUtf8String(data);\n } else if (method === RpcMethod.SIGN_TYPED_DATA || method === RpcMethod.SIGN_TYPED_DATA_V1) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = isTypedDataV1(data) ? beautifySimpleMessage(data) : beautifyComplexMessage(data);\n } else if (method === RpcMethod.SIGN_TYPED_DATA_V3 || method === RpcMethod.SIGN_TYPED_DATA_V4) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n const { types, primaryType, ...messageToDisplay } = data;\n messageDetails = beautifyComplexMessage(messageToDisplay);\n }\n\n if (!signingData || !messageDetails) {\n return {\n success: false,\n error: rpcErrors.internal('Unable to generate signing data'),\n };\n }\n\n const simulationResult = await processJsonRpcSimulation({\n request,\n accountAddress: address,\n chainId: network.chainId,\n data: { method, params: request.params },\n dAppUrl: request.dappInfo.url,\n blockaid,\n });\n\n const displayData: DisplayData = {\n title: 'Sign Message',\n dAppInfo: {\n name: request.dappInfo.name,\n action: `${request.dappInfo.name} is requesting to sign the following message`,\n logoUri: request.dappInfo.icon,\n },\n network: {\n chainId: network.chainId,\n name: network.chainName,\n logoUri: network.logoUri,\n },\n account: address,\n details: [\n {\n title: 'Message',\n items: [textItem('Message', messageDetails, 'vertical')],\n },\n ],\n alert: simulationResult?.alert ?? alert,\n balanceChange: simulationResult?.balanceChange,\n tokenApprovals: simulationResult?.tokenApprovals,\n };\n\n // prompt user for approval\n const response = await approvalController.requestApproval({ request, displayData, signingData });\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n if (!('signedData' in response)) {\n return {\n error: rpcErrors.internal('No signed data returned'),\n };\n }\n\n return { result: response.signedData };\n};\n","/* eslint-disable @typescript-eslint/no-explicit-any */\nexport const beautifyComplexMessage = (data: { domain: Record<string, any>; message: Record<string, any> }) => {\n let result = '';\n\n result += 'Domain\\n';\n for (const key in data.domain) {\n result += ` ${key}: ${toUnicodeBold(String(data.domain[key]))}\\n`;\n }\n\n result += '\\nMessage\\n';\n for (const key in data.message) {\n if (typeof data.message[key] === 'object' && !Array.isArray(data.message[key])) {\n result += ` ${key}: \\n`;\n for (const subKey in data.message[key]) {\n if (Array.isArray(data.message[key][subKey])) {\n result += ` ${subKey}: \\n`;\n data.message[key][subKey].forEach((item: any, index: number) => {\n result += ` ${index}: ${toUnicodeBold(item)}\\n`;\n });\n } else {\n result += ` ${subKey}: ${toUnicodeBold(String(data.message[key][subKey]))}\\n`;\n }\n }\n } else if (Array.isArray(data.message[key])) {\n result += ` ${key}: \\n`;\n data.message[key].forEach((item: any, index: number) => {\n result += ` ${index}: \\n`;\n for (const subKey in item) {\n if (Array.isArray(item[subKey])) {\n result += ` ${subKey}: \\n`;\n item[subKey].forEach((item: any, index: number) => {\n result += ` ${index}: ${toUnicodeBold(item)}\\n`;\n });\n } else {\n result += ` ${subKey}: ${toUnicodeBold(String(item[subKey]))}\\n`;\n }\n }\n });\n } else {\n result += ` ${key}: ${toUnicodeBold(String(data.message[key]))}\\n`;\n }\n }\n\n return result;\n};\n\nexport const beautifySimpleMessage = (\n data: {\n name: string;\n type: string;\n value: any;\n }[],\n) => {\n let result = '';\n\n data.forEach((item) => {\n result += `${item.name}:\\n`;\n result += `${toUnicodeBold(String(item.value))}\\n\\n`;\n });\n\n return result;\n};\n\nconst toUnicodeBold = (str: string) => {\n const boldMap: Record<string, string> = {\n '0': '𝟬',\n '1': '𝟭',\n '2': '𝟮',\n '3': '𝟯',\n '4': '𝟰',\n '5': '𝟱',\n '6': '𝟲',\n '7': '𝟳',\n '8': '𝟴',\n '9': '𝟵',\n a: '𝗮',\n b: '𝗯',\n c: '𝗰',\n d: '𝗱',\n e: '𝗲',\n f: '𝗳',\n g: '𝗴',\n h: '𝗵',\n i: '𝗶',\n j: '𝗷',\n k: '𝗸',\n l: '𝗹',\n m: '𝗺',\n n: '𝗻',\n o: '𝗼',\n p: '𝗽',\n q: '𝗾',\n r: '𝗿',\n s: '𝘀',\n t: '𝘁',\n u: '𝘂',\n v: '𝘃',\n w: '𝘄',\n x: '𝘅',\n y: '𝘆',\n z: '𝘇',\n A: '𝗔',\n B: '𝗕',\n C: '𝗖',\n D: '𝗗',\n E: '𝗘',\n F: '𝗙',\n G: '𝗚',\n H: '𝗛',\n I: '𝗜',\n J: '𝗝',\n K: '𝗞',\n L: '𝗟',\n M: '𝗠',\n N: '𝗡',\n O: '𝗢',\n P: '𝗣',\n Q: '𝗤',\n R: '𝗥',\n S: '𝗦',\n T: '𝗧',\n U: '𝗨',\n V: '𝗩',\n W: '𝗪',\n X: '𝗫',\n Y: '𝗬',\n Z: '𝗭',\n };\n\n return str\n .split('')\n .map((char) => boldMap[char] || char)\n .join('');\n};\n","import { z } from 'zod';\nimport { ethSignSchema } from '../eth-sign';\nimport {\n combinedTypedDataSchema,\n ethSignTypedDataSchema,\n ethSignTypedDataV1Schema,\n ethSignTypedDataV3Schema,\n ethSignTypedDataV4Schema,\n typedDataSchema,\n} from '../eth-sign-typed-data';\nimport { personalSignSchema } from '../personal-sign';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\nconst paramsSchema = z\n .discriminatedUnion('method', [\n personalSignSchema,\n ethSignSchema,\n ethSignTypedDataSchema,\n ethSignTypedDataV1Schema,\n ethSignTypedDataV3Schema,\n ethSignTypedDataV4Schema,\n ])\n .transform((value, ctx) => {\n const { method, params } = value;\n\n switch (method) {\n case RpcMethod.PERSONAL_SIGN:\n return {\n data: params[0],\n address: params[1],\n method,\n };\n case RpcMethod.ETH_SIGN:\n return {\n data: params[1],\n address: params[0],\n method,\n };\n case RpcMethod.SIGN_TYPED_DATA:\n case RpcMethod.SIGN_TYPED_DATA_V1: {\n const address = params[0];\n const data = params[1];\n\n if (typeof data !== 'string') return { data, address, method };\n\n try {\n const parsed = JSON.parse(data);\n const result = combinedTypedDataSchema.parse(parsed);\n\n return {\n data: result,\n address,\n method,\n };\n } catch (e) {\n ctx.addIssue({\n code: z.ZodIssueCode.custom,\n message: 'param is not a valid json',\n });\n\n return z.NEVER;\n }\n }\n case RpcMethod.SIGN_TYPED_DATA_V3:\n case RpcMethod.SIGN_TYPED_DATA_V4: {\n const address = params[0];\n const data = params[1];\n\n if (typeof data !== 'string') return { data, address, method };\n\n try {\n const parsed = JSON.parse(data);\n const result = typedDataSchema.parse(parsed);\n\n return {\n data: result,\n address,\n method,\n };\n } catch (e) {\n ctx.addIssue({\n code: z.ZodIssueCode.custom,\n message: 'param is not a valid json',\n });\n\n return z.NEVER;\n }\n }\n }\n });\n\nexport function parseRequestParams(params: { method: RpcMethod; params: unknown }) {\n return paramsSchema.safeParse(params);\n}\n","import { z } from 'zod';\nimport { addressSchema, messageSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n// https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sign\nexport const ethSignSchema = z.object({\n method: z.literal(RpcMethod.ETH_SIGN),\n params: z.tuple([addressSchema, messageSchema]),\n});\n","import { z } from 'zod';\n\nexport const messageSchema = z.string().describe('message');\n\nexport const addressSchema = z.string().describe('address');\n","import { z } from 'zod';\nimport { addressSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n/**\n * For the different eth_signTypedData methods, the payload differs based on the version.\n *\n * V1 is based upon [an early version of EIP-712](https://github.com/ethereum/EIPs/pull/712/commits/21abe254fe0452d8583d5b132b1d7be87c0439ca)\n * that lacked some later security improvements, and should generally be neglected in favor of\n * later versions.\n *\n * V3 is based on [EIP-712](https://eips.ethereum.org/EIPS/eip-712), except that arrays and\n * recursive data structures are not supported.\n *\n * V4 is based on [EIP-712](https://eips.ethereum.org/EIPS/eip-712), and includes full support of\n * arrays and recursive data structures.\n *\n * References:\n * - https://eips.ethereum.org/EIPS/eip-712#specification-of-the-eth_signtypeddata-json-rpc\n * - https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4\n */\nconst messageTypeSchema = z.object({ name: z.string(), type: z.string() });\n\nexport const typedDataSchema = z.object({\n types: z.object({ EIP712Domain: z.array(messageTypeSchema) }).catchall(z.array(messageTypeSchema)),\n primaryType: z.string(),\n domain: z.record(z.any()),\n message: z.record(z.any()),\n});\n\nexport const typedDataV1Schema = z\n .array(\n z.object({\n type: z.string(),\n name: z.string(),\n value: z.union([z.string(), z.number(), z.boolean(), z.object({}).passthrough(), z.array(z.unknown()), z.null()]),\n }),\n )\n .nonempty();\n\nexport const combinedTypedDataSchema = typedDataSchema.or(typedDataV1Schema);\n\nconst dataSchema = z.union([z.string().describe('data string'), typedDataSchema]);\n\nexport const combinedDataSchema = z.union([z.string().describe('data string'), combinedTypedDataSchema]);\n\nexport const ethSignTypedDataSchema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA),\n params: z.tuple([addressSchema, combinedDataSchema]),\n});\n\nexport const ethSignTypedDataV1Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V1),\n params: z.tuple([addressSchema, combinedDataSchema]),\n});\n\nexport const ethSignTypedDataV3Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V3),\n params: z.tuple([addressSchema, dataSchema]),\n});\n\nexport const ethSignTypedDataV4Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V4),\n params: z.tuple([addressSchema, dataSchema]),\n});\n","import { z } from 'zod';\nimport { addressSchema, messageSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n// https://github.com/ethereum/go-ethereum/pull/2940\nexport const personalSignSchema = z.object({\n method: z.literal(RpcMethod.PERSONAL_SIGN),\n params: z.union([\n z.tuple([messageSchema, addressSchema]),\n z.tuple([messageSchema, addressSchema, z.string().optional().describe('password')]),\n ]),\n});\n","import type { TypedDataV1, TypedData, MessageTypes } from '@avalabs/vm-module-types';\nimport { typedDataSchema, typedDataV1Schema } from '../schemas/eth-sign-typed-data';\n\nexport const isTypedDataV1 = (data: unknown): data is TypedDataV1 => {\n return typedDataV1Schema.safeParse(data).success;\n};\n\nexport const isTypedData = (data: unknown): data is TypedData<MessageTypes> => {\n return typedDataSchema.safeParse(data).success;\n};\n","import { type TypedData, type MessageTypes } from '@avalabs/vm-module-types';\nimport { TypedDataEncoder } from 'ethers';\n\ntype Result = { isValid: true } | { isValid: false; error: unknown };\n\nexport const isTypedDataValid = (data: TypedData<MessageTypes>): Result => {\n try {\n // getPayload verifies the types and the content of the message throwing an error if the data is not valid.\n // We don't want to immediately reject the request even if there are errors for compatiblity reasons.\n // dApps tend to make small mistakes in the message format like leaving the verifyingContract emptry,\n // in which cases we should be able to continue just like other wallets do (even if it's technically incorrect).\n\n // remove EIP712Domain from types since ethers.js handles it separately\n const { EIP712Domain, ...types } = data.types;\n TypedDataEncoder.getPayload(data.domain, types, data.message);\n\n return {\n isValid: true,\n };\n } catch (e) {\n return {\n isValid: false,\n error: e,\n };\n }\n};\n","import type { Network, RpcRequest } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getProvider } from '../../utils/get-provider';\n\nexport const forwardToRpcNode = async (request: RpcRequest, network: Network) => {\n try {\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n const response = await provider.send(request.method, request.params as unknown[]);\n return { result: response };\n } catch (error) {\n // extracting the error message based on the error object structure from ethers lib\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const message = (error as any).info?.error?.message || (error as any).error?.message || (error as Error).message;\n return { error: rpcErrors.internal(message) };\n }\n};\n","import { NetworkVMType, WalletType, type GetAddressParams, type GetAddressResponse } from '@avalabs/vm-module-types';\nimport { getAddressFromXPub, getEvmAddressFromPubKey } from '@avalabs/core-wallets-sdk';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\ntype GetAddress = Omit<GetAddressParams, 'isTestnet' | 'xpubXP'>;\n\nexport const getAddress = async ({ accountIndex, xpub, walletType }: GetAddress): Promise<GetAddressResponse> => {\n switch (walletType) {\n case WalletType.Mnemonic:\n case WalletType.Ledger:\n case WalletType.Keystone: {\n return {\n [NetworkVMType.EVM]: getAddressFromXPub(xpub, accountIndex),\n };\n }\n case WalletType.LedgerLive:\n case WalletType.Seedless: {\n const pubKeyBuffer = Buffer.from(xpub, 'hex');\n return {\n [NetworkVMType.EVM]: getEvmAddressFromPubKey(pubKeyBuffer),\n };\n }\n default:\n throw rpcErrors.invalidParams(`Unsupported wallet type: ${walletType}`);\n }\n};\n","import {\n NetworkVMType,\n type ApprovalController,\n type DeriveAddressParams,\n type DeriveAddressResponse,\n} from '@avalabs/vm-module-types';\nimport { computeAddress } from 'ethers';\n\nimport { hasDerivationDetails } from '@internal/utils/src/utils/address-derivation';\nimport { buildDerivationPath } from '../build-derivation-path/build-derivation-path';\n\nexport const deriveAddress = async (\n params: DeriveAddressParams & { approvalController: ApprovalController },\n): Promise<DeriveAddressResponse> => {\n const { secretId, approvalController } = params;\n\n // When dealing with single-account private keys, we don't need the derivation path any more.\n const derivationPath = hasDerivationDetails(params) ? buildDerivationPath(params).EVM : undefined;\n const publicKeyHex = await approvalController.requestPublicKey({\n curve: 'secp256k1',\n secretId,\n derivationPath,\n });\n\n return {\n [NetworkVMType.EVM]: computeAddress(`0x${publicKeyHex}`), // ApprovalController does not return the 0x prefix\n };\n};\n","import type { DeriveAddressParams, DetailedDeriveAddressParams } from '@avalabs/vm-module-types';\n\nexport const hasDerivationDetails = (params: DeriveAddressParams): params is DetailedDeriveAddressParams =>\n 'derivationPathType' in params &&\n 'accountIndex' in params &&\n typeof params.accountIndex === 'number' &&\n typeof params.derivationPathType === 'string';\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport {\n NetworkVMType,\n type BuildDerivationPathParams,\n type BuildDerivationPathResponse,\n} from '@avalabs/vm-module-types';\n\nexport const buildDerivationPath = ({\n accountIndex,\n derivationPathType,\n}: BuildDerivationPathParams): Pick<BuildDerivationPathResponse, NetworkVMType.EVM> => {\n if (accountIndex < 0) {\n throw rpcErrors.invalidParams('Account index must be a non-negative integer');\n }\n\n switch (derivationPathType) {\n case 'bip44':\n return {\n [NetworkVMType.EVM]: `m/44'/60'/0'/0/${accountIndex}`,\n };\n\n case 'ledger_live':\n return {\n [NetworkVMType.EVM]: `m/44'/60'/${accountIndex}'/0/0`,\n };\n\n default:\n throw rpcErrors.invalidParams(`Unsupported derivation path type: ${derivationPathType}`);\n }\n};\n","import { CurrencyCode, NftTokenMetadataStatus } from '@avalabs/glacier-sdk';\nimport type { BalanceServiceInterface, TokenId } from '@src/handlers/get-balances/balance-service-interface';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { isHexString } from 'ethers';\nimport {\n type Error,\n type NetworkTokenWithBalance,\n type NftTokenWithBalance,\n TokenType,\n type TokenWithBalanceEVM,\n} from '@avalabs/vm-module-types';\nimport { DeBank, type DeBankChainInfo, type DeBankNftToken } from './de-bank';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getExchangeRates } from '@internal/utils';\nimport { getSmallImageForNFT } from '@src/utils/get-small-image-for-nft';\n\nexport class DeBankService implements BalanceServiceInterface {\n #deBank: DeBank;\n\n constructor({ proxyApiUrl }: { proxyApiUrl: string }) {\n this.#deBank = new DeBank(`${proxyApiUrl}/proxy/debank`);\n }\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n return this.#deBank.isNetworkSupported(chainId);\n }\n\n async #getChainInfo(chainId: number): Promise<{ chainInfo: DeBankChainInfo; chainIdString: string }> {\n const chainList = await this.#deBank.getChainList();\n const chainIdString = chainList.find((value) => value.community_id === chainId)?.id;\n if (!chainIdString) {\n throw rpcErrors.invalidParams('getNativeBalance: not valid chainId: ' + chainId);\n }\n\n const chainInfo = await this.#deBank.getChainInfo({ chainId: chainIdString });\n\n if (!chainInfo) {\n throw rpcErrors.invalidParams('getNativeBalance: not valid chainId: ' + chainId);\n }\n\n return {\n chainInfo,\n chainIdString,\n };\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n }): Promise<NetworkTokenWithBalance> {\n if (!isHexString(address)) throw rpcErrors.invalidParams('getNativeBalance: not valid address: ' + address);\n const { chainInfo, chainIdString } = await this.#getChainInfo(chainId);\n\n const tokenId = chainInfo.native_token_id;\n const nativeTokenBalance = await this.#deBank.getTokenBalance({ address, chainId: chainIdString, tokenId });\n const tokenUnit = new TokenUnit(\n nativeTokenBalance.raw_amount,\n nativeTokenBalance.decimals,\n nativeTokenBalance.symbol,\n );\n const balanceDisplayValue = tokenUnit.toDisplay();\n const exchangeRates = await getExchangeRates();\n const usdToCurrencyRate = exchangeRates.usd[currency.toLowerCase()];\n const priceInCurrency = usdToCurrencyRate ? usdToCurrencyRate * nativeTokenBalance.price : undefined;\n const balanceCurrencyDisplayValue = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2 })\n : undefined;\n const balanceInCurrency = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n return {\n name: nativeTokenBalance.name,\n symbol: nativeTokenBalance.symbol,\n decimals: nativeTokenBalance.decimals,\n type: TokenType.NATIVE,\n logoUri: nativeTokenBalance.logo_url,\n balance: tokenUnit.toSubUnit(),\n balanceDisplayValue,\n balanceInCurrency,\n balanceCurrencyDisplayValue,\n priceInCurrency,\n } as NetworkTokenWithBalance;\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n pageSize: number;\n pageToken?: string;\n }): Promise<Record<TokenId, TokenWithBalanceEVM | Error>> {\n if (!isHexString(address)) throw rpcErrors.invalidParams('listErc20Balances: not valid address');\n const { chainInfo, chainIdString } = await this.#getChainInfo(chainId);\n\n const tokenBalances = await this.#deBank.getTokensBalanceOnChain({ chainId: chainIdString, address });\n const exchangeRates = await getExchangeRates();\n\n const erc20TokenBalances: Record<TokenId, TokenWithBalanceEVM | Error> = {};\n for (const tokenBalance of tokenBalances) {\n // skip native token or tokens which are not core tokens\n if (tokenBalance.id === chainInfo.native_token_id || tokenBalance.is_core === false) {\n continue;\n }\n\n const tokenUnit = new TokenUnit(tokenBalance.raw_amount, tokenBalance.decimals, tokenBalance.symbol);\n const balanceDisplayValue = tokenUnit.toDisplay();\n const usdToCurrencyRate = exchangeRates.usd[currency.toLowerCase()];\n const priceInCurrency = usdToCurrencyRate ? usdToCurrencyRate * tokenBalance.price : undefined;\n const balanceCurrencyDisplayValue = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2 })\n : undefined;\n const balanceInCurrency = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n erc20TokenBalances[tokenBalance.id] = {\n chainId: chainInfo.community_id,\n address: tokenBalance.id,\n name: tokenBalance.name,\n symbol: tokenBalance.symbol,\n decimals: tokenBalance.decimals,\n logoUri: tokenBalance.logo_url,\n balance: tokenUnit.toSubUnit(),\n balanceCurrencyDisplayValue,\n balanceDisplayValue,\n balanceInCurrency,\n priceInCurrency,\n type: TokenType.ERC20,\n reputation: null,\n };\n }\n\n return erc20TokenBalances;\n }\n\n #mapNftList(deBankNftList: DeBankNftToken[], chainId: number): Record<string, NftTokenWithBalance | Error> {\n return deBankNftList.reduce(\n (accumulator, token) => ({\n ...accumulator,\n [`${token.contract_id}-${token.id}`]: {\n chainId,\n address: token.contract_id,\n description: token.description ?? '',\n logoUri: token.thumbnail_url,\n logoSmall: getSmallImageForNFT(token.content),\n name: token.name,\n symbol: '',\n tokenId: `${token.inner_id}`,\n tokenUri: token.detail_url,\n collectionName: token.collection_name,\n balance: BigInt(token.amount),\n balanceDisplayValue: `${token.amount}`,\n type: token.is_erc721 ? TokenType.ERC721 : TokenType.ERC1155,\n metadata: {\n indexStatus: NftTokenMetadataStatus.UNINDEXED,\n description: token.description ?? '',\n properties: '',\n },\n },\n }),\n {} as Record<string, NftTokenWithBalance>,\n );\n }\n\n async listNftBalances({\n chainId,\n address,\n }: {\n chainId: number;\n address: string;\n }): Promise<Record<string, NftTokenWithBalance | Error>> {\n if (!isHexString(address)) {\n throw rpcErrors.invalidParams('listNftBalances: not valid address');\n }\n const { chainIdString } = await this.#getChainInfo(chainId);\n\n const nftList = await this.#deBank.getNftList({ chainId: chainIdString, address });\n return this.#mapNftList(nftList, chainId);\n }\n}\n","import type { Hex } from '@avalabs/vm-module-types';\nimport { isHexString } from 'ethers';\nimport { fetchAndVerify } from '@internal/utils';\nimport { z } from 'zod';\n\nexport class DeBank {\n #supportedChainIds: DeBankChainInfo[] = [];\n\n constructor(private baseUrl: string) {}\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n const chainList: DeBankChainInfo[] = await this.getChainList();\n const chainIds = chainList.map((value) => value.community_id);\n return chainIds.some((id) => id === chainId);\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n */\n async getChainInfo({ chainId }: { chainId: string }): Promise<DeBankChainInfo | undefined> {\n const chainList = await this.getChainList();\n return chainList.find((chain) => chain.id === chainId);\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n * @param tokenId - The address of the token contract or a native token id (eth, matic, bsc)\n */\n async getTokenBalance({\n chainId,\n address,\n tokenId,\n }: {\n chainId: string;\n address: Hex;\n tokenId: Hex | string;\n }): Promise<DeBankToken> {\n const tokenBalanceResponse = await fetch(\n `${this.baseUrl}/v1/user/token?id=${address}&chain_id=${chainId}&token_id=${tokenId}`,\n );\n if (tokenBalanceResponse.ok) {\n return await tokenBalanceResponse.json();\n } else {\n throw new Error(`${tokenBalanceResponse.status}:${tokenBalanceResponse.statusText}`);\n }\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getTokensBalanceOnChain({ chainId, address }: { chainId: string; address: Hex }): Promise<DeBankToken[]> {\n const tokenBalanceResponse = await fetch(`${this.baseUrl}/v1/user/token_list?id=${address}&chain_id=${chainId}`);\n if (tokenBalanceResponse.ok) {\n return await tokenBalanceResponse.json();\n } else {\n throw new Error(`${tokenBalanceResponse.status}:${tokenBalanceResponse.statusText}`);\n }\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getTokenList({ chainId, address }: { chainId: string; address: Hex }): Promise<DeBankToken[]> {\n const response = await fetch(`${this.baseUrl}/v1/user/token_list?id=${address}&chain_id=${chainId}`);\n if (response.ok) {\n return await response.json();\n } else {\n throw new Error(`${response.status}:${response.statusText}`);\n }\n }\n\n /**\n * @returns Cached chain list if there is any, or tries to fetch it from API\n */\n async getChainList(): Promise<DeBankChainInfo[]> {\n if (this.#supportedChainIds.length === 0) {\n const chainListResponse = await fetch(`${this.baseUrl}/v1/chain/list`);\n if (chainListResponse.ok) {\n this.#supportedChainIds = await chainListResponse.json();\n } else {\n throw new Error(`${chainListResponse.status}:${chainListResponse.statusText}`);\n }\n }\n return this.#supportedChainIds;\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getNftList({ address, chainId }: { chainId: string; address: Hex }): Promise<DeBankNftToken[]> {\n return fetchAndVerify<typeof DeBankNftTokenListSchema>(\n [`${this.baseUrl}/v1/user/nft_list?id=${address}&chain_id=${chainId}`],\n DeBankNftTokenListSchema,\n );\n }\n}\n\nconst BaseDeBankTokenSchema = z.object({\n id: z.union([z.custom<string>(isHexString), z.string()]), //address or native token id (eth, matic, bsc)\n chain: z.string(),\n name: z.string(),\n});\n\ntype BaseDeBankToken = z.infer<typeof BaseDeBankTokenSchema>;\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"0x19225f002b65eefb22950b9739fc8b448d900d44\",\n * \"chain\": \"base\",\n * \"name\": \"BASE\",\n * \"symbol\": \"Collect on: swap-based.com\",\n * \"display_symbol\": null,\n * \"optimized_symbol\": \"Collect on: swap-based.com\",\n * \"decimals\": 0,\n * \"logo_url\": null,\n * \"protocol_id\": \"\",\n * \"price\": 0,\n * \"price_24h_change\": null,\n * \"is_verified\": false,\n * \"is_core\": false,\n * \"is_wallet\": false,\n * \"time_at\": 1712662101,\n * \"amount\": 250000,\n * \"raw_amount\": 250000,\n * \"raw_amount_hex_str\": \"0x3d090\"\n * }\n * ```\n */\n\n// TODO: create zod schema\nexport type DeBankToken = BaseDeBankToken & {\n symbol: string;\n optimized_symbol: string;\n decimals: number;\n logo_url: string;\n protocol_id: string;\n price: number;\n is_core: boolean;\n is_wallet: boolean;\n time_at: number;\n amount: bigint;\n raw_amount: bigint;\n};\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"eth\",\n * \"community_id\": 1,\n * \"name\": \"Ethereum\",\n * \"native_token_id\": \"eth\",\n * \"logo_url\": \"https://static.debank.com/image/chain/logo_url/eth/42ba589cd077e7bdd97db6480b0ff61d.png\",\n * \"wrapped_token_id\": \"0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2\",\n * \"is_support_pre_exec\": true\n * }\n * ```\n */\n// TODO: create zod schema\nexport type DeBankChainInfo = {\n id: string;\n community_id: number;\n name: string;\n native_token_id: string;\n logo_url: string;\n wrapped_token_id: Hex;\n is_support_pre_exec: boolean;\n};\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"defc948fbe6d3b138b49bf981e276f0b\",\n * \"contract_id\": \"0x495f947276749ce646f68ac8c248420045cb7b5e\",\n * \"inner_id\": \"55575360221028374465659771733000318579577403829328624053715758637886677712897\",\n * \"chain\": \"eth\",\n * \"name\": \"A New Era has begun\",\n * \"description\": \"3 of 9\\n\\nOn February 8, 2021, one of the most influential men in the world decided to invest in Bitcoin. Elon Musk, owner of Tesl\",\n * \"content_type\": \"image_url\",\n * \"content\": \"https://lh3.googleusercontent.com/WQnK8JxSSPj5YIxegh9iaprMaMmv-JswrcnTp9Mi5PXKDWmigkOzTBBIAkhdXtLPe7EwIe6Q1gi2gdtLzV08d2y67rMVTHx0Ei0S\",\n * \"detail_url\": \"https://opensea.io/assets/0x495f947276749ce646f68ac8c248420045cb7b5e/55575360221028374465659771733000318579577403829328624053715758637886677712897\",\n * \"contract_name\": \"OpenSea Shared Storefront\",\n * \"is_erc1155\": true,\n * \"amount\": 1,\n * \"protocol\": {\n * \"id\": \"opensea\",\n * \"chain\": \"eth\",\n * \"name\": \"OpenSea\",\n * \"site_url\": \"https://opensea.io\",\n * \"logo_url\": \"https://static.debank.com/image/project/logo_url/opensea/4b23246fac2d4ce53bd8e8079844821c.png\",\n * \"has_supported_portfolio\": false,\n * \"tvl\": 114295.77061458935\n * },\n * \"pay_token\": {\n * \"id\": \"eth\",\n * \"chain\": \"eth\",\n * \"name\": \"ETH\",\n * \"symbol\": \"ETH\",\n * \"display_symbol\": null,\n * \"optimized_symbol\": \"ETH\",\n * \"decimals\": 18,\n * \"logo_url\": \"https://static.debank.com/image/token/logo_url/eth/935ae4e4d1d12d59a99717a24f2540b5.png\",\n * \"protocol_id\": \"\",\n * \"price\": 2510.46,\n * \"is_verified\": true,\n * \"is_core\": true,\n * \"is_wallet\": true,\n * \"time_at\": 1628248886,\n * \"amount\": 0.0178,\n * \"date_at\": \"2021-08-06\"\n * },\n * \"attributes\": [\n * {\n * \"trait_type\": \"Artist\",\n * \"value\": \"SpaceTurtleShip\"\n * },\n * {\n * \"trait_type\": \"Edition\",\n * \"value\": \"1\"\n * }\n * ],\n * \"usd_price\": 51.492552,\n * \"collection_id\": null\n * }\n * ```\n * */\n\nconst DeBankNftTokenSchema = BaseDeBankTokenSchema.merge(\n z.object({\n contract_id: z.string(),\n description: z.string().nullable(),\n content_type: z.string().nullable(),\n content: z.string(),\n thumbnail_url: z.string(),\n total_supply: z.number(),\n detail_url: z.string(),\n collection_id: z.string(),\n is_core: z.boolean(),\n inner_id: z.string(),\n collection_name: z.string(),\n contract_name: z.string(),\n amount: z.number(),\n usd_price: z.number().optional(),\n is_erc721: z.boolean().optional(),\n is_erc1155: z.boolean().optional(),\n }),\n);\n\nconst DeBankNftTokenListSchema = z.array(DeBankNftTokenSchema);\n\nexport type DeBankNftToken = z.infer<typeof DeBankNftTokenSchema>;\n","import {\n type Network,\n type RpcRequest,\n type DisplayData,\n type BatchApprovalController,\n type Hex,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nimport { getProvider } from '../../utils/get-provider';\n\nimport { parseRequestParams } from './schema';\nimport { ensureProperNonces } from './utils/ensure-proper-nonces';\nimport { buildTxApprovalRequest } from '../../utils/build-tx-approval-request';\nimport { getTxHash } from '../../utils/get-tx-hash';\nimport { waitForTransactionReceipt } from '../../utils/wait-for-transaction-receipt';\nimport { getTxBatchUpdater } from '../../utils/evm-tx-batch-updater';\nimport { simulateTransactionBatch } from './utils/process-transaction-batch-simulation';\nimport { addressItem, linkItem, networkItem } from '@internal/utils/src/utils/detail-item';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSendTransactionBatch = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: BatchApprovalController;\n blockaid: Blockaid;\n}) => {\n const { params } = request;\n const { data: parsedParams, success, error } = parseRequestParams(params);\n\n if (!success) {\n console.error('invalid params', error);\n return {\n error: rpcErrors.invalidParams({ message: 'Transaction params are invalid', data: { cause: error } }),\n };\n }\n\n const { transactions: transactionRequests, options = {} } = parsedParams;\n const { onlyWaitForLastTx = false } = options;\n\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n // Ensure all transactions in the batch have the nonce set\n try {\n await ensureProperNonces(transactionRequests, provider);\n } catch (error) {\n console.error('Unable to calculate nonce', error);\n return {\n error: rpcErrors.internal({\n message: 'Unable to calculate nonce',\n data: {\n originalError: error,\n },\n }),\n };\n }\n\n const allTransactionPayloadsHaveGas = transactionRequests.every((tx) => Number(tx.gas) > 0);\n const { alert, balanceChange, isSimulationSuccessful, tokenApprovals, scans } = await simulateTransactionBatch({\n rpcMethod: request.method,\n chainId: network.chainId,\n params: transactionRequests,\n dAppUrl: request.dappInfo.url,\n provider,\n populateMissingGas: !allTransactionPayloadsHaveGas,\n blockaid,\n });\n const allTransactionsHaveGas = scans.every(({ transaction }) => Number(transaction.gas) > 0);\n\n if (!allTransactionsHaveGas) {\n console.error('Gas limit is missing in some transactions');\n return {\n error: rpcErrors.internal({\n message: 'Gas limit is missing in some transactions',\n }),\n };\n }\n\n const displayData: DisplayData = {\n title: 'Do you approve these transactions?',\n details: [\n {\n title: 'Transaction Details',\n items: [\n addressItem('Account', transactionRequests[0].from),\n networkItem('Network', {\n name: network.chainName,\n logoUri: network.logoUri,\n }),\n linkItem('Website', request.dappInfo),\n ],\n },\n ],\n isSimulationSuccessful,\n balanceChange,\n alert,\n tokenApprovals,\n networkFeeSelector: true,\n };\n\n const signingRequests = scans.map((scan) => buildTxApprovalRequest(request, network, scan.transaction, scan));\n const { cleanup, updateTx } = getTxBatchUpdater(request.requestId, signingRequests, displayData);\n\n const response = await approvalController.requestBatchApproval({ request, signingRequests, displayData, updateTx });\n\n cleanup();\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n if (response.result.length !== transactionRequests.length) {\n return {\n error: rpcErrors.internal({\n message: `Invalid number of signatures. Expected ${transactionRequests.length}, got ${response.result.length}`,\n }),\n };\n }\n\n const txHashes: Hex[] = [];\n\n for (const [index, result] of response.result.entries()) {\n const isLast = index === response.result.length - 1;\n // Execute transactions one-by-one, as they may depend on one another\n const txHash = await getTxHash(provider, result);\n\n const receiptPromise = waitForTransactionReceipt({\n explorerUrl: network.explorerUrl ?? '',\n provider,\n txHash,\n onTransactionPending: approvalController.onTransactionPending,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n request,\n });\n\n // When onlyWaitForLastTx is true:\n // - Broadcast all transactions immediately without waiting for intermediate receipts\n // - This is useful when the ApprovalController handles sequential broadcasting\n // and we want to return tx hashes quickly\n //\n // When onlyWaitForLastTx is false (default):\n // - Wait for each transaction's receipt before broadcasting the next\n // - Skip waiting for the last transaction since there are no more transactions\n // that would rely on it\n const shouldWaitForReceipt = onlyWaitForLastTx ? false : !isLast;\n\n if (shouldWaitForReceipt) {\n const isSuccess = await receiptPromise;\n\n if (!isSuccess) {\n return {\n error: rpcErrors.internal({\n message: `Transaction ${index + 1} failed! Batch execution stopped`,\n }),\n };\n }\n }\n\n txHashes.push(txHash);\n }\n\n return {\n result: txHashes,\n };\n};\n","import { z } from 'zod';\nimport { transactionSchema } from '../../utils/transaction-schema';\n\nexport const transactionArraySchema = z\n .tuple([transactionSchema, transactionSchema])\n .rest(transactionSchema)\n .refine((transactions) => areAllEqualByProp(transactions, 'chainId'), {\n message: 'All transactions must use the same \"chainId\"',\n })\n .refine((transactions) => areAllEqualByProp(transactions, 'from'), {\n message: 'All transactions must use the same \"from\" address',\n })\n .refine((transactions) => areAllEmptyByProp(transactions, 'nonce') || areAllDifferentByProp(transactions, 'nonce'), {\n message: `Each transaction needs a different \"nonce\". Make them different, or leave them empty for all transactions.`,\n });\n\nexport const batchOptionsSchema = z.object({\n /**\n * When true, broadcasts all transactions immediately without waiting for\n * intermediate receipts. Only the last transaction's receipt will be awaited.\n *\n * This is useful when the ApprovalController handles the sequential broadcasting\n * and we want the vm-module to just return the transaction hashes quickly.\n *\n * Default: false (wait for each transaction's receipt before broadcasting the next)\n */\n onlyWaitForLastTx: z.boolean().optional(),\n});\n\nexport type BatchOptions = z.infer<typeof batchOptionsSchema>;\n\n/**\n * Support both legacy array format and new object format with options:\n * - Legacy: [tx1, tx2, ...]\n * - New: { transactions: [tx1, tx2, ...], options?: { onlyWaitForLastTx?: boolean } }\n */\nexport const transactionBatchSchema = z.union([\n transactionArraySchema,\n z.object({\n transactions: transactionArraySchema,\n options: batchOptionsSchema.optional(),\n }),\n]);\n\nfunction areAllDifferentByProp<T>(elements: T[], prop: keyof T) {\n return !areAllEqualByProp(elements, prop);\n}\n\nfunction areAllEmptyByProp<T>(elements: T[], prop: keyof T) {\n return elements.every((el) => el[prop] === undefined || el[prop] === '');\n}\n\nfunction areAllEqualByProp<T>(elements: T[], prop: keyof T) {\n const uniqValues = new Set(elements.map((el) => el[prop]));\n\n return uniqValues.size === 1;\n}\n\n/** Input type for batch transaction params - either array or object format */\nexport type TransactionBatchInput = z.input<typeof transactionBatchSchema>;\n\n/** Normalized output after parsing - always has transactions, options is optional */\nexport type ParsedParams = {\n transactions: z.infer<typeof transactionArraySchema>;\n options?: BatchOptions;\n};\n\nexport const parseRequestParams = (params: unknown): z.SafeParseReturnType<TransactionBatchInput, ParsedParams> => {\n const result = transactionBatchSchema.safeParse(params);\n\n if (!result.success) {\n return result;\n }\n\n // Normalize to the new format\n if (Array.isArray(result.data)) {\n return {\n success: true,\n data: {\n transactions: result.data,\n options: {},\n },\n };\n }\n\n return {\n success: true,\n data: {\n transactions: result.data.transactions,\n options: result.data.options ?? {},\n },\n };\n};\n","import type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport { getNonce } from '../../../utils/get-nonce';\nimport type { TransactionBatchParams } from '../../../types';\n\nexport const ensureProperNonces = async (transactions: TransactionBatchParams, provider: JsonRpcBatchInternal) => {\n // The parameters schema ensures all transactions have empty or different nonces,\n // so if we find the first one is populated, we know the rest should be both populated & unique.\n if (transactions[0].nonce) {\n return;\n }\n const nonce = await getNonce({\n from: transactions[0].from,\n provider,\n });\n\n return Promise.all(\n transactions.map(async (transaction, index) => {\n transaction.nonce = String(nonce + index);\n }),\n );\n};\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport type { DisplayData, EvmTxBatchUpdateFn, SigningData_EthSendTx } from '@avalabs/vm-module-types';\n\ntype SigningRequests = {\n displayData: DisplayData;\n signingData: SigningData_EthSendTx;\n}[];\n\nconst requests = new Map<\n string,\n {\n signingRequests: SigningRequests;\n displayData: DisplayData;\n }\n>();\n\nexport const getTxBatchUpdater = (\n requestId: string,\n signingRequests: SigningRequests,\n displayData: DisplayData,\n): { updateTx: EvmTxBatchUpdateFn; cleanup: () => void } => {\n requests.set(requestId, { signingRequests, displayData });\n\n return {\n updateTx: ({ maxFeeRate, maxTipRate }, txIndex) => {\n const request = requests.get(requestId);\n\n if (!request) {\n throw rpcErrors.resourceNotFound();\n }\n\n const { signingRequests } = request;\n\n const newSigningRequests = signingRequests.map((signingRequest, index) => {\n if (txIndex === undefined || txIndex === index) {\n const { signingData } = signingRequest;\n const newSigningData = {\n ...signingData,\n data: {\n ...signingData.data,\n maxFeePerGas: maxFeeRate ?? signingData.data.maxFeePerGas,\n maxPriorityFeePerGas: maxTipRate ?? signingData.data.maxPriorityFeePerGas,\n },\n };\n\n return { signingData: newSigningData, displayData: signingRequest.displayData };\n }\n\n return signingRequest;\n });\n\n const updatedRequest = { signingRequests: newSigningRequests, displayData };\n\n requests.set(requestId, updatedRequest);\n return updatedRequest;\n },\n cleanup: () => requests.delete(requestId),\n };\n};\n","import {\n AlertType,\n RpcMethod,\n type Alert,\n type BalanceChange,\n type TokenApprovals,\n type TokenDiff,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport type { TransactionBatchParams, TransactionParams } from '../../../types';\nimport { isEmpty, isNetworkToken } from '../../../utils/type-utils';\nimport { processTransactionSimulation } from '../../../utils/process-transaction-simulation';\nimport { scanTransactionBatch } from '../../../utils/scan-transaction';\nimport { transactionAlerts } from '../../../utils/transaction-alerts';\nimport type Blockaid from '@blockaid/client';\n\nexport const simulateTransactionBatch = async ({\n rpcMethod,\n dAppUrl,\n params,\n chainId,\n provider,\n populateMissingGas,\n blockaid,\n}: {\n rpcMethod: RpcMethod;\n dAppUrl?: string;\n params: TransactionBatchParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n populateMissingGas?: boolean;\n blockaid: Blockaid;\n}): Promise<\n TransactionSimulationResult & { scans: (TransactionSimulationResult & { transaction: TransactionParams })[] }\n> => {\n let simulationResults;\n\n try {\n simulationResults = await scanTransactionBatch({\n chainId,\n params,\n domain: dAppUrl,\n withGasEstimation: populateMissingGas,\n blockaid,\n });\n } catch (error) {\n console.error('simulateTransactionBatch error', error);\n return {\n isSimulationSuccessful: false,\n alert: transactionAlerts[AlertType.WARNING],\n scans: [],\n };\n }\n\n const scans = await Promise.all(\n simulationResults.map(async (simulationResult, index) => {\n const matchingTx = params[index]!;\n const processedResult = await processTransactionSimulation({\n chainId,\n params: params[index]!,\n provider,\n rpcMethod,\n simulationResult,\n });\n\n if (populateMissingGas && !matchingTx.gas && processedResult.estimatedGasLimit) {\n matchingTx.gas = `0x${processedResult.estimatedGasLimit.toString(16)}`;\n }\n\n return {\n transaction: matchingTx,\n ...processedResult,\n };\n }),\n );\n\n // Make sure `isSimulatioSuccessful` is only true for a batch if all nested transactions were simulated successfully\n const isSimulationSuccessful = scans.every((scan) => scan.isSimulationSuccessful);\n // Make sure the alert in the aggregated view is of the upmost priority (DANGER -> WARNING -> INFO)\n const alert = getHighestAlert(scans);\n const balanceChange = aggregateBalanceChange(scans);\n const tokenApprovals = aggregateTokenApprovals(scans);\n\n return {\n isSimulationSuccessful,\n alert,\n balanceChange,\n tokenApprovals,\n // For now, we're disabling the editing of approvals in the aggregated view since\n // it could disrupt the execution of the entire batch.\n scans: scans.map((scan) => {\n if (!scan.tokenApprovals) {\n return scan;\n }\n\n scan.tokenApprovals.isEditable = false;\n return scan;\n }),\n };\n};\n\nconst getHighestAlert = (scans: { alert?: Alert }[]): Alert | undefined => {\n const alertPrio = {\n [AlertType.INFO]: 0,\n [AlertType.WARNING]: 1,\n [AlertType.DANGER]: 2,\n };\n\n return scans.reduce(\n (highest, { alert }) => {\n if (!alert) return highest;\n if (!highest || alertPrio[alert.type] > alertPrio[highest.type]) {\n return alert;\n }\n return highest;\n },\n undefined as Alert | undefined,\n );\n};\n\nconst aggregateTokenApprovals = (scans: TransactionSimulationResult[]): TokenApprovals | undefined => {\n const tokenApprovals = scans.reduce(\n (aggregatedTokenApprovals, { tokenApprovals }) => {\n if (!tokenApprovals) {\n return aggregatedTokenApprovals;\n }\n\n tokenApprovals.approvals.forEach((appr) => {\n const aggregatedApproval = aggregatedTokenApprovals.approvals.find(\n (aggrApproval) =>\n appr.token.address === aggrApproval.token.address && appr.spenderAddress === aggrApproval.spenderAddress,\n );\n\n if (!aggregatedApproval) {\n aggregatedTokenApprovals.approvals.push({ ...appr });\n return;\n }\n\n // Replace with new values\n aggregatedApproval.value = appr.value;\n aggregatedApproval.usdPrice = appr.usdPrice;\n });\n\n return aggregatedTokenApprovals;\n },\n {\n approvals: [],\n isEditable: false,\n } as TokenApprovals,\n );\n\n clearSpentApprovals(tokenApprovals, scans);\n\n return isEmpty(tokenApprovals) ? undefined : tokenApprovals;\n};\n\nconst aggregateBalanceChange = (scans: TransactionSimulationResult[]): BalanceChange | undefined => {\n const aggregated = scans.reduce(\n (aggregatedBalanceChange, { balanceChange }) => {\n if (!balanceChange) {\n return aggregatedBalanceChange;\n }\n\n balanceChange.ins.forEach((diff) => {\n const aggregatedDiff = findTokenDiff(aggregatedBalanceChange.ins, diff);\n\n if (aggregatedDiff) {\n aggregatedDiff.items.push(...diff.items);\n } else {\n aggregatedBalanceChange.ins.push(diff);\n }\n });\n\n balanceChange.outs.forEach((diff) => {\n const aggregatedDiff = findTokenDiff(aggregatedBalanceChange.outs, diff);\n\n if (aggregatedDiff) {\n aggregatedDiff.items.push(...diff.items);\n } else {\n aggregatedBalanceChange.outs.push({ ...diff });\n }\n });\n\n return aggregatedBalanceChange;\n },\n {\n ins: [],\n outs: [],\n } as BalanceChange,\n );\n\n return isEmpty(aggregated) ? undefined : aggregated;\n};\n\nconst findTokenDiff = (diffs: TokenDiff[], lookupDiff: TokenDiff): TokenDiff | undefined =>\n diffs.find((diff) => {\n if (isNetworkToken(diff.token) && isNetworkToken(lookupDiff.token)) {\n return diff.token.symbol === lookupDiff.token.symbol;\n }\n\n if (!isNetworkToken(diff.token) && !isNetworkToken(lookupDiff.token)) {\n return diff.token.address === lookupDiff.token.address;\n }\n\n return false;\n });\n\nconst clearSpentApprovals = (tokenApprovals: TokenApprovals, scans: TransactionSimulationResult[]) => {\n // Go through each transaction. If the approvals were spent, remove them from the aggregated token approvals:\n for (const approval of tokenApprovals.approvals) {\n for (const scan of scans) {\n if (!scan.isSimulationSuccessful || !scan.balanceChange) {\n continue;\n }\n\n const touchesApproval = scan.balanceChange.outs.some((diff) => {\n return (\n !isNetworkToken(diff.token) &&\n !isNetworkToken(approval.token) &&\n diff.token.address === approval.token.address\n );\n });\n\n if (!touchesApproval) {\n continue;\n }\n\n if (!scan.tokenApprovals) {\n // Allowance was completely spent, remove the approval from aggregated view\n //\n // NOTE: With the current implementation of the BlockAid API, this is just an assumption on our part!\n // It's possible that the next transaction simply does not touch the approval at all.\n // Since we're initially be using this handler for the Swap feature, though, most of the time the allowance\n // will be spent completely.\n //\n // The BlockAid team is working to update their API to provide more accurate data.\n tokenApprovals.approvals = tokenApprovals.approvals.filter(\n (appr) => appr.spenderAddress !== approval.spenderAddress && approval.token.address !== appr.token.address,\n );\n } else {\n const latestApprovalState = scan.tokenApprovals.approvals.find(\n (appr) => appr.spenderAddress === approval.spenderAddress && appr.token.address === approval.token.address,\n );\n\n if (!latestApprovalState) {\n continue;\n }\n approval.usdPrice = latestApprovalState.usdPrice;\n approval.value = latestApprovalState.value;\n }\n }\n }\n};\n"]}
1
+ {"version":3,"sources":["../src/module.ts","../src/handlers/get-tokens/get-tokens.ts","../src/utils/get-provider.ts","../../../packages-internal/utils/src/services/token-service/token-service.ts","../../../packages-internal/utils/src/utils/retry.ts","../../../packages-internal/utils/src/utils/coingecko-retry.ts","../../../packages-internal/utils/src/utils/charsum.ts","../../../packages-internal/utils/src/utils/array-hash.ts","../../../packages-internal/utils/src/services/token-service/coingecko-proxy-client.ts","../../../packages-internal/utils/src/utils/fetch-and-verify.ts","../../../packages-internal/utils/src/services/token-service/watchlist-proxy-client.ts","../../../packages-internal/utils/src/services/pricing-service/exchange-rates.ts","../../../packages-internal/utils/src/utils/detail-item.ts","../../../packages-internal/utils/src/utils/get-core-headers.ts","../../../packages-internal/utils/src/consts.ts","../../../packages-internal/utils/src/utils/get-glacier-api-key.ts","../../../packages-internal/utils/src/utils/glacier-fetch-http-request.ts","../src/handlers/get-network-fee/get-network-fee.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/convert-transaction-normal.ts","../src/handlers/get-transaction-history/utils/get-explorer-address-by-network.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/convert-transaction-erc20.ts","../src/handlers/get-transaction-history/converters/etherscan-transaction-converter/get-transaction-from-etherscan.ts","../src/handlers/get-transaction-history/utils/is-ethereum-chain-id.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-tx-type.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-sender-info.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-tokens.ts","../src/handlers/get-transaction-history/utils/resolve.ts","../src/handlers/get-transaction-history/utils/ipfs-resolver-with-fallback.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-nft-metadata.ts","../src/handlers/get-transaction-history/utils/get-small-image-for-nft.ts","../src/types.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/convert-transaction.ts","../src/handlers/get-transaction-history/converters/evm-transaction-converter/get-transactions-from-glacier.ts","../src/handlers/get-transaction-history/get-transaction-history.ts","../manifest.json","../src/handlers/eth-send-transaction/eth-send-transaction.ts","../src/utils/get-nonce.ts","../src/utils/evm-tx-updater.ts","../src/utils/parse-erc20-transaction-type.ts","../src/utils/encode-erc20-approval.ts","../src/utils/estimate-gas-limit.ts","../src/utils/build-tx-approval-request.ts","../src/utils/process-transaction-simulation.ts","../src/utils/scan-transaction.ts","../src/utils/parse-erc20-tx.ts","../src/utils/type-utils.ts","../src/utils/transaction-alerts.ts","../src/handlers/eth-send-transaction/schema.ts","../src/utils/transaction-schema.ts","../src/utils/wait-for-transaction-receipt.ts","../src/utils/get-tx-hash.ts","../src/handlers/get-balances/get-balances.ts","../src/utils/find-async.ts","../src/utils/id-promise.ts","../src/services/rpc-service/rpc-service.ts","../src/env.ts","../src/services/glacier-service/glacier-service.ts","../src/constants.ts","../src/utils/ipsf-resolver-with-fallback.ts","../src/utils/get-small-image-for-nft.ts","../src/handlers/eth-sign/eth-sign.ts","../src/handlers/eth-sign/utils/beautify-message/beautify-message.ts","../src/handlers/eth-sign/schemas/parse-request-params/parse-request-params.ts","../src/handlers/eth-sign/schemas/eth-sign.ts","../src/handlers/eth-sign/schemas/shared.ts","../src/handlers/eth-sign/schemas/eth-sign-typed-data.ts","../src/handlers/eth-sign/schemas/personal-sign.ts","../src/handlers/eth-sign/utils/typeguards.ts","../src/handlers/eth-sign/utils/is-typed-data-valid.ts","../src/handlers/forward-to-rpc-node/forward-to-rpc-node.ts","../src/handlers/get-address/get-address.ts","../src/handlers/derive-address/derive-address.ts","../../../packages-internal/utils/src/utils/address-derivation.ts","../src/handlers/build-derivation-path/build-derivation-path.ts","../src/services/debank-service/debank-service.ts","../src/services/debank-service/de-bank.ts","../src/handlers/eth-send-transaction-batch/eth-send-transaction-batch.ts","../src/handlers/eth-send-transaction-batch/schema.ts","../src/handlers/eth-send-transaction-batch/utils/ensure-proper-nonces.ts","../src/utils/evm-tx-batch-updater.ts","../src/handlers/eth-send-transaction-batch/utils/process-transaction-batch-simulation.ts"],"names":["RpcMethod","parseManifest","rpcErrors","getTokens","chainId","proxyApiUrl","response","JsonRpcBatchInternal","VsCurrencyType","getBasicCoingeckoHttp","simplePrice","simpleTokenPrice","retry","operation","isSuccess","maxRetries","backoffPolicy","RetryBackoffPolicy","backoffPeriodMillis","retries","lastError","delay","result","err","errorMessage","retryIndex","secondsToDelay","_","msToDelay","linearCount","linearStepMs","n","base","incSum","ms","r","coingeckoRetry","charsum","s","i","sum","arrayHash","array","cs","RawSimplePriceResponseSchema","fetchAndVerify","fetchOptions","schema","responseJson","CoingeckoProxyClient","params","queryParams","id","rawQueryParams","z","WatchlistTokenResponseSchema","WatchlistProxyClient","coingeckoBasicClient","_storage","_proxyApiUrl","TokenService","storage","__privateAdd","__publicField","data","currencies","formattedData","tokenData","currency","__privateSet","tokenDetails","tokenInfo","__privateGet","token","coinIds","cacheId","useCoingeckoProxy","tokenAddresses","assetPlatformId","rawData","marketCap","vol24","change24","lastUpdated","shouldThrow","number","object","record","string","CURRENCY_EXCHANGE_RATES_URL","CURRENCY_EXCHANGE_RATES_FALLBACK_URL","ExchangeRateSchema","getExchangeRates","DetailItemType","textItem","label","value","alignment","linkItem","addressItem","networkItem","AppName","getCoreHeaders","name","version","GLACIER_API_KEY","getGlacierApiKey","FetchHttpRequest","GLOBAL_QUERY_PARAMS","GlacierFetchHttpRequest","config","options","mergedQuery","modifiedOptions","EVMNetwork","getProvider","chainName","rpcUrl","multiContractAddress","pollingInterval","provider","addGlacierAPIKeyIfNeeded","knownHosts","url","urlObj","glacierApiKey","search_params","TokenUnit","ChainId","DEFAULT_PRESETS","BASE_PRIORITY_FEE_WEI","isCChain","getNetworkFee","caipId","suggestedFees","suggestPriceOptions","lastBlock","baseFeePerGas","gasMultiplier","getGasMultiplier","multiplier","baseFee","maxPriorityFeePerGas","maxFee","lowMaxTip","mediumMaxTip","highMaxTip","respond","multipliers","TokenType","TransactionType","getExplorerAddressByNetwork","explorerUrl","hash","hashType","convertTransactionNormal","tx","networkToken","address","isSender","timestamp","decimals","amountDisplayValue","txType","from","to","gasPrice","gasUsed","explorerLink","isContractCall","convertTransactionERC20","tokenDecimal","tokenName","tokenSymbol","contractAddress","getErc20Txs","getNormalTxs","getTransactionFromEtherscan","isTestnet","nextPageToken","offset","parsedPageToken","page","queries","normalHist","erc20Hist","erc20TxHashes","filteredNormalTxs","next","isEthereumChainId","startCase","getTxType","nativeTransaction","erc20Transfers","erc721Transfers","erc1155Transfers","userAddress","tokens","nativeOnly","method","parseRawMethod","isSwap","isNativeSend","isNativeReceive","isNFTPurchase","isApprove","isTransfer","isAirdrop","isUnwrap","isFillOrder","isNFTSend","isNFT","isNFTReceive","tokenType","getSenderInfo","isOutgoing","isIncoming","resolve","promise","res","ipfsResolver","CLOUDFLARE_IPFS_URL","ipfsResolverWithFallback","sourceUrl","desiredGatewayPrefix","fetchWithTimeout","uri","timeout","controller","getNftMetadata","tokenUri","json","COVALENT_IMG_SIZER","getSmallImageForNFT","imgUrl","imageSize","decimal","erc20Transfer","erc721Transfer","imageUri","getImageUri","erc1155Transfer","metadata","error","NonContractCallTypes","ERC20TransactionType","convertTransaction","transactions","blockTimestamp","getTransactionsFromGlacier","glacierService","tranasaction","transaction","getTransactionHistory","manifest_default","getNonce","Interface","ERC20","parseERC20TransactionType","description","functionName","isERC20TransactionType","ethers","encodeApprovalLimit","tokenAddress","spenderAddress","limit","requests","getTxUpdater","requestId","signingData","displayData","maxFeeRate","maxTipRate","approvalLimit","gasLimit","request","newSigningData","newDisplayData","tokenApprovals","ensureTokenApprovalCanBeEdited","ensureApproveWasCalled","approval","updatedRequest","txData","estimateGasLimit","accessList","nonce","buildTxApprovalRequest","network","isSimulationSuccessful","balanceChange","alert","dappInfo","transactionType","title","transactionDetails","AlertType","balanceToDisplayValue","numberToBN","isHexString","MaxUint256","scanTransactionBatch","domain","withGasEstimation","blockaid","scanTransaction","scanJsonRpc","accountAddress","xss","parseWithErc20Abi","contract","contractCalls","symbol","iface","calledFunction","decodeFunctionData","isNetworkToken","isERC20Token","hasToField","supportsBatchApprovals","isBalanceChange","input","isTokenApprovals","isEmpty","transactionAlerts","simulateTransaction","rpcMethod","dAppUrl","simulationResult","processTransactionSimulation","estimatedGasLimit","validation","simulation","gasEstimation","processTokenApprovals","processBalanceChange","erc20ParseResult","isExposureTrace","trace","normalizeAddresses","spender","assetAddress","mapExposureTracesToSpenderAsset","traces","accumulator","accountSummary","mappedExposureTraces","approvals","convertAssetToNetworkContractToken","tokenApproval","raw_value","usd_price","amount","assetDiffs","ins","processAssetDiffs","outs","type","assetDiff","a","b","asset","convertNativeAssetToToken","items","diff","displayValue","valueBN","x","processJsonRpcSimulation","transactionSchema","paramsSchema","parseRequestParams","waitForTransactionReceipt","txHash","onTransactionPending","onTransactionConfirmed","onTransactionReverted","getTxHash","ethSendTransaction","approvalController","success","scan","updateTx","cleanup","findAsync","asyncCallback","promises","item","index","found","addIdToPromise","reason","settleAllIdPromises","results","acc","_network","_customTokens","RpcService","customTokens","coingeckoTokenId","tokenService","simplePriceResponse","priceInCurrency","balance","balanceUnit","balanceInCurrency","coingeckoPlatformId","erc20TokenList","getTokenWithBalance","tokenBalancePromises","tokenBalancesResults","tokenIds","erc20TokenBalances","tokenId","tokenBalance","getBalances","addresses","tokenTypes","balanceServices","services","supportingService","balanceService","balances","nativeTokenPromises","erc20TokenPromises","nftTokenPromises","nativeBalance","avaxMarketData","nativeTokenBalances","nftTokenBalances","balanceOrError","balancesOrError","Environment","prodEnv","devEnv","getEnv","environment","Erc1155Token","Erc721Token","Glacier","BLOCKAID_API_KEY","IPFS_URL","GlacierUnhealthyError","CHAINS_TO_FILTER","EvmGlacierService","glacierApiUrl","headers","supportedChains","chain","coingeckoId","nativeTokenBalance","entries","tokenlist","tokensWithBalance","convertErc20TokenWithBalanceToTokenWithBalance","convertErc20TokenToTokenWithBalance","pageToken","pageSize","tokenBalances","balanceDisplayValue","balanceCurrencyDisplayValue","toUtf8String","beautifyComplexMessage","key","toUnicodeBold","subKey","beautifySimpleMessage","str","boldMap","char","messageSchema","addressSchema","ethSignSchema","messageTypeSchema","typedDataSchema","typedDataV1Schema","combinedTypedDataSchema","dataSchema","combinedDataSchema","ethSignTypedDataSchema","ethSignTypedDataV1Schema","ethSignTypedDataV3Schema","ethSignTypedDataV4Schema","personalSignSchema","ctx","parsed","isTypedDataV1","isTypedData","TypedDataEncoder","isTypedDataValid","EIP712Domain","types","ethSign","typedDataValidationResult","messageDetails","primaryType","messageToDisplay","forwardToRpcNode","message","NetworkVMType","WalletType","getAddressFromXPub","getEvmAddressFromPubKey","getAddress","accountIndex","xpub","walletType","pubKeyBuffer","computeAddress","hasDerivationDetails","buildDerivationPath","derivationPathType","deriveAddress","secretId","derivationPath","publicKeyHex","NftTokenMetadataStatus","_supportedChainIds","DeBank","baseUrl","tokenBalanceResponse","chainListResponse","DeBankNftTokenListSchema","BaseDeBankTokenSchema","DeBankNftTokenSchema","_deBank","_getChainInfo","getChainInfo_fn","_mapNftList","mapNftList_fn","DeBankService","chainInfo","chainIdString","__privateMethod","tokenUnit","usdToCurrencyRate","exchangeRates","nftList","deBankNftList","transactionArraySchema","areAllEqualByProp","areAllEmptyByProp","areAllDifferentByProp","batchOptionsSchema","transactionBatchSchema","elements","prop","el","ensureProperNonces","getTxBatchUpdater","signingRequests","txIndex","signingRequest","simulateTransactionBatch","populateMissingGas","simulationResults","scans","matchingTx","processedResult","getHighestAlert","aggregateBalanceChange","aggregateTokenApprovals","alertPrio","highest","aggregatedTokenApprovals","appr","aggregatedApproval","aggrApproval","clearSpentApprovals","aggregated","aggregatedBalanceChange","aggregatedDiff","findTokenDiff","diffs","lookupDiff","latestApprovalState","ethSendTransactionBatch","parsedParams","transactionRequests","skipIntermediateTxs","allTransactionPayloadsHaveGas","txHashes","isLast","Blockaid","_glacierService","_deBankService","_approvalController","_blockaid","EvmModule","appInfo","runtime","utilityAddresses","shouldForwardToRpcNode"],"mappings":"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","sourcesContent":["import {\n type Module,\n type Manifest,\n type NetworkFees,\n type GetTransactionHistory,\n type RpcRequest,\n type Network,\n type ApprovalController,\n type GetBalancesParams,\n type GetBalancesResponse,\n type GetAddressParams,\n type GetAddressResponse,\n RpcMethod,\n parseManifest,\n type ConstructorParams,\n type NetworkFeeParam,\n type DeriveAddressParams,\n type DeriveAddressResponse,\n type BuildDerivationPathParams,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getTokens } from './handlers/get-tokens/get-tokens';\nimport { getNetworkFee } from './handlers/get-network-fee/get-network-fee';\nimport { getTransactionHistory } from './handlers/get-transaction-history/get-transaction-history';\nimport ManifestJson from '../manifest.json';\nimport { ethSendTransaction } from './handlers/eth-send-transaction/eth-send-transaction';\nimport { getBalances } from './handlers/get-balances/get-balances';\nimport { getEnv } from './env';\nimport { EvmGlacierService } from './services/glacier-service/glacier-service';\nimport { ethSign } from './handlers/eth-sign/eth-sign';\nimport { forwardToRpcNode } from './handlers/forward-to-rpc-node/forward-to-rpc-node';\nimport { getAddress } from './handlers/get-address/get-address';\nimport { deriveAddress } from './handlers/derive-address/derive-address';\nimport { DeBankService } from './services/debank-service/debank-service';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getProvider } from './utils/get-provider';\nimport { getCoreHeaders, TokenService } from '@internal/utils';\nimport { ethSendTransactionBatch } from './handlers/eth-send-transaction-batch/eth-send-transaction-batch';\nimport { supportsBatchApprovals } from './utils/type-utils';\nimport { buildDerivationPath } from './handlers/build-derivation-path/build-derivation-path';\nimport Blockaid from '@blockaid/client';\nimport { BLOCKAID_API_KEY } from './constants';\n\nexport class EvmModule implements Module {\n #glacierService: EvmGlacierService;\n #deBankService: DeBankService;\n #proxyApiUrl: string;\n #approvalController: ApprovalController;\n #blockaid: Blockaid;\n\n constructor({ approvalController, environment, appInfo, runtime }: ConstructorParams) {\n const { glacierApiUrl, proxyApiUrl } = getEnv(environment);\n this.#glacierService = new EvmGlacierService({\n glacierApiUrl,\n headers: getCoreHeaders(appInfo),\n });\n this.#deBankService = new DeBankService({ proxyApiUrl });\n this.#proxyApiUrl = proxyApiUrl;\n this.#approvalController = approvalController;\n this.#blockaid = new Blockaid({\n baseURL: proxyApiUrl + '/proxy/blockaid/',\n apiKey: BLOCKAID_API_KEY,\n httpAgent: runtime?.httpAgent,\n fetch: runtime?.fetch,\n });\n }\n\n getProvider(network: Network): Promise<JsonRpcBatchInternal> {\n return getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n }\n\n getAddress({ accountIndex, xpub, walletType }: GetAddressParams): Promise<GetAddressResponse> {\n return getAddress({ accountIndex, xpub, walletType });\n }\n\n buildDerivationPath(params: BuildDerivationPathParams) {\n return buildDerivationPath(params);\n }\n\n deriveAddress(params: DeriveAddressParams): Promise<DeriveAddressResponse> {\n return deriveAddress({\n ...params,\n approvalController: this.#approvalController,\n });\n }\n\n getBalances({\n addresses,\n network,\n currency,\n customTokens,\n storage,\n tokenTypes,\n }: GetBalancesParams): Promise<GetBalancesResponse> {\n const tokenService = new TokenService({ storage, proxyApiUrl: this.#proxyApiUrl });\n return getBalances({\n addresses,\n currency,\n network,\n proxyApiUrl: this.#proxyApiUrl,\n customTokens,\n balanceServices: [this.#glacierService, this.#deBankService],\n storage,\n tokenTypes,\n tokenService,\n });\n }\n\n getManifest(): Manifest | undefined {\n const result = parseManifest(ManifestJson);\n return result.success ? result.data : undefined;\n }\n\n getNetworkFee(network: NetworkFeeParam): Promise<NetworkFees> {\n const { chainId, chainName, rpcUrl, utilityAddresses, caipId } = network;\n return getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress: utilityAddresses?.multicall,\n caipId,\n proxyApiUrl: this.#proxyApiUrl,\n });\n }\n\n getTransactionHistory(params: GetTransactionHistory) {\n const { network, address, nextPageToken, offset } = params;\n const { chainId, isTestnet, networkToken, explorerUrl = '' } = network;\n\n return getTransactionHistory({\n chainId,\n isTestnet,\n networkToken,\n explorerUrl,\n address,\n nextPageToken,\n offset,\n glacierService: this.#glacierService,\n });\n }\n\n getTokens(network: Network) {\n const { chainId } = network;\n return getTokens({ chainId, proxyApiUrl: this.#proxyApiUrl });\n }\n\n async onRpcRequest(request: RpcRequest, network: Network) {\n switch (request.method) {\n case RpcMethod.ETH_SEND_TRANSACTION:\n return ethSendTransaction({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n case RpcMethod.ETH_SEND_TRANSACTION_BATCH: {\n if (!supportsBatchApprovals(this.#approvalController)) {\n return {\n error: rpcErrors.methodNotSupported(`Method ${request.method} requires BatchApprovalController`),\n };\n }\n\n return ethSendTransactionBatch({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n }\n case RpcMethod.PERSONAL_SIGN:\n case RpcMethod.ETH_SIGN:\n case RpcMethod.SIGN_TYPED_DATA:\n case RpcMethod.SIGN_TYPED_DATA_V1:\n case RpcMethod.SIGN_TYPED_DATA_V3:\n case RpcMethod.SIGN_TYPED_DATA_V4:\n return ethSign({\n request,\n network,\n approvalController: this.#approvalController,\n blockaid: this.#blockaid,\n });\n default:\n if (shouldForwardToRpcNode(request.method)) {\n return forwardToRpcNode(request, network);\n }\n\n return { error: rpcErrors.methodNotSupported(`Method ${request.method} not supported`) };\n }\n }\n}\n\nconst shouldForwardToRpcNode = (method: RpcMethod) => {\n return (\n method.startsWith('eth_') ||\n ['web3_clientVersion', 'web3_sha3', 'net_version', 'net_peerCount', 'net_listening'].includes(method)\n );\n};\n","import type { NetworkContractToken } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nexport async function getTokens({\n chainId,\n proxyApiUrl,\n}: {\n chainId: number;\n proxyApiUrl: string;\n}): Promise<NetworkContractToken[]> {\n const response = await fetch(`${proxyApiUrl}/tokens?evmChainId=${chainId}`);\n\n if (!response.ok) {\n throw rpcErrors.internal(`Failed to fetch tokens for chainId ${chainId}`);\n }\n\n return response.json();\n}\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getGlacierApiKey } from '@internal/utils';\nimport { Network as EVMNetwork } from 'ethers';\n\ntype ProviderParams = {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n multiContractAddress?: string;\n pollingInterval?: number;\n};\n\nexport const getProvider = async (params: ProviderParams): Promise<JsonRpcBatchInternal> => {\n const { chainId, chainName, rpcUrl, multiContractAddress, pollingInterval = 2000 } = params;\n\n const provider = new JsonRpcBatchInternal(\n { maxCalls: 40, multiContractAddress },\n addGlacierAPIKeyIfNeeded(rpcUrl),\n new EVMNetwork(chainName, chainId),\n );\n\n provider.pollingInterval = pollingInterval;\n\n return provider;\n};\n\n// RPC urls returned in the token list are always using the production URL\nconst knownHosts = ['glacier-api.avax.network', 'proxy-api.avax.network'];\n\n/**\n * Glacier needs an API key for development, this adds the key if needed.\n */\nexport function addGlacierAPIKeyIfNeeded(url: string): string {\n const urlObj = new URL(url);\n const glacierApiKey = getGlacierApiKey();\n\n if (glacierApiKey && knownHosts.includes(urlObj.hostname)) {\n const search_params = urlObj.searchParams; // copy, does not update the URL\n search_params.set('token', glacierApiKey);\n urlObj.search = search_params.toString();\n return urlObj.toString();\n }\n return url;\n}\n","import {\n VsCurrencyType,\n getBasicCoingeckoHttp,\n simplePrice,\n simpleTokenPrice,\n type SimplePriceParams,\n} from '@avalabs/core-coingecko-sdk';\nimport type { Storage, RawSimplePriceResponse, SimplePriceResponse } from '@avalabs/vm-module-types';\nimport { coingeckoRetry } from '../../utils/coingecko-retry';\nimport { arrayHash } from '../../utils/array-hash';\nimport { CoingeckoProxyClient } from './coingecko-proxy-client';\nimport { WatchlistProxyClient } from './watchlist-proxy-client';\n\nconst coingeckoBasicClient = getBasicCoingeckoHttp();\n\nexport class TokenService {\n #storage?: Storage;\n #proxyApiUrl: string;\n\n constructor({ storage, proxyApiUrl }: { proxyApiUrl: string; storage?: Storage }) {\n this.#storage = storage;\n this.#proxyApiUrl = proxyApiUrl;\n }\n\n async getWatchlistDataForToken({\n tokenDetails,\n currency = VsCurrencyType.USD,\n }: {\n tokenDetails: {\n symbol: string;\n isNative: boolean;\n caip2Id: string;\n address?: string;\n };\n currency: VsCurrencyType;\n }): Promise<{\n priceInCurrency: number;\n change24: number;\n marketCap: number;\n vol24: number;\n }> {\n const data = (\n await new WatchlistProxyClient(this.#proxyApiUrl).watchlistToken({\n tokens: tokenDetails.symbol,\n currency: currency,\n })\n ).filter((token) => {\n return tokenDetails.isNative\n ? token.internalId === `NATIVE-${tokenDetails.symbol.toLowerCase()}`\n : token.platforms[tokenDetails.caip2Id] === tokenDetails.address;\n });\n\n const tokenInfo = data[0];\n\n if (!tokenInfo) {\n return {\n priceInCurrency: 0,\n change24: 0,\n marketCap: 0,\n vol24: 0,\n };\n }\n\n return {\n priceInCurrency: tokenInfo.current_price ?? 0,\n change24: tokenInfo.price_change_percentage_24h ?? 0,\n marketCap: tokenInfo.market_cap ?? 0,\n vol24: tokenInfo.total_volume ?? 0,\n };\n }\n\n /**\n * Get token price with market data first on coingecko (free tier) directly,\n * if we get 429 error, retry it on coingecko proxy (paid service)\n * @returns token price with market data\n */\n async getSimplePrice({\n coinIds = [],\n currencies = [VsCurrencyType.USD],\n }: SimplePriceParams): Promise<SimplePriceResponse | undefined> {\n let data: SimplePriceResponse | undefined;\n\n const key = coinIds ? `${arrayHash(coinIds)}-${currencies.toString()}` : `${currencies.toString()}`;\n\n const cacheId = `getSimplePrice-${key}`;\n\n data = this.#storage?.get?.<SimplePriceResponse>(cacheId);\n\n if (data) return data;\n\n try {\n data = await coingeckoRetry((useCoingeckoProxy) =>\n this.simplePrice({\n coinIds,\n currencies,\n marketCap: true,\n vol24: true,\n change24: true,\n useCoingeckoProxy,\n }),\n );\n } catch {\n data = undefined;\n }\n this.#storage?.set?.(cacheId, data);\n return data;\n }\n\n /**\n * Get token price with market data for a list of addresses\n * @param tokenAddresses the token addresses\n * @param assetPlatformId The platform id for all the tokens in the list\n * @param currency the currency to be used\n * @returns a list of token price with market data\n */\n async getPricesByAddresses(\n tokenAddresses: string[],\n assetPlatformId: string,\n currency: VsCurrencyType = VsCurrencyType.USD,\n ): Promise<SimplePriceResponse | undefined> {\n let data: SimplePriceResponse | undefined;\n\n const key = `${arrayHash(tokenAddresses)}-${assetPlatformId}-${currency}`;\n\n const cacheId = `getPricesWithMarketDataByAddresses-${key}`;\n data = this.#storage?.get?.<SimplePriceResponse>(cacheId);\n\n if (data) return data;\n\n try {\n data = await coingeckoRetry((useCoingeckoProxy) =>\n this.fetchPricesByAddresses({\n assetPlatformId,\n tokenAddresses,\n currency,\n useCoingeckoProxy,\n }),\n );\n } catch (err) {\n console.error(err);\n data = undefined;\n }\n this.#storage?.set?.(cacheId, data);\n return data;\n }\n\n private async fetchPricesByAddresses({\n assetPlatformId,\n tokenAddresses,\n currency = VsCurrencyType.USD,\n useCoingeckoProxy = false,\n }: {\n assetPlatformId: string;\n tokenAddresses: string[];\n currency: VsCurrencyType;\n useCoingeckoProxy?: boolean;\n }): Promise<SimplePriceResponse> {\n if (useCoingeckoProxy) {\n const rawData = await new CoingeckoProxyClient(this.#proxyApiUrl).simplePriceByContractAddresses({\n id: assetPlatformId,\n contract_addresses: tokenAddresses,\n vs_currencies: [currency],\n include_market_cap: true,\n include_24hr_vol: true,\n include_24hr_change: true,\n });\n return this.transformSimplePriceResponse(rawData, [currency]);\n }\n\n return simpleTokenPrice(coingeckoBasicClient, {\n assetPlatformId,\n tokenAddresses,\n currencies: [currency],\n marketCap: true,\n vol24: true,\n change24: true,\n });\n }\n\n private async simplePrice({\n coinIds = [],\n currencies = [VsCurrencyType.USD],\n marketCap = false,\n vol24 = false,\n change24 = false,\n lastUpdated = false,\n useCoingeckoProxy = false,\n shouldThrow = true,\n }: SimplePriceParams & { useCoingeckoProxy?: boolean }): Promise<SimplePriceResponse> {\n if (useCoingeckoProxy) {\n const rawData = await new CoingeckoProxyClient(this.#proxyApiUrl).simplePrice({\n ids: coinIds,\n vs_currencies: currencies,\n include_market_cap: marketCap,\n include_24hr_vol: vol24,\n include_24hr_change: change24,\n include_last_updated_at: lastUpdated,\n });\n return this.transformSimplePriceResponse(rawData, currencies);\n }\n return simplePrice(coingeckoBasicClient, {\n coinIds,\n currencies,\n marketCap,\n vol24,\n change24,\n lastUpdated,\n shouldThrow,\n });\n }\n\n private transformSimplePriceResponse = (\n data: RawSimplePriceResponse,\n currencies = [VsCurrencyType.USD],\n ): SimplePriceResponse => {\n const formattedData: SimplePriceResponse = {};\n Object.keys(data).forEach((id) => {\n const tokenData = data[id];\n formattedData[id] = {};\n currencies.forEach((currency: VsCurrencyType) => {\n formattedData[id] = {\n [currency]: {\n price: tokenData?.[currency],\n change24: tokenData?.[`${currency}_24h_change`],\n vol24: tokenData?.[`${currency}_24h_vol`],\n marketCap: tokenData?.[`${currency}_market_cap`],\n },\n };\n });\n });\n return formattedData;\n };\n}\n","const DEFAULT_MAX_RETRIES = 10;\n\ntype RetryParams<T> = {\n operation: (retryIndex: number) => Promise<T>;\n isSuccess: (result: T) => boolean;\n maxRetries?: number;\n backoffPolicy?: RetryBackoffPolicyInterface;\n};\n/*\n * Retries an operation with defined backoff policy.\n *\n * @param operation - The operation to retry.\n * @param isSuccess - The predicate to check if the operation succeeded.\n * @param maxRetries - The maximum number of retries.\n * @param backoffPolicy - Function to generate delay time based on current retry count.\n *\n * @returns The result of the operation.\n * @throws An error if the operation fails after the maximum number of retries.\n *\n * @example\n * const result = await retry(\n * async () => {\n * const response = await fetch('https://example.com')\n * return response.json()\n * },\n * result => result.status === 200\n * )\n */\nexport const retry = async <T>({\n operation,\n isSuccess,\n maxRetries = DEFAULT_MAX_RETRIES,\n backoffPolicy = RetryBackoffPolicy.exponential(),\n}: RetryParams<T>): Promise<T> => {\n let backoffPeriodMillis = 0;\n let retries = 0;\n let lastError: unknown;\n\n while (retries < maxRetries) {\n if (retries > 0) {\n await delay(backoffPeriodMillis);\n }\n\n try {\n const result = await operation(retries);\n\n if (isSuccess(result)) {\n return result;\n }\n } catch (err) {\n // when the operation throws an error, we still retry\n lastError = err;\n }\n\n backoffPeriodMillis = backoffPolicy(retries);\n retries++;\n }\n\n const errorMessage = lastError ? `Max retry exceeded. ${lastError}` : 'Max retry exceeded.';\n\n throw new Error(errorMessage);\n};\n\ntype RetryBackoffPolicyInterface = (retryIndex: number) => number;\n\nexport class RetryBackoffPolicy {\n static exponential(): RetryBackoffPolicyInterface {\n return (retryIndex: number): number => {\n return Math.pow(2, retryIndex) * 1000;\n };\n }\n\n static constant(secondsToDelay: number): RetryBackoffPolicyInterface {\n return (_: number): number => {\n return secondsToDelay * 1000;\n };\n }\n\n static constantMs(msToDelay: number): RetryBackoffPolicyInterface {\n return (_: number): number => {\n return msToDelay;\n };\n }\n\n /**\n * linearThenExponential backoff:\n * - First `linearCount` retries: linear increase by `linearStepMs`\n * - After that: increment grows exponentially based on `linearStepMs`\n * Example (linearCount=4, linearStepMs=1000):\n * 1s, 2s, 3s, 4s, 6s, 10s, 18s, 34s...\n */\n static linearThenExponential(linearCount: number, linearStepMs: number): RetryBackoffPolicyInterface {\n return (retryIndex: number): number => {\n if (retryIndex < linearCount) {\n // Linear phase: (i+1) * step\n return (retryIndex + 1) * linearStepMs;\n }\n // Exponential-increment phase (closed form):\n // n = number of exponential increments applied\n // base = linearCount * step\n // increment sum = 2*step * (2^n - 1)\n const n = retryIndex - linearCount + 1;\n const base = linearCount * linearStepMs;\n const incSum = 2 * linearStepMs * (Math.pow(2, n) - 1);\n return base + incSum;\n };\n }\n}\n\nfunction delay(ms: number) {\n return new Promise((r) => setTimeout(r, ms));\n}\n","import { RetryBackoffPolicy, retry } from './retry';\n\ntype Error = {\n status: {\n error_code: number;\n error_message: string;\n };\n};\n\nexport const coingeckoRetry = <T>(\n operation: (useCoingeckoProxy: boolean) => Promise<T | Error>,\n): Promise<T | undefined> => {\n return retry({\n operation: (retryIndex: number) => operation(retryIndex > 0),\n maxRetries: 2,\n backoffPolicy: RetryBackoffPolicy.constant(1),\n isSuccess: (response: T | Error) => {\n const errorStatus = (response as Error)?.status;\n return errorStatus?.error_code !== 429;\n },\n }) as Promise<T | undefined>;\n};\n","export function charsum(s: string): number {\n let i,\n sum = 0;\n for (i = 0; i < s.length; i++) {\n sum += s.charCodeAt(i) * (i + 1);\n }\n return sum;\n}\n","import { charsum } from './charsum';\n\n// from https://stackoverflow.com/a/25105589\nexport function arrayHash(array: string[]): string {\n let i,\n sum = 0;\n for (i = 0; i < array.length; i++) {\n const cs = charsum(array[i] ?? '');\n sum = sum + 65027 / cs;\n }\n return ('' + sum).slice(0, 16);\n}\n","import { RawSimplePriceResponseSchema } from '@avalabs/vm-module-types';\nimport { fetchAndVerify } from '../../utils/fetch-and-verify';\n\nexport class CoingeckoProxyClient {\n constructor(private proxyApiUrl: string) {}\n\n simplePrice(params: {\n ids: string[];\n vs_currencies: string[];\n include_market_cap?: boolean;\n include_24hr_vol?: boolean;\n include_24hr_change?: boolean;\n include_last_updated_at?: boolean;\n }) {\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(params as any);\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/proxy/coingecko/simple/price?${queryParams}`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n RawSimplePriceResponseSchema,\n );\n }\n\n simplePriceByContractAddresses(params: {\n id: string;\n contract_addresses: string[];\n vs_currencies?: string[];\n include_market_cap?: boolean;\n include_24hr_vol?: boolean;\n include_24hr_change?: boolean;\n }) {\n const { id, ...rawQueryParams } = params;\n\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(rawQueryParams as any);\n\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/proxy/coingecko/simple/token_price/${id}?${queryParams}`,\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n RawSimplePriceResponseSchema,\n );\n }\n}\n","import z from 'zod';\nimport type { ZodSchema } from 'zod';\n\nexport async function fetchAndVerify<T extends ZodSchema>(\n fetchOptions: Parameters<typeof fetch>,\n schema: T,\n): Promise<z.infer<T>> {\n const response = await fetch(...fetchOptions);\n\n if (!response.ok) {\n throw new Error(`Request failed with status ${response.status}`);\n }\n\n const responseJson = await response.json();\n return schema.parse(responseJson);\n}\n","import { fetchAndVerify } from '../../utils/fetch-and-verify';\nimport { z } from 'zod';\n\nconst WatchlistTokenResponseSchema = z.array(\n z.object({\n // the object has more properties than the ones listed here, but we only need these at the moment\n internalId: z.string(),\n id: z.string(),\n symbol: z.string(),\n name: z.string(),\n image: z.string().optional().nullable(),\n current_price: z.number().optional().nullable(),\n price_change_percentage_24h: z.number().optional().nullable(),\n market_cap: z.number().optional().nullable(),\n total_volume: z.number().optional().nullable(),\n platforms: z.record(z.string(), z.string()),\n }),\n);\n\nexport class WatchlistProxyClient {\n constructor(private proxyApiUrl: string) {}\n\n watchlistToken(params: { tokens: string; currency: string }) {\n // casting params as any since typing does not allow boolean and other non-string values\n // even though NodeJS does not have this restriction itself: https://nodejs.org/api/url.html#new-urlsearchparamsobj\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const queryParams = new URLSearchParams(params as any);\n return fetchAndVerify(\n [\n `${this.proxyApiUrl}/watchlist/tokens?${queryParams}`,\n {\n method: 'GET',\n headers: {\n 'Content-Type': 'application/json',\n },\n },\n ],\n WatchlistTokenResponseSchema,\n );\n }\n}\n","import z, { number, object, record, string } from 'zod';\nimport { fetchAndVerify } from '../../utils/fetch-and-verify';\n\nconst CURRENCY_EXCHANGE_RATES_URL =\n 'https://cdn.jsdelivr.net/npm/@fawazahmed0/currency-api@latest/v1/currencies/usd.min.json';\n\nconst CURRENCY_EXCHANGE_RATES_FALLBACK_URL = 'https://latest.currency-api.pages.dev/v1/currencies/usd.min.json';\n\nconst ExchangeRateSchema = object({\n date: string(),\n usd: record(number()),\n});\n\ntype ExchangeRate = z.infer<typeof ExchangeRateSchema>;\n\nexport const getExchangeRates = async (): Promise<ExchangeRate> => {\n try {\n return await fetchAndVerify([CURRENCY_EXCHANGE_RATES_URL], ExchangeRateSchema);\n } catch {\n return await fetchAndVerify([CURRENCY_EXCHANGE_RATES_FALLBACK_URL], ExchangeRateSchema);\n }\n};\n","import {\n type AddressItem,\n type CurrencyItem,\n type NodeIDItem,\n type TextItem,\n type DataItem,\n type DateItem,\n type LinkItemValue,\n DetailItemType,\n type LinkItem,\n type FundsRecipientItem,\n type AddressListItem,\n type NetworkItemValue,\n type NetworkItem,\n} from '@avalabs/vm-module-types';\n\nexport const fundsRecipientItem = (\n address: string,\n amount: bigint,\n maxDecimals: number,\n symbol: string,\n): FundsRecipientItem => ({\n type: DetailItemType.FUNDS_RECIPIENT,\n label: address,\n amount,\n maxDecimals,\n symbol,\n});\n\nexport const currencyItem = (label: string, value: bigint, maxDecimals: number, symbol: string): CurrencyItem => ({\n label,\n type: DetailItemType.CURRENCY,\n value,\n maxDecimals,\n symbol,\n});\n\nexport const textItem = (\n label: string,\n value: string,\n alignment: 'horizontal' | 'vertical' = 'horizontal',\n): TextItem => ({\n label,\n alignment,\n type: DetailItemType.TEXT,\n value,\n});\n\nexport const linkItem = (label: string, value: LinkItemValue): LinkItem => ({\n label,\n value,\n type: DetailItemType.LINK,\n});\n\nexport const addressItem = (label: string, value: string): AddressItem => ({\n label,\n type: DetailItemType.ADDRESS,\n value,\n});\n\nexport const addressListItem = (label: string, value: string[]): AddressListItem => ({\n label,\n type: DetailItemType.ADDRESS_LIST,\n value,\n});\n\nexport const nodeIDItem = (label: string, value: string): NodeIDItem => ({\n label,\n type: DetailItemType.NODE_ID,\n value,\n});\n\nexport const dataItem = (label: string, value: string): DataItem => ({\n label,\n type: DetailItemType.DATA,\n value,\n});\n\nexport const dateItem = (label: string, value: string): DateItem => ({\n label,\n type: DetailItemType.DATE,\n value,\n});\n\nexport const networkItem = (label: string, value: NetworkItemValue): NetworkItem => ({\n label,\n type: DetailItemType.NETWORK,\n value,\n});\n","import { AppName, type AppInfo } from '@avalabs/vm-module-types';\n\nexport const getCoreHeaders = ({ name, version }: AppInfo): Record<string, string> | undefined => {\n switch (name) {\n case AppName.CORE_MOBILE_IOS:\n case AppName.CORE_MOBILE_ANDROID:\n case AppName.CORE_WEB:\n case AppName.CORE_EXTENSION:\n case AppName.EXPLORER:\n return {\n 'x-application-name': name,\n 'x-application-version': version,\n };\n case AppName.OTHER:\n return undefined;\n }\n};\n","export const GLACIER_API_KEY = process.env.GLACIER_API_KEY;\n","import { GLACIER_API_KEY } from '../consts';\n\n// this key is only needed in development to bypass rate limit\n// it should never be used in production\nexport const getGlacierApiKey = (): string | undefined => {\n return GLACIER_API_KEY;\n};\n","import { FetchHttpRequest, type OpenAPIConfig, type ApiRequestOptions, CancelablePromise } from '@avalabs/glacier-sdk';\nimport { getGlacierApiKey } from './get-glacier-api-key';\n\nconst GLOBAL_QUERY_PARAMS: Record<string, string | undefined> = {\n rltoken: getGlacierApiKey(),\n};\n\n/**\n * Custom HTTP request handler that automatically appends the Glacier API key (if present)\n * to bypass rate limits in development environments.\n */\nexport class GlacierFetchHttpRequest extends FetchHttpRequest {\n constructor(config: OpenAPIConfig) {\n super(config);\n }\n\n public override request<T>(options: ApiRequestOptions): CancelablePromise<T> {\n // Merge global query parameters with request-specific ones\n const mergedQuery = {\n ...GLOBAL_QUERY_PARAMS,\n ...(options.query || {}), // Request-specific params (override globals if same key)\n };\n\n // Create modified options with merged query\n const modifiedOptions: ApiRequestOptions = {\n ...options,\n query: Object.keys(mergedQuery).length > 0 ? mergedQuery : undefined,\n };\n\n // Call the base class's request method\n return super.request<T>(modifiedOptions);\n }\n}\n","import type { NetworkFees, SuggestGasPriceOptionsResponse } from '@avalabs/vm-module-types';\nimport { getProvider } from '../../utils/get-provider';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { ChainId } from '@avalabs/core-chains-sdk';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nconst DEFAULT_PRESETS = {\n LOW: 1n,\n MEDIUM: 4n,\n HIGH: 6n,\n};\n\nconst BASE_PRIORITY_FEE_WEI = 500000000n; //0.5 GWei\n\nconst isCChain = (chainId: number) =>\n chainId === ChainId.AVALANCHE_TESTNET_ID || chainId === ChainId.AVALANCHE_MAINNET_ID;\n\n/**\n * Returns {@link NetworkFees} based on {@link DEFAULT_PRESETS} multipliers.\n * @throws Error if provider does not support eip-1559\n */\nexport async function getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n caipId,\n proxyApiUrl,\n}: {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n proxyApiUrl?: string;\n multiContractAddress?: string;\n caipId?: string;\n}): Promise<NetworkFees> {\n const provider = await getProvider({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n });\n\n if (!proxyApiUrl) {\n throw rpcErrors.internal('Proxy API URL is needed');\n }\n\n if (isCChain(chainId)) {\n try {\n const suggestedFees = await suggestPriceOptions(provider);\n\n return {\n ...suggestedFees,\n baseFee: suggestedFees.medium.maxFeePerGas,\n displayDecimals: 9,\n isFixedFee: false,\n };\n } catch (err) {\n console.error('eth_suggestPriceOptions call failed, falling back to legacy fee fetching');\n }\n }\n\n const lastBlock = await provider.getBlock('latest', false);\n\n if (!lastBlock) {\n throw rpcErrors.internal('There is no block');\n }\n const baseFeePerGas = lastBlock.baseFeePerGas;\n\n if (!baseFeePerGas) {\n throw rpcErrors.internal('Pre-EIP-1559 networks are not supported');\n }\n\n const gasMultiplier = await getGasMultiplier(proxyApiUrl, caipId);\n\n const multiplier = new TokenUnit(gasMultiplier, 0, '');\n\n const baseFee = new TokenUnit(baseFeePerGas, 0, '');\n const maxPriorityFeePerGas = BigInt('1000000000');\n\n const maxFee = baseFee.mul(multiplier).add(maxPriorityFeePerGas).toSubUnit();\n\n const lowMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.LOW;\n const mediumMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.MEDIUM;\n const highMaxTip = BASE_PRIORITY_FEE_WEI * DEFAULT_PRESETS.HIGH;\n return {\n baseFee: maxFee,\n low: {\n maxFeePerGas: maxFee + lowMaxTip,\n maxPriorityFeePerGas: lowMaxTip,\n },\n medium: {\n maxFeePerGas: maxFee + mediumMaxTip,\n maxPriorityFeePerGas: mediumMaxTip,\n },\n high: {\n maxFeePerGas: maxFee + highMaxTip,\n maxPriorityFeePerGas: highMaxTip,\n },\n isFixedFee: false,\n displayDecimals: 9,\n };\n}\n\nasync function suggestPriceOptions(\n provider: JsonRpcBatchInternal,\n): Promise<Pick<NetworkFees, 'low' | 'medium' | 'high'>> {\n const options: SuggestGasPriceOptionsResponse = await provider.send('eth_suggestPriceOptions', []);\n\n return {\n low: {\n maxFeePerGas: BigInt(options.slow.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.slow.maxPriorityFeePerGas),\n },\n medium: {\n maxFeePerGas: BigInt(options.normal.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.normal.maxPriorityFeePerGas),\n },\n high: {\n maxFeePerGas: BigInt(options.fast.maxFeePerGas),\n maxPriorityFeePerGas: BigInt(options.fast.maxPriorityFeePerGas),\n },\n };\n}\n\nasync function getGasMultiplier(proxyApiUrl: string, caipId?: string) {\n const defaultMultiplier = 1.5;\n if (!caipId) {\n return defaultMultiplier;\n }\n\n try {\n const respond = await fetch(`${proxyApiUrl}/gas/multiplier`);\n\n if (!respond.ok) {\n throw new Error(respond.statusText);\n }\n const multipliers = await respond.json();\n\n return multipliers[caipId] ?? multipliers.default;\n } catch (e) {\n return defaultMultiplier;\n }\n}\n","import type { NormalTx } from '@avalabs/core-etherscan-sdk';\nimport { TokenType, TransactionType, type NetworkToken, type Transaction } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\n\nexport const convertTransactionNormal = ({\n tx,\n networkToken,\n chainId,\n explorerUrl,\n address,\n}: {\n tx: NormalTx;\n networkToken: NetworkToken;\n chainId: number;\n explorerUrl: string;\n address: string;\n}): Transaction => {\n const isSender = tx.from.toLowerCase() === address.toLowerCase();\n const timestamp = parseInt(tx.timeStamp) * 1000;\n const decimals = networkToken.decimals;\n const amount = new TokenUnit(tx.value, networkToken.decimals, networkToken.symbol);\n const amountDisplayValue = amount.toDisplay();\n const txType = isSender ? TransactionType.SEND : TransactionType.RECEIVE;\n\n const { from, to, gasPrice, gasUsed, hash } = tx;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n\n return {\n isIncoming: !isSender,\n isOutgoing: isSender,\n isContractCall: isContractCall(tx),\n timestamp,\n hash,\n isSender,\n from,\n to,\n tokens: [\n {\n decimal: decimals.toString(),\n name: networkToken.name,\n symbol: networkToken.symbol,\n amount: amountDisplayValue,\n type: TokenType.NATIVE,\n },\n ],\n gasUsed,\n gasPrice,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n};\n\nfunction isContractCall(tx: NormalTx): boolean {\n return tx.input !== '0x';\n}\n","export function getExplorerAddressByNetwork(\n explorerUrl: string,\n hash: string,\n hashType: 'address' | 'tx' = 'tx',\n): string {\n return `${explorerUrl}/${hashType}/${hash}`;\n}\n","import type { Erc20Tx } from '@avalabs/core-etherscan-sdk';\nimport { TokenType, TransactionType, type Transaction } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\n\nexport function convertTransactionERC20({\n tx,\n address,\n explorerUrl,\n chainId,\n}: {\n tx: Erc20Tx;\n address: string;\n chainId: number;\n explorerUrl: string;\n}): Transaction {\n const isSender = tx.from.toLowerCase() === address.toLowerCase();\n const timestamp = parseInt(tx.timeStamp) * 1000;\n const amount = new TokenUnit(tx.value, Number(tx.tokenDecimal), tx.tokenSymbol);\n const amountDisplayValue = amount.toDisplay();\n const { from, to, gasPrice, gasUsed, hash, tokenDecimal, tokenName, tokenSymbol, contractAddress } = tx;\n const txType = isSender ? TransactionType.SEND : TransactionType.RECEIVE;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n\n return {\n isIncoming: !isSender,\n isOutgoing: isSender,\n isContractCall: false,\n timestamp,\n hash,\n isSender,\n from,\n to,\n tokens: [\n {\n decimal: tokenDecimal,\n name: tokenName,\n symbol: tokenSymbol,\n type: TokenType.ERC20,\n amount: amountDisplayValue,\n address: contractAddress,\n },\n ],\n gasUsed,\n gasPrice,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n}\n","import { convertTransactionNormal } from './convert-transaction-normal';\nimport { convertTransactionERC20 } from './convert-transaction-erc20';\nimport type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { getErc20Txs, getNormalTxs } from '@avalabs/core-etherscan-sdk';\n\ninterface EtherscanPagination {\n queries: ('normal' | 'erc20')[];\n page?: number;\n}\n\nexport const getTransactionFromEtherscan = async ({\n isTestnet,\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n}: {\n isTestnet?: boolean;\n networkToken: NetworkToken;\n explorerUrl: string;\n chainId: number;\n address: string;\n nextPageToken?: string;\n offset?: number;\n}): Promise<TransactionHistoryResponse> => {\n /*\n Using JSON for nextPageToken because this function is managing both the Normal\n and ERC20 queries. It encodes the current page and the queries that should be\n run. For example, if 'normal' has no more records to fetch then it will be\n excluded from the list and the JSON will be something like:\n { page: 3, queries: ['erc20'] }\n */\n const parsedPageToken = nextPageToken ? (JSON.parse(nextPageToken) as EtherscanPagination) : undefined;\n const page = parsedPageToken?.page || 1;\n const queries = parsedPageToken?.queries || ['normal', 'erc20'];\n\n const normalHist = (queries.includes('normal') ? await getNormalTxs(address, !isTestnet, { page, offset }) : []).map(\n (tx) => convertTransactionNormal({ tx, chainId, networkToken, explorerUrl, address }),\n );\n\n const erc20Hist = (\n queries.includes('erc20')\n ? await getErc20Txs(address, !isTestnet, undefined, {\n page,\n offset,\n })\n : []\n ).map((tx) =>\n convertTransactionERC20({\n tx,\n address,\n explorerUrl,\n chainId,\n }),\n );\n\n // Filter erc20 transactions from normal tx list\n const erc20TxHashes = erc20Hist.map((tx) => tx.hash);\n const filteredNormalTxs = normalHist.filter((tx) => {\n return !erc20TxHashes.includes(tx.hash);\n });\n\n const next: EtherscanPagination = { queries: [], page: page + 1 };\n if (normalHist.length) next.queries.push('normal');\n if (erc20Hist.length) next.queries.push('erc20');\n\n return {\n transactions: [...filteredNormalTxs, ...erc20Hist],\n nextPageToken: next.queries.length ? JSON.stringify(next) : '', // stop pagination\n };\n};\n","enum ChainId {\n ETHEREUM_HOMESTEAD = 1,\n ETHEREUM_TEST_RINKEBY = 4,\n ETHEREUM_TEST_GOERLY = 5,\n ETHEREUM_TEST_SEPOLIA = 11155111,\n}\n\nexport const isEthereumChainId = (chainId: number): boolean => {\n return (\n ChainId.ETHEREUM_HOMESTEAD === chainId ||\n ChainId.ETHEREUM_TEST_GOERLY === chainId ||\n ChainId.ETHEREUM_TEST_RINKEBY === chainId ||\n ChainId.ETHEREUM_TEST_SEPOLIA === chainId\n );\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TokenType, TransactionType, type TxToken } from '@avalabs/vm-module-types';\nimport startCase from 'lodash.startcase';\n\nexport const getTxType = (\n { nativeTransaction, erc20Transfers, erc721Transfers, erc1155Transfers }: TransactionDetails,\n userAddress: string,\n tokens: TxToken[],\n): TransactionType => {\n const nativeOnly = !erc20Transfers && !erc721Transfers && !erc1155Transfers;\n const method = parseRawMethod(nativeTransaction.method?.methodName);\n\n const address = userAddress.toLowerCase();\n\n const isSwap = method.toLowerCase().includes('swap');\n const isNativeSend = nativeOnly && nativeTransaction.from.address.toLowerCase() === address;\n const isNativeReceive = nativeOnly && nativeTransaction.to.address.toLowerCase() === address;\n const isNFTPurchase = method === 'Market Buy Orders With Eth' || method === 'Buy NFT';\n const isApprove = method === 'Approve';\n const isTransfer = method.toLowerCase().includes('transfer');\n const isAirdrop = method.toLowerCase().includes('airdrop');\n const isUnwrap = method.toLowerCase().includes('unwrap');\n const isFillOrder = method === 'Fill Order';\n const isNFTSend = !!tokens[0] && isNFT(tokens[0].type) && tokens[0].from?.address.toLowerCase() === address;\n const isNFTReceive = !!tokens[0] && isNFT(tokens[0].type) && tokens[0].to?.address.toLowerCase() === address;\n\n if (isSwap) return TransactionType.SWAP;\n if (isNativeSend) return TransactionType.SEND;\n if (isNativeReceive) return TransactionType.RECEIVE;\n if (isNFTPurchase) return TransactionType.NFT_BUY;\n if (isApprove) return TransactionType.APPROVE;\n if (isNFTSend) return TransactionType.NFT_SEND;\n if (isNFTReceive) return TransactionType.NFT_RECEIVE;\n if (isTransfer) return TransactionType.TRANSFER;\n if (isAirdrop) return TransactionType.AIRDROP;\n if (isUnwrap) return TransactionType.UNWRAP;\n if (isFillOrder) return TransactionType.FILL_ORDER;\n return TransactionType.UNKNOWN;\n};\n\nfunction isNFT(tokenType: TokenType) {\n return tokenType === TokenType.ERC721 || tokenType === TokenType.ERC1155;\n}\n\nconst parseRawMethod = (method = ''): string => {\n if (method.includes('(')) {\n return startCase(method.split('(', 1)[0]);\n }\n return method;\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TransactionType } from '@avalabs/vm-module-types';\n\nexport const getSenderInfo = (\n txType: TransactionType,\n { nativeTransaction, erc20Transfers, erc721Transfers }: TransactionDetails,\n address: string,\n): { isOutgoing: boolean; isIncoming: boolean; isSender: boolean; from: string; to: string } => {\n const isTransfer = txType === TransactionType.TRANSFER;\n const isNativeSend = txType === TransactionType.SEND;\n const isNativeReceive = txType === TransactionType.RECEIVE;\n let from = nativeTransaction?.from?.address;\n let to = nativeTransaction?.to?.address;\n\n // Until multi tokens transaction is supported in UI, using from and to of the only token is helpful for UI\n if (isTransfer && erc20Transfers && erc20Transfers[0]) {\n from = erc20Transfers[0].from.address;\n to = erc20Transfers[0].to.address;\n }\n\n if (isTransfer && erc721Transfers && erc721Transfers[0]) {\n from = erc721Transfers[0].from.address;\n to = erc721Transfers[0].to.address;\n }\n\n const isOutgoing = isNativeSend || (isTransfer && from.toLowerCase() === address.toLowerCase());\n const isIncoming = isNativeReceive || (isTransfer && to.toLowerCase() === address.toLowerCase());\n\n const isSender = from === address;\n\n return {\n isOutgoing,\n isIncoming,\n isSender,\n from,\n to,\n };\n};\n","import type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport type { TxToken, NetworkToken } from '@avalabs/vm-module-types';\nimport { TokenType } from '@avalabs/vm-module-types';\nimport { resolve } from '../../utils/resolve';\nimport { getNftMetadata } from './get-nft-metadata';\nimport { getSmallImageForNFT } from '../../utils/get-small-image-for-nft';\nimport { ipfsResolverWithFallback } from '../../utils/ipfs-resolver-with-fallback';\n\nexport const getTokens = async (\n { nativeTransaction, erc20Transfers, erc721Transfers, erc1155Transfers }: TransactionDetails,\n networkToken: NetworkToken,\n): Promise<TxToken[]> => {\n const result: TxToken[] = [];\n\n if (nativeTransaction.value !== '0') {\n const decimal = networkToken.decimals;\n const amount = new TokenUnit(nativeTransaction.value, networkToken.decimals, networkToken.symbol);\n const amountDisplayValue = amount.toDisplay();\n result.push({\n decimal: decimal.toString(),\n name: networkToken.name,\n symbol: networkToken.symbol,\n amount: amountDisplayValue,\n from: nativeTransaction.from,\n to: nativeTransaction.to,\n type: TokenType.NATIVE,\n });\n }\n\n erc20Transfers?.forEach((erc20Transfer) => {\n const decimals = erc20Transfer.erc20Token.decimals;\n const amount = new TokenUnit(erc20Transfer.value, decimals, erc20Transfer.erc20Token.symbol);\n const amountDisplayValue = amount.toDisplay();\n\n result.push({\n decimal: decimals.toString(),\n address: erc20Transfer.erc20Token.address,\n name: erc20Transfer.erc20Token.name,\n symbol: erc20Transfer.erc20Token.symbol,\n amount: amountDisplayValue,\n from: erc20Transfer.from,\n to: erc20Transfer.to,\n imageUri: erc20Transfer.erc20Token.logoUri,\n type: TokenType.ERC20,\n });\n });\n\n if (erc721Transfers) {\n await Promise.allSettled(\n erc721Transfers.map(async (erc721Transfer) => {\n const token = erc721Transfer.erc721Token;\n const imageUri = await getImageUri(token.tokenUri, token.metadata.imageUri);\n\n result.push({\n name: erc721Transfer.erc721Token.name,\n symbol: erc721Transfer.erc721Token.symbol,\n address: erc721Transfer.erc721Token.address,\n amount: '1',\n imageUri,\n from: erc721Transfer.from,\n to: erc721Transfer.to,\n collectableTokenId: erc721Transfer.erc721Token.tokenId,\n type: TokenType.ERC721,\n });\n }),\n );\n }\n\n if (erc1155Transfers) {\n await Promise.allSettled(\n erc1155Transfers.map(async (erc1155Transfer) => {\n const token = erc1155Transfer.erc1155Token;\n const imageUri = await getImageUri(token.tokenUri, token.metadata.imageUri);\n\n result.push({\n name: erc1155Transfer.erc1155Token.metadata.name ?? '',\n symbol: erc1155Transfer.erc1155Token.metadata.symbol ?? '',\n address: erc1155Transfer.erc1155Token.address,\n amount: erc1155Transfer.value,\n imageUri,\n from: erc1155Transfer.from,\n to: erc1155Transfer.to,\n collectableTokenId: erc1155Transfer.erc1155Token.tokenId,\n type: TokenType.ERC1155,\n });\n }),\n );\n }\n\n return result;\n};\n\nconst getImageUri = async (tokenUri: string, imageUri?: string): Promise<string> => {\n if (imageUri) {\n if (imageUri.startsWith('ipfs://')) {\n return ipfsResolverWithFallback(imageUri);\n } else {\n return imageUri;\n }\n } else {\n const [metadata, error] = await resolve(getNftMetadata(tokenUri));\n if (error) {\n return '';\n } else {\n return metadata.image ? getSmallImageForNFT(metadata.image) : '';\n }\n }\n};\n","export async function resolve<T = unknown>(promise: Promise<T>) {\n try {\n return promise.then((res) => [res, null]).catch((err) => [null, err]);\n } catch (err) {\n return Promise.resolve([null, err]);\n }\n}\n","import { ipfsResolver } from '@avalabs/core-utils-sdk';\n\nexport const CLOUDFLARE_IPFS_URL = 'https://ipfs.io';\n\nexport function ipfsResolverWithFallback(\n sourceUrl: string | undefined,\n desiredGatewayPrefix: string = CLOUDFLARE_IPFS_URL,\n) {\n if (!sourceUrl) {\n return '';\n }\n\n try {\n return ipfsResolver(sourceUrl, desiredGatewayPrefix);\n } catch {\n return sourceUrl;\n }\n}\n","import { ipfsResolverWithFallback } from '../../utils/ipfs-resolver-with-fallback';\n\ninterface NftMetadata {\n attributes?: string;\n name?: string;\n image?: string;\n description?: string;\n}\n\nasync function fetchWithTimeout(uri: string, timeout = 5000) {\n const controller = new AbortController();\n setTimeout(() => controller.abort(), timeout);\n\n return fetch(uri, { signal: controller.signal });\n}\n\nexport async function getNftMetadata(tokenUri: string) {\n let data: NftMetadata = {};\n if (!tokenUri) {\n return {};\n } else if (tokenUri.startsWith('data:application/json;base64,')) {\n const value = tokenUri.substring(29);\n try {\n const json = Buffer.from(value, 'base64').toString();\n data = JSON.parse(json);\n } catch {\n data = {};\n }\n } else {\n data = await fetchWithTimeout(ipfsResolverWithFallback(tokenUri))\n .then((r) => r.json())\n .catch(() => ({}));\n }\n return data;\n}\n","import { ipfsResolverWithFallback } from './ipfs-resolver-with-fallback';\n\nconst COVALENT_IMG_SIZER = 'https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image';\n\n/**\n * Covalent has an on the fly image resizer, it resolves image urls then resizes the image.\n *\n * This allows us to request smaller images depending on the UI needs\n *\n * @param imgUrl the url of the image to convert to size\n * @returns The url to the image which is sized at the time of request\n */\nexport function getSmallImageForNFT(imgUrl: string, imageSize: '256' | '512' | '1024' = '256') {\n const url = ipfsResolverWithFallback(imgUrl);\n return `${COVALENT_IMG_SIZER}/width=${imageSize},fit/${url}`;\n}\n","import type { z } from 'zod';\nimport { TransactionType } from '@avalabs/vm-module-types';\n\nimport type { transactionSchema } from './utils/transaction-schema';\nimport type { transactionArraySchema } from './handlers/eth-send-transaction-batch/schema';\n\nexport const NonContractCallTypes = [TransactionType.SEND, TransactionType.RECEIVE, TransactionType.TRANSFER];\n\nexport type TransactionParams = z.infer<typeof transactionSchema>;\n\nexport type RequiredBy<T, K extends keyof T> = T & Required<Pick<T, K>>;\n\nexport type TransactionBatchParams = z.infer<typeof transactionArraySchema>;\n\nexport enum ERC20TransactionType {\n TOTAL_SUPPLY = 'totalSupply',\n BALANCE_OF = 'balanceOf',\n TRANSFER = 'transfer',\n TRANSFER_FROM = 'transferFrom',\n APPROVE = 'approve',\n ALLOWANCE = 'allowance',\n}\n","import type { Transaction, NetworkToken } from '@avalabs/vm-module-types';\nimport { getTxType } from './get-tx-type';\nimport { getSenderInfo } from './get-sender-info';\nimport { getTokens } from './get-tokens';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\nimport type { TransactionDetails } from '@avalabs/glacier-sdk';\nimport { NonContractCallTypes } from '../../../../types';\n\ntype ConvertTransactionParams = {\n transactions: TransactionDetails;\n explorerUrl: string;\n networkToken: NetworkToken;\n chainId: number;\n address: string;\n};\n\nexport const convertTransaction = async ({\n transactions,\n explorerUrl,\n networkToken,\n chainId,\n address,\n}: ConvertTransactionParams): Promise<Transaction> => {\n const tokens = await getTokens(transactions, networkToken);\n const txType = getTxType(transactions, address, tokens);\n const { isOutgoing, isIncoming, isSender, from, to } = getSenderInfo(txType, transactions, address);\n const { blockTimestamp, txHash: hash, gasPrice, gasUsed } = transactions.nativeTransaction;\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, hash);\n const isContractCall = !NonContractCallTypes.includes(txType);\n\n return {\n isContractCall,\n isIncoming,\n isOutgoing,\n isSender,\n timestamp: blockTimestamp * 1000, // s to ms\n hash,\n from,\n to,\n tokens,\n gasPrice,\n gasUsed,\n chainId: chainId.toString(),\n txType,\n explorerLink,\n };\n};\n","import type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { convertTransaction } from './convert-transaction';\nimport type { EvmGlacierService } from '../../../../services/glacier-service/glacier-service';\n\nexport const getTransactionsFromGlacier = async ({\n chainId,\n explorerUrl,\n networkToken,\n address,\n nextPageToken,\n offset,\n glacierService,\n}: {\n chainId: number;\n explorerUrl: string;\n networkToken: NetworkToken;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierService: EvmGlacierService;\n}): Promise<TransactionHistoryResponse> => {\n try {\n const response = await glacierService.listTransactions({\n chainId: chainId.toString(),\n address,\n pageToken: nextPageToken,\n pageSize: offset,\n });\n\n const convertedTxs = await Promise.all(\n response.transactions\n .filter(\n // Currently not showing failed tx\n (tranasaction) => tranasaction.nativeTransaction.txStatus === '1',\n )\n .map((transactions) =>\n convertTransaction({\n transactions,\n explorerUrl,\n networkToken,\n chainId,\n address,\n }).then((tx) => tx),\n ),\n );\n\n const transactions = convertedTxs.filter(\n // Filtering txs with 0 value since there is no change in balance\n (transaction) => transaction.tokens.find((token) => Number(token.amount) !== 0),\n );\n\n return {\n transactions,\n nextPageToken: response.nextPageToken,\n };\n } catch {\n return {\n transactions: [],\n nextPageToken: '',\n };\n }\n};\n","import { getTransactionFromEtherscan } from './converters/etherscan-transaction-converter/get-transaction-from-etherscan';\nimport { isEthereumChainId } from './utils/is-ethereum-chain-id';\nimport type { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { getTransactionsFromGlacier } from './converters/evm-transaction-converter/get-transactions-from-glacier';\nimport type { EvmGlacierService } from '../../services/glacier-service/glacier-service';\n\nexport const getTransactionHistory = async ({\n chainId,\n isTestnet,\n networkToken,\n explorerUrl,\n address,\n nextPageToken,\n offset,\n glacierService,\n}: {\n chainId: number;\n isTestnet?: boolean;\n networkToken: NetworkToken;\n explorerUrl: string;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierService: EvmGlacierService;\n}): Promise<TransactionHistoryResponse> => {\n if (isEthereumChainId(chainId)) {\n return getTransactionFromEtherscan({\n isTestnet,\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n });\n }\n\n const isHealthy = glacierService.isHealthy();\n if (!isHealthy) {\n return {\n transactions: [],\n nextPageToken: '',\n };\n }\n\n return getTransactionsFromGlacier({\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n glacierService,\n });\n};\n","{\n \"name\": \"EVM\",\n \"description\": \"\",\n \"version\": \"0.0.1\",\n \"sources\": {\n \"module\": {\n \"checksum\": \"\",\n \"location\": {\n \"npm\": {\n \"filePath\": \"dist/index.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n },\n \"provider\": {\n \"checksum\": \"\",\n \"location\": {\n \"npm\": {\n \"filePath\": \"dist/provider/index.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n }\n },\n \"network\": {\n \"chainIds\": [],\n \"namespaces\": [\"eip155\"]\n },\n \"cointype\": \"60\",\n \"permissions\": {\n \"rpc\": {\n \"dapps\": true,\n \"methods\": [\n \"personal_sign\",\n \"eth_sendTransaction\",\n \"eth_sendTransactionBatch\",\n \"eth_*\",\n \"web3_clientVersion\",\n \"web3_sha3\",\n \"net_version\",\n \"net_peerCount\",\n \"net_listening\"\n ],\n \"nonRestrictedMethods\": [\n \"eth_requestAccounts\",\n \"eth_decrypt\",\n \"eth_getEncryptionPublicKey\",\n \"eth_subscribe\",\n \"eth_unsubscribe\",\n \"eth_blockNumber\",\n \"eth_call\",\n \"eth_chainId\",\n \"eth_coinbase\",\n \"eth_estimateGas\",\n \"eth_feeHistory\",\n \"eth_gasPrice\",\n \"eth_getBalance\",\n \"eth_getBlockByHash\",\n \"eth_getBlockByNumber\",\n \"eth_getBlockTransactionCountByHash\",\n \"eth_getBlockTransactionCountByNumber\",\n \"eth_getCode\",\n \"eth_getFilterChanges\",\n \"eth_getFilterLogs\",\n \"eth_getLogs\",\n \"eth_getProof\",\n \"eth_getStorageAt\",\n \"eth_getTransactionByBlockHashAndIndex\",\n \"eth_getTransactionByBlockNumberAndIndex\",\n \"eth_getTransactionByHash\",\n \"eth_getTransactionCount\",\n \"eth_getTransactionReceipt\",\n \"eth_getUncleCountByBlockHash\",\n \"eth_getUncleCountByBlockNumber\",\n \"eth_newBlockFilter\",\n \"eth_newFilter\",\n \"eth_newPendingTransactionFilter\",\n \"eth_syncing\",\n \"eth_uninstallFilter\",\n \"net_version\",\n \"net_peerCount\",\n \"net_listening\",\n \"web3_sha3\",\n \"web3_clientVersion\"\n ]\n }\n },\n \"manifestVersion\": \"0.1\"\n}\n","import { type Network, type RpcRequest, type ApprovalController } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nimport { getNonce } from '../../utils/get-nonce';\nimport { getProvider } from '../../utils/get-provider';\nimport { getTxUpdater } from '../../utils/evm-tx-updater';\nimport { estimateGasLimit } from '../../utils/estimate-gas-limit';\nimport { buildTxApprovalRequest } from '../../utils/build-tx-approval-request';\nimport { simulateTransaction } from '../../utils/process-transaction-simulation';\n\nimport { parseRequestParams } from './schema';\nimport { waitForTransactionReceipt } from '../../utils/wait-for-transaction-receipt';\nimport { getTxHash } from '../../utils/get-tx-hash';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSendTransaction = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n blockaid: Blockaid;\n}) => {\n const { params } = request;\n\n // validate params\n const { data, error, success } = parseRequestParams(params);\n\n if (!success) {\n console.error('invalid params', error);\n return {\n error: rpcErrors.invalidParams({ message: 'Transaction params are invalid', data: { cause: error } }),\n };\n }\n\n const [transaction] = data;\n\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n // calculate gas limit if not provided/invalid\n if (!transaction.gas || Number(transaction.gas) < 0) {\n try {\n const gasLimit = await estimateGasLimit({\n transactionParams: {\n from: transaction.from,\n to: transaction.to,\n data: transaction.data,\n value: transaction.value,\n accessList: transaction.accessList,\n },\n provider,\n });\n\n transaction.gas = '0x' + gasLimit.toString(16);\n } catch (error) {\n return {\n error: rpcErrors.internal('Unable to calculate gas limit'),\n };\n }\n }\n\n // calculate nonce if not provided\n if (!transaction.nonce) {\n try {\n const nonce = await getNonce({\n from: transaction.from,\n provider,\n });\n transaction.nonce = String(nonce);\n } catch (error) {\n return {\n error: rpcErrors.internal('Unable to calculate nonce'),\n };\n }\n }\n\n const scan = await simulateTransaction({\n rpcMethod: request.method,\n chainId: network.chainId,\n params: transaction,\n dAppUrl: request.dappInfo.url,\n provider,\n blockaid,\n });\n\n const { displayData, signingData } = buildTxApprovalRequest(request, network, transaction, scan);\n\n const { updateTx, cleanup } = getTxUpdater(request.requestId, signingData, displayData);\n // prompt user for approval\n const response = await approvalController.requestApproval({ request, displayData, signingData, updateTx });\n\n cleanup();\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n let txHash;\n\n try {\n txHash = await getTxHash(provider, response);\n } catch (error) {\n return {\n error: rpcErrors.internal({ message: 'Unable to get transaction hash', data: { cause: error } }),\n };\n }\n\n waitForTransactionReceipt({\n explorerUrl: network.explorerUrl ?? '',\n provider,\n txHash,\n onTransactionPending: approvalController.onTransactionPending,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n request,\n });\n\n return { result: txHash };\n};\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nexport const getNonce = async ({\n from,\n provider,\n}: {\n from: string;\n provider: JsonRpcBatchInternal;\n}): Promise<number> => {\n return provider.getTransactionCount(from);\n};\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport type { DisplayData, EvmTxUpdateFn, SigningData_EthSendTx, TokenApprovals } from '@avalabs/vm-module-types';\n\nimport { ERC20TransactionType } from '../types';\nimport { parseERC20TransactionType } from './parse-erc20-transaction-type';\nimport { encodeApprovalLimit } from './encode-erc20-approval';\n\nconst requests = new Map<string, { displayData: DisplayData; signingData: SigningData_EthSendTx }>();\n\nexport const getTxUpdater = (\n requestId: string,\n signingData: SigningData_EthSendTx,\n displayData: DisplayData,\n): { updateTx: EvmTxUpdateFn; cleanup: () => void } => {\n requests.set(requestId, { signingData, displayData });\n\n return {\n updateTx: ({ maxFeeRate, maxTipRate, approvalLimit, gasLimit }) => {\n const request = requests.get(requestId);\n\n if (!request) {\n throw rpcErrors.resourceNotFound();\n }\n\n const { signingData, displayData } = request;\n\n const newSigningData = {\n ...signingData,\n data: {\n ...signingData.data,\n gasLimit: gasLimit ?? signingData.data.gasLimit,\n maxFeePerGas: maxFeeRate ?? signingData.data.maxFeePerGas,\n maxPriorityFeePerGas: maxTipRate ?? signingData.data.maxPriorityFeePerGas,\n },\n };\n\n const newDisplayData = { ...displayData };\n\n if (typeof approvalLimit === 'string') {\n if (!approvalLimit.startsWith('0x')) {\n throw rpcErrors.invalidInput('Expected approvalLimit to be a hexadecimal number (0x-prefixed)');\n }\n\n const tokenApprovals = displayData.tokenApprovals;\n\n ensureTokenApprovalCanBeEdited(tokenApprovals);\n ensureApproveWasCalled(signingData.data.data);\n\n const approval = tokenApprovals.approvals[0]!;\n\n newSigningData.data.data = encodeApprovalLimit(approval.token.address, approval.spenderAddress, approvalLimit);\n newDisplayData.tokenApprovals = {\n approvals: [\n {\n ...approval,\n value: approvalLimit,\n },\n ],\n isEditable: true,\n };\n }\n\n const updatedRequest = { signingData: newSigningData, displayData: newDisplayData };\n\n requests.set(requestId, updatedRequest);\n\n return updatedRequest;\n },\n cleanup: () => requests.delete(requestId),\n };\n};\n\nfunction ensureTokenApprovalCanBeEdited(\n tokenApprovals?: TokenApprovals,\n): asserts tokenApprovals is TokenApprovals & { isEditable: true } {\n if (!tokenApprovals || !tokenApprovals.isEditable || tokenApprovals.approvals.length !== 1) {\n throw rpcErrors.invalidInput(\n 'Cannot edit the token approval for this request. Please start a new request instead.',\n );\n }\n}\n\nfunction ensureApproveWasCalled(txData?: string | null) {\n const previousMethod = !txData || txData === '0x' ? null : parseERC20TransactionType({ data: txData ?? undefined });\n\n // If original tx was not setting any approvals, do not allow it to be set later on.\n if (previousMethod !== ERC20TransactionType.APPROVE) {\n throw rpcErrors.invalidInput(\n 'Cannot change invoked method for requests in progress. Please start a new request instead.',\n );\n }\n}\n","import { Interface } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { ERC20TransactionType } from '../types';\n\nexport const parseERC20TransactionType = (transaction: {\n data?: string;\n value?: string;\n}): ERC20TransactionType | undefined => {\n if (!transaction.data) {\n return undefined;\n }\n\n try {\n const contractInterface = new Interface(ERC20.abi);\n\n const description = contractInterface.parseTransaction({\n data: transaction.data,\n value: transaction.value,\n });\n\n const functionName = description?.name ?? description?.fragment?.name;\n\n if (functionName && isERC20TransactionType(functionName)) {\n return functionName;\n }\n\n return undefined;\n } catch (e) {\n return undefined;\n }\n};\n\nfunction isERC20TransactionType(value: string): value is ERC20TransactionType {\n return Object.values(ERC20TransactionType).includes(value as ERC20TransactionType);\n}\n","import { ethers } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { ERC20TransactionType } from '../types';\n\nexport function encodeApprovalLimit(tokenAddress: string, spenderAddress: string, limit: string) {\n const contract = new ethers.Contract(tokenAddress, ERC20.abi);\n\n return contract.interface.encodeFunctionData(ERC20TransactionType.APPROVE, [spenderAddress, limit]);\n}\n","import { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport type { AccessList, BigNumberish } from 'ethers';\n\nexport const estimateGasLimit = async ({\n transactionParams: { from, to, data, value, accessList },\n provider,\n}: {\n transactionParams: {\n from: string;\n to?: string;\n data?: string;\n value?: BigNumberish;\n accessList?: AccessList;\n };\n provider: JsonRpcBatchInternal;\n}): Promise<number> => {\n const nonce = await provider.getTransactionCount(from);\n\n return Number(\n await provider.estimateGas({\n from,\n to,\n nonce,\n data,\n value,\n accessList,\n }),\n );\n};\n","import {\n RpcMethod,\n type DetailItem,\n type DisplayData,\n type Network,\n type RpcRequest,\n type SigningData,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\n\nimport { addressItem, linkItem, networkItem } from '@internal/utils/src/utils/detail-item';\n\nimport { ERC20TransactionType } from '../types';\nimport type { TransactionParams } from './transaction-schema';\nimport { parseERC20TransactionType } from './parse-erc20-transaction-type';\n\nexport const buildTxApprovalRequest = (\n request: RpcRequest,\n network: Network,\n transaction: TransactionParams,\n { isSimulationSuccessful, balanceChange, tokenApprovals, alert }: TransactionSimulationResult,\n) => {\n const { dappInfo } = request;\n const transactionType = parseERC20TransactionType(transaction);\n\n // generate display and signing data\n let title = 'Do you approve this transaction?';\n if (transactionType === ERC20TransactionType.APPROVE) {\n title = 'Do you approve this spend limit?';\n }\n\n const transactionDetails: DetailItem[] = [\n addressItem('Account', transaction.from),\n networkItem('Network', {\n name: network.chainName,\n logoUri: network.logoUri,\n }),\n linkItem('Website', dappInfo),\n ];\n\n if (transaction.to) {\n transactionDetails.push(addressItem('Contract', transaction.to));\n }\n\n const displayData: DisplayData = {\n title,\n details: [\n {\n title: 'Transaction Details',\n items: transactionDetails,\n },\n ],\n networkFeeSelector: true,\n alert,\n balanceChange,\n tokenApprovals,\n isSimulationSuccessful,\n };\n\n const signingData: SigningData = {\n type: RpcMethod.ETH_SEND_TRANSACTION,\n account: transaction.from,\n data: {\n type: 2, // hardcoding to 2 for now as we only support EIP-1559\n nonce: Number(transaction.nonce),\n gasLimit: Number(transaction.gas),\n maxFeePerGas: transaction.maxFeePerGas,\n maxPriorityFeePerGas: transaction.maxPriorityFeePerGas,\n to: transaction.to,\n from: transaction.from,\n data: transaction.data,\n value: transaction.value,\n chainId: transaction.chainId ?? network.chainId,\n accessList: transaction.accessList,\n },\n };\n\n return {\n displayData,\n signingData,\n };\n};\n","import Blockaid from '@blockaid/client';\nimport type { TransactionParams } from '../types';\nimport {\n type NetworkContractToken,\n type NetworkToken,\n TokenType,\n AlertType,\n type Alert,\n type BalanceChange,\n type TokenApproval,\n type TokenDiff,\n type TokenDiffItem,\n type TokenApprovals,\n type RpcRequest,\n RpcMethod,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue, numberToBN } from '@avalabs/core-utils-sdk';\nimport { isHexString, MaxUint256 } from 'ethers';\nimport { scanJsonRpc, scanTransaction } from './scan-transaction';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { parseWithErc20Abi } from './parse-erc20-tx';\nimport { hasToField } from './type-utils';\nimport { transactionAlerts } from './transaction-alerts';\n\ntype Erc20ExposureTrace = Blockaid.Evm.AccountSummary.Erc20ExposureTrace;\ntype Erc721ExposureTrace = Blockaid.Evm.AccountSummary.Erc721ExposureTrace;\ntype Erc1155ExposureTrace = Blockaid.Evm.AccountSummary.Erc1155ExposureTrace;\n\n/*\n * Although in the type definition they don't specify it, but for traces they are returning the asset as well:\n * https://docs.blockaid.io/changelog/november-12-2024#account-traces\n * */\ntype ExposureTrace =\n | (Erc20ExposureTrace & { asset: Blockaid.Evm.Erc20TokenDetails })\n | (Erc721ExposureTrace & { asset: Blockaid.Evm.Erc721TokenDetails })\n | (Erc1155ExposureTrace & { asset: Blockaid.Evm.Erc1155TokenDetails });\ntype Trace = Blockaid.Evm.AccountSummary['traces']['0'];\nexport type AssetDiffs = Blockaid.Evm.AccountSummary['assets_diffs'];\n\nexport const simulateTransaction = async ({\n rpcMethod,\n dAppUrl,\n params,\n chainId,\n provider,\n blockaid,\n}: {\n rpcMethod: RpcMethod;\n dAppUrl?: string;\n params: TransactionParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n blockaid: Blockaid;\n}) => {\n let simulationResult: Pick<Blockaid.TransactionScanResponse, 'simulation' | 'validation'> | undefined;\n\n try {\n simulationResult = await scanTransaction({\n chainId,\n params,\n domain: dAppUrl,\n blockaid,\n });\n } catch (error) {\n console.error('simulateTransaction error', error);\n }\n\n return processTransactionSimulation({ rpcMethod, params, chainId, provider, simulationResult });\n};\n\nexport const processTransactionSimulation = async ({\n rpcMethod,\n params,\n chainId,\n provider,\n simulationResult,\n}: {\n rpcMethod: RpcMethod;\n params: TransactionParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n simulationResult?: Pick<Blockaid.TransactionScanResponse, 'simulation' | 'validation' | 'gas_estimation'>;\n}): Promise<TransactionSimulationResult> => {\n let alert: Alert | undefined;\n let balanceChange: BalanceChange | undefined;\n let tokenApprovals: TokenApprovals | undefined;\n let isSimulationSuccessful = false;\n let estimatedGasLimit: number | undefined;\n\n if (simulationResult) {\n const { validation, simulation, gas_estimation: gasEstimation } = simulationResult;\n\n if (!validation || validation.result_type === 'Error' || validation.result_type === 'Warning') {\n alert = transactionAlerts[AlertType.WARNING];\n } else if (validation.result_type === 'Malicious') {\n alert = transactionAlerts[AlertType.DANGER];\n }\n\n if (simulation?.status === 'Success') {\n isSimulationSuccessful = true;\n tokenApprovals = processTokenApprovals(rpcMethod, simulation.account_summary);\n balanceChange = processBalanceChange(simulation.account_summary.assets_diffs);\n }\n\n if (gasEstimation?.status === 'Success') {\n estimatedGasLimit = Number(gasEstimation.estimate);\n }\n }\n\n // If debank parsing failed, check if toAddress is a known ERC20\n if (!isSimulationSuccessful && hasToField(params)) {\n const erc20ParseResult = await parseWithErc20Abi(params, chainId, provider);\n balanceChange = erc20ParseResult.balanceChange;\n tokenApprovals = erc20ParseResult.tokenApprovals;\n }\n\n return { alert, balanceChange, tokenApprovals, isSimulationSuccessful, estimatedGasLimit };\n};\n\nconst isExposureTrace = (trace: Trace): trace is ExposureTrace => {\n return (trace as ExposureTrace).trace_type === 'ExposureTrace';\n};\n\nconst normalizeAddresses = (spender: string, assetAddress: string): string =>\n `${spender.toLowerCase()}.${assetAddress.toLowerCase()}`;\n\nconst mapExposureTracesToSpenderAsset = (traces: Trace[]): Record<string, ExposureTrace> => {\n if (traces === undefined || traces.length === 0) {\n return {};\n }\n\n return traces.reduce(\n (accumulator, trace) => {\n if (!isExposureTrace(trace)) {\n return accumulator;\n }\n return {\n [normalizeAddresses(trace.spender, trace.asset.address)]: trace,\n };\n },\n {} as Record<string, ExposureTrace>,\n );\n};\n\nconst processTokenApprovals = (\n rpcMethod: RpcMethod,\n accountSummary: Blockaid.Evm.AccountSummary,\n): TokenApprovals | undefined => {\n const { traces } = accountSummary;\n\n const mappedExposureTraces = mapExposureTracesToSpenderAsset(traces);\n\n const approvals = Object.entries(mappedExposureTraces)\n .map(([_, trace]) => {\n const token = convertAssetToNetworkContractToken(trace.asset);\n if (!token) {\n return;\n }\n\n const tokenApproval: TokenApproval = {\n token,\n spenderAddress: trace.spender,\n logoUri: token.logoUri,\n };\n\n if (trace.type === 'ERC20ExposureTrace') {\n const {\n exposed: { raw_value, usd_price },\n } = trace as Erc20ExposureTrace;\n\n tokenApproval.value = raw_value;\n tokenApproval.usdPrice = `${usd_price}`;\n }\n\n if (trace.type === 'ERC721ExposureTrace') {\n // When dApp attempts to call setApprovalForAll(), the \"exposed\" field is not present.\n // However, the result of such transaction is that the spender gets approval for all\n // tokens in the NFT collection, so we treat it the same as \"Unlimited\" approval for ERC-20.\n if (!trace.exposed) {\n tokenApproval.value = `0x${MaxUint256.toString(16)}`;\n } else {\n const { usd_price, amount } = trace.exposed;\n\n tokenApproval.value = `${amount}`;\n tokenApproval.usdPrice = `${usd_price}`;\n }\n }\n\n return tokenApproval;\n })\n .filter(Boolean) as TokenApproval[];\n\n if (approvals.length === 0) {\n return undefined;\n }\n\n const isEditable =\n approvals.length === 1 &&\n approvals[0]?.token.type === TokenType.ERC20 &&\n rpcMethod === RpcMethod.ETH_SEND_TRANSACTION;\n\n return { isEditable, approvals };\n};\n\nexport const processBalanceChange = (assetDiffs: AssetDiffs): BalanceChange | undefined => {\n const ins = processAssetDiffs(assetDiffs, 'in');\n const outs = processAssetDiffs(assetDiffs, 'out');\n\n if (ins.length === 0 && outs.length === 0) {\n return undefined;\n }\n\n return { ins, outs };\n};\n\nconst processAssetDiffs = (assetDiffs: AssetDiffs, type: 'in' | 'out'): TokenDiff[] => {\n return (\n assetDiffs\n .filter((assetDiff) => assetDiff[type].length > 0)\n // sort asset diffs by length of in/out array\n // this is done to ensure that the token with multiple in/out values are displayed last,\n // to put them in groups with appropriate UI(i.e. accordion), after single in/out tokens\n .sort((a, b) => a[type].length - b[type].length)\n .map((assetDiff) => {\n const asset = assetDiff.asset;\n // convert blockaid asset to network token\n const token: NetworkToken | NetworkContractToken | undefined =\n 'address' in asset ? convertAssetToNetworkContractToken(asset) : convertNativeAssetToToken(asset);\n if (!token) {\n return undefined;\n }\n\n const items = assetDiff[type]\n .map((diff) => {\n let displayValue;\n if ('value' in diff && diff.value) {\n if ('decimals' in token) {\n const valueBN = numberToBN(diff.value, token.decimals);\n displayValue = balanceToDisplayValue(valueBN, token.decimals);\n } else if (isHexString(diff.value)) {\n // for some token (like ERC1155) blockaid returns value in hex format\n displayValue = parseInt(diff.value, 16).toString();\n }\n } else if ('type' in token && token.type === TokenType.ERC721) {\n // for ERC721 type token, we just display 1 to indicate that a single NFT will be transferred\n displayValue = '1';\n }\n\n return displayValue ? { displayValue, usdPrice: diff.usd_price } : undefined;\n })\n .filter((x): x is TokenDiffItem => x !== undefined);\n\n return { token, items };\n })\n .filter((x): x is TokenDiff => x !== undefined)\n );\n};\n\nconst convertAssetToNetworkContractToken = (\n asset:\n | Blockaid.Erc20TokenDetails\n | Blockaid.Erc1155TokenDetails\n | Blockaid.Erc721TokenDetails\n | Blockaid.NonercTokenDetails,\n): NetworkContractToken | undefined => {\n let token: NetworkContractToken | undefined;\n if (asset.type === 'ERC20') {\n token = {\n type: TokenType.ERC20,\n address: asset.address,\n decimals: asset.decimals,\n name: asset.name ?? asset.symbol ?? '',\n symbol: asset.symbol ?? '',\n logoUri: asset.logo_url,\n };\n } else if (asset.type === 'ERC1155') {\n token = {\n type: TokenType.ERC1155,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n } else if (asset.type === 'ERC721') {\n token = {\n type: TokenType.ERC721,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n } else if (asset.type === 'NONERC') {\n token = {\n type: TokenType.NONERC,\n address: asset.address,\n logoUri: asset.logo_url,\n name: asset.name,\n symbol: asset.symbol,\n };\n }\n\n return token;\n};\n\nconst convertNativeAssetToToken = (asset: Blockaid.NativeAssetDetails): NetworkToken => {\n return {\n name: asset.name ?? '',\n symbol: asset.symbol ?? '',\n decimals: asset.decimals,\n description: '',\n logoUri: asset.logo_url,\n };\n};\n\nexport const processJsonRpcSimulation = async ({\n request,\n dAppUrl,\n accountAddress,\n chainId,\n data,\n blockaid,\n}: {\n request: RpcRequest;\n dAppUrl?: string;\n accountAddress: string;\n data: { method: string; params: unknown };\n chainId: number;\n blockaid: Blockaid;\n}) => {\n let alert: Alert | undefined;\n let balanceChange: BalanceChange | undefined;\n let tokenApprovals: TokenApprovals | undefined;\n\n try {\n const { validation, simulation } = await scanJsonRpc({\n chainId,\n accountAddress,\n data: data as Blockaid.Evm.JsonRpcScanParams.Data,\n domain: dAppUrl,\n blockaid,\n });\n\n if (!validation || validation.result_type === 'Error' || validation.result_type === 'Warning') {\n alert = transactionAlerts[AlertType.WARNING];\n } else if (validation.result_type === 'Malicious') {\n alert = transactionAlerts[AlertType.DANGER];\n }\n\n if (simulation?.status === 'Success') {\n tokenApprovals = processTokenApprovals(request.method, simulation.account_summary);\n balanceChange = processBalanceChange(simulation.account_summary.assets_diffs);\n }\n } catch (error) {\n console.error('processJsonRpcSimulation error', error);\n alert = transactionAlerts[AlertType.WARNING];\n }\n\n return { alert, balanceChange, tokenApprovals };\n};\n","import Blockaid from '@blockaid/client';\nimport type { TransactionBulkScanParams } from '@blockaid/client/resources/evm/transaction-bulk';\n\nimport type { TransactionBatchParams, TransactionParams } from '../types';\n\nexport const scanTransactionBatch = async ({\n chainId,\n params,\n domain,\n withGasEstimation,\n blockaid,\n}: {\n chainId: number;\n params: TransactionBatchParams;\n domain?: string;\n withGasEstimation?: boolean;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse[]> => {\n const options: TransactionBulkScanParams['options'] = ['validation', 'simulation'];\n\n if (withGasEstimation) {\n options.push('gas_estimation');\n }\n\n return blockaid.evm.transactionBulk.scan({\n chain: chainId.toString(),\n options,\n data: params,\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n\nexport const scanTransaction = async ({\n chainId,\n params,\n domain,\n blockaid,\n}: {\n chainId: number;\n params: TransactionParams;\n domain?: string;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse> => {\n return blockaid.evm.transaction.scan({\n account_address: params.from,\n chain: chainId.toString(),\n options: ['validation', 'simulation'],\n data: {\n from: params.from,\n to: params.to,\n data: params.data,\n value: params.value,\n gas: params.gas,\n gas_price: params.gasPrice,\n // TODO: provide accessList once Blockaid supports it\n // access_list: params.accessList\n },\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n\nexport const scanJsonRpc = async ({\n chainId,\n accountAddress,\n data,\n domain,\n blockaid,\n}: {\n chainId: number;\n accountAddress: string;\n data: Blockaid.Evm.JsonRpcScanParams.Data;\n domain?: string;\n blockaid: Blockaid;\n}): Promise<Blockaid.TransactionScanResponse> => {\n return blockaid.evm.jsonRpc.scan({\n chain: chainId.toString(),\n options: ['validation', 'simulation'],\n account_address: accountAddress,\n data,\n metadata: (domain && domain.length > 0 ? { domain } : { non_dapp: true }) as Blockaid.Evm.MetadataParam,\n });\n};\n","import xss from 'xss';\nimport { ethers } from 'ethers';\nimport { TokenType } from '@avalabs/vm-module-types';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport { ERC20TransactionType, type RequiredBy, type TransactionParams } from '../types';\n\nexport const parseWithErc20Abi = async (\n params: RequiredBy<TransactionParams, 'to'>,\n chainId: number,\n provider: JsonRpcBatchInternal,\n) => {\n if (!params.data) {\n return {\n tokenApprovals: undefined,\n balanceChange: undefined,\n };\n }\n\n try {\n const contract = new ethers.Contract(params.to, ERC20.abi, provider);\n const contractCalls = await Promise.all([contract.name?.(), contract.symbol?.(), contract.decimals?.()]);\n // Purify the values for XSS protection\n const name = xss(contractCalls[0]);\n const symbol = xss(contractCalls[1]);\n const decimals = parseInt(contractCalls[2]);\n const token = {\n type: TokenType.ERC20,\n name,\n chainId,\n symbol,\n decimals,\n address: params.to,\n contractType: 'ERC-20',\n } as const;\n\n const iface = new ethers.Interface(ERC20.abi);\n const calledFunction = iface.getFunction(params.data.slice(0, 10));\n\n const decodeFunctionData = iface.decodeFunctionData(params.data.slice(0, 10), params.data);\n if (calledFunction?.name === ERC20TransactionType.TRANSFER) {\n return {\n balanceChange: {\n outs: [\n {\n items: [\n {\n displayValue: new TokenUnit(decodeFunctionData['amount'], token.decimals, token.symbol).toDisplay(),\n usdPrice: undefined,\n },\n ],\n token,\n },\n ],\n ins: [],\n },\n };\n } else if (calledFunction?.name === ERC20TransactionType.APPROVE) {\n return {\n tokenApprovals: {\n isEditable: true,\n approvals: [\n {\n token,\n spenderAddress: decodeFunctionData['spender'],\n value: decodeFunctionData['amount'],\n },\n ],\n },\n };\n }\n } catch (error) {\n console.error('parseErc20Tx error', error);\n }\n\n return {\n tokenApprovals: undefined,\n balanceChange: undefined,\n };\n};\n","import {\n TokenType,\n type ApprovalController,\n type BalanceChange,\n type BatchApprovalController,\n type ERC1155Token,\n type ERC20Token,\n type ERC721Token,\n type NetworkContractToken,\n type NetworkToken,\n type TokenApprovals,\n} from '@avalabs/vm-module-types';\nimport type { RequiredBy, TransactionParams } from '../types';\n\nexport function isNetworkToken(token: NetworkContractToken | NetworkToken): token is NetworkToken {\n return !('type' in token);\n}\n\nexport function isERC20Token(token: NetworkContractToken): token is ERC20Token {\n return token.type === TokenType.ERC20;\n}\n\nexport function isNftToken(token: NetworkContractToken): token is ERC1155Token | ERC721Token {\n return token.type === TokenType.ERC1155 || token.type === TokenType.ERC721;\n}\n\nexport function hasToField(params: TransactionParams): params is RequiredBy<TransactionParams, 'to'> {\n return typeof params.to === 'string';\n}\n\nexport function supportsBatchApprovals(\n controller: ApprovalController | BatchApprovalController,\n): controller is BatchApprovalController {\n return 'requestBatchApproval' in controller && typeof controller.requestBatchApproval === 'function';\n}\n\nexport function isBalanceChange(input: unknown): input is BalanceChange {\n return (\n typeof input === 'object' &&\n input !== null &&\n 'ins' in input &&\n 'outs' in input &&\n Array.isArray(input.ins) &&\n Array.isArray(input.outs)\n );\n}\n\nexport function isTokenApprovals(input: unknown): input is TokenApprovals {\n return (\n typeof input === 'object' &&\n input !== null &&\n 'approvals' in input &&\n 'isEditable' in input &&\n Array.isArray(input.approvals)\n );\n}\n\nexport function isEmpty(input: unknown): boolean {\n if (isBalanceChange(input)) {\n return input.ins.length === 0 && input.outs.length === 0;\n }\n\n if (isTokenApprovals(input)) {\n return input.approvals.length === 0;\n }\n\n return false;\n}\n","import { AlertType } from '@avalabs/vm-module-types';\n\nexport const transactionAlerts = {\n [AlertType.WARNING]: {\n type: AlertType.WARNING,\n details: {\n title: 'Suspicious transaction',\n description: 'Use caution, this transaction might be malicious.',\n },\n },\n [AlertType.DANGER]: {\n type: AlertType.DANGER,\n details: {\n title: 'Scam transaction',\n description: 'This transaction has been flagged as malicious, I understand the risk.',\n body: ['This transaction is malicious', 'do not proceed'],\n actionTitles: {\n reject: 'Reject Transaction',\n proceed: 'Proceed Anyway',\n },\n },\n },\n};\n","import { z } from 'zod';\n\nimport { transactionSchema } from '../../utils/transaction-schema';\n\nconst paramsSchema = z.tuple([transactionSchema]);\n\nexport const parseRequestParams = (params: unknown) => {\n return paramsSchema.safeParse(params);\n};\n","import { z } from 'zod';\n\nexport const transactionSchema = z.object({\n from: z.string().length(42),\n to: z.string().length(42).optional(),\n data: z.string().optional(),\n value: z.string().startsWith('0x').optional(),\n gas: z.string().startsWith('0x').optional(),\n gasPrice: z.string().startsWith('0x').optional(),\n maxFeePerGas: z.string().startsWith('0x').optional(),\n maxPriorityFeePerGas: z.string().startsWith('0x').optional(),\n nonce: z.string().optional(),\n chainId: z.string().optional().or(z.number().optional()),\n accessList: z\n .array(\n z.object({\n address: z.string().startsWith('0x'),\n storageKeys: z.array(z.string()),\n }),\n )\n .optional(),\n});\n\nexport type TransactionParams = z.infer<typeof transactionSchema>;\n","import type { Hex, RpcRequest } from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\nimport { getExplorerAddressByNetwork } from '../handlers/get-transaction-history/utils/get-explorer-address-by-network';\nimport type { TransactionReceipt } from 'ethers';\nimport { retry, RetryBackoffPolicy } from '@internal/utils';\n\nexport const waitForTransactionReceipt = async ({\n explorerUrl,\n provider,\n txHash,\n onTransactionPending,\n onTransactionConfirmed,\n onTransactionReverted,\n request,\n}: {\n explorerUrl: string;\n provider: JsonRpcBatchInternal;\n txHash: Hex;\n onTransactionPending: ({\n txHash,\n request,\n explorerLink,\n }: {\n txHash: Hex;\n request: RpcRequest;\n explorerLink: string;\n }) => void;\n onTransactionConfirmed: ({\n txHash,\n explorerLink,\n request,\n }: {\n txHash: Hex;\n explorerLink: string;\n request: RpcRequest;\n }) => void;\n onTransactionReverted: ({ txHash, request }: { txHash: Hex; request: RpcRequest }) => void;\n request: RpcRequest;\n}) => {\n try {\n const explorerLink = getExplorerAddressByNetwork(explorerUrl, txHash);\n onTransactionPending({ txHash, request, explorerLink });\n\n const receipt = await retry<TransactionReceipt | null>({\n operation: async () => provider.getTransactionReceipt(txHash),\n isSuccess: (r): r is TransactionReceipt => !!r, // success when receipt is present (>= 1 confirmation)\n backoffPolicy: RetryBackoffPolicy.linearThenExponential(15, 750),\n maxRetries: 20,\n });\n\n const success = receipt?.status === 1; // 1 indicates success, 0 indicates revert\n\n if (success) {\n onTransactionConfirmed({ txHash, explorerLink, request });\n return true;\n }\n } catch (error) {\n console.error(error);\n }\n\n onTransactionReverted({ txHash, request });\n\n return false;\n};\n","import type { SigningResult } from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nexport const getTxHash = async (provider: JsonRpcBatchInternal, response: SigningResult) => {\n if ('txHash' in response) {\n return response.txHash;\n }\n\n // broadcast the signed transaction\n const txHash = await provider.send('eth_sendRawTransaction', [response.signedData]);\n return txHash;\n};\n","import {\n type GetBalancesParams,\n type NetworkTokenWithBalance,\n type Storage,\n type Error,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n TokenType,\n} from '@avalabs/vm-module-types';\nimport type { TokenService } from '@internal/utils';\nimport { findAsync } from '../../utils/find-async';\nimport type { BalanceServiceInterface } from './balance-service-interface';\nimport type { CurrencyCode } from '@avalabs/glacier-sdk';\nimport { addIdToPromise, type IdPromise, settleAllIdPromises } from '../../utils/id-promise';\nimport { RpcService } from '../../services/rpc-service/rpc-service';\nimport { isERC20Token } from '../../utils/type-utils';\nimport type { VsCurrencyType } from '@avalabs/core-coingecko-sdk';\n\ntype AccountAddress = string;\ntype TokenSymbol = string;\ntype GetEvmBalancesResponse = Record<AccountAddress, Record<TokenSymbol, TokenWithBalanceEVM | Error> | Error>;\n\nexport const getBalances = async ({\n addresses,\n currency,\n network,\n proxyApiUrl,\n tokenTypes = [TokenType.NATIVE, TokenType.ERC20, TokenType.ERC721, TokenType.ERC1155],\n customTokens = [],\n storage,\n balanceServices = [],\n tokenService,\n}: GetBalancesParams & {\n proxyApiUrl: string;\n balanceServices: BalanceServiceInterface[];\n storage?: Storage;\n tokenService: TokenService;\n}): Promise<GetEvmBalancesResponse> => {\n const chainId = network.chainId;\n const services: BalanceServiceInterface[] = [\n ...balanceServices,\n new RpcService({ network, storage, proxyApiUrl, customTokens }),\n ];\n\n const supportingService: BalanceServiceInterface | undefined = await findAsync(\n services,\n (balanceService: BalanceServiceInterface) => balanceService.isNetworkSupported(network.chainId),\n );\n\n const balances: GetEvmBalancesResponse = {};\n if (supportingService) {\n const nativeTokenPromises: Promise<IdPromise<NetworkTokenWithBalance>>[] = [];\n const erc20TokenPromises: Promise<IdPromise<Record<string, TokenWithBalanceEVM | Error>>>[] = [];\n const nftTokenPromises: Promise<IdPromise<Record<string, NftTokenWithBalance | Error>>>[] = [];\n addresses.forEach((address) => {\n if (tokenTypes.includes(TokenType.NATIVE)) {\n nativeTokenPromises.push(\n addIdToPromise(\n supportingService\n .getNativeBalance({\n address,\n currency: currency.toUpperCase() as CurrencyCode,\n chainId,\n })\n .then(async (nativeBalance) => {\n if (nativeBalance.symbol === 'AVAX') {\n // we want to standardise AVAX price across the chains\n const avaxMarketData = await tokenService.getWatchlistDataForToken({\n tokenDetails: {\n symbol: network.networkToken.symbol,\n isNative: true,\n caip2Id: network.caipId ?? '',\n },\n currency: currency.toUpperCase() as VsCurrencyType,\n });\n\n return {\n ...nativeBalance,\n priceInCurrency: avaxMarketData.priceInCurrency,\n marketCap: avaxMarketData.marketCap,\n vol24: avaxMarketData.vol24,\n change24: avaxMarketData.change24,\n };\n }\n\n return nativeBalance;\n }),\n address,\n ),\n );\n }\n\n if (tokenTypes.includes(TokenType.ERC20)) {\n erc20TokenPromises.push(\n addIdToPromise(\n supportingService.listErc20Balances({\n customTokens: customTokens.filter(isERC20Token),\n currency: currency.toUpperCase() as CurrencyCode,\n chainId,\n address,\n pageSize: 100,\n }),\n address,\n ),\n );\n }\n\n if (tokenTypes.includes(TokenType.ERC721) || tokenTypes.includes(TokenType.ERC1155)) {\n nftTokenPromises.push(\n addIdToPromise(\n supportingService.listNftBalances({\n chainId,\n address,\n }),\n address,\n ),\n );\n }\n });\n const nativeTokenBalances = await settleAllIdPromises(nativeTokenPromises);\n const erc20TokenBalances = await settleAllIdPromises(erc20TokenPromises);\n const nftTokenBalances = await settleAllIdPromises(nftTokenPromises);\n Object.keys(nativeTokenBalances).forEach((address) => {\n const balanceOrError = nativeTokenBalances[address];\n if (!balanceOrError || 'error' in balanceOrError) {\n balances[address] = {\n error: `getNativeBalance failed: ${balanceOrError?.error ?? 'unknown error'}`,\n } as Error;\n return;\n }\n const tokenSymbol = balanceOrError.symbol as string;\n balances[address] = {\n [tokenSymbol]: balanceOrError,\n };\n });\n Object.keys(erc20TokenBalances).forEach((address) => {\n const balancesOrError = erc20TokenBalances[address];\n if (!balancesOrError || ('error' in balancesOrError && typeof balancesOrError.error !== 'string')) {\n balances[address] = {\n ...balances[address],\n error: `listErc20Balances failed: unknown error`,\n };\n } else if ('error' in balancesOrError && typeof balancesOrError.error === 'string') {\n balances[address] = {\n ...balances[address],\n error: `listErc20Balances failed: ${balancesOrError.error}`,\n };\n } else {\n balances[address] = {\n ...balances[address],\n ...balancesOrError,\n };\n }\n });\n Object.keys(nftTokenBalances).forEach((address) => {\n const balancesOrError = nftTokenBalances[address];\n if (!balancesOrError || ('error' in balancesOrError && typeof balancesOrError.error !== 'string')) {\n balances[address] = {\n ...balances[address],\n error: `listNftBalances failed: unknown error`,\n };\n } else if ('error' in balancesOrError && typeof balancesOrError.error === 'string') {\n balances[address] = {\n ...balances[address],\n error: `listNftBalances failed: ${balancesOrError.error}`,\n };\n } else {\n balances[address] = {\n ...balances[address],\n ...balancesOrError,\n };\n }\n });\n } else {\n addresses.forEach((address) => {\n balances[address] = {\n error: 'unsupported network',\n };\n });\n }\n\n return balances;\n};\n","/**\n * Asynchronously searches through an array to find the first element that satisfies the provided async callback function.\n *\n * @param {T[]} array - The array to search through.\n * @param {(item: T) => Promise<boolean>} asyncCallback - An async function that takes an element of the array and returns a Promise resolving to a boolean.\n * @returns {Promise<T | undefined>} - A Promise that resolves to the first element in the array that satisfies the asyncCallback function, or undefined if no such element is found.\n *\n * @example\n * ```ts\n * const array = [1, 2, 3, 4];\n * const asyncCallback = async (num) => {\n * return new Promise((resolve) => {\n * setTimeout(() => {\n * resolve(num % 2 === 0);\n * }, 100);\n * });\n * };\n *\n * findAsync(array, asyncCallback).then((result) => {\n * console.log(result); // Output: 2 (the first even number)\n * });\n * ```\n */\nexport async function findAsync<T>(array: T[], asyncCallback: (item: T) => Promise<boolean>): Promise<T | undefined> {\n const promises = array.map(async (item, index) => ({\n index,\n result: await asyncCallback(item),\n }));\n\n const results = await Promise.allSettled(promises);\n\n const found = results\n .filter((item) => item.status === 'fulfilled')\n .map((item) => item as PromiseFulfilledResult<{ index: number; result: boolean }>)\n .find((item) => {\n return item.value.result;\n });\n return found ? array[found.value.index] : undefined;\n}\n","import { type Error } from '@avalabs/vm-module-types';\n\ntype Id = string;\nexport type IdPromise<T> =\n | { id: string; status: 'fulfilled'; value: T }\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n | { id: string; status: 'rejected'; reason: any };\n\n/**\n * The addIdToPromise function takes an existing promise and an id string as input and returns a new promise that\n * resolves to an object containing the id, the status of the original promise (either 'fulfilled' or 'rejected'),\n * and the result or error of the original promise.\n *\n * Usage:\n * This function is useful for tracking the status and outcome of multiple promises when used with Promise.allSettled\n *\n * @param promise The original promise that resolves to a value of type T.\n * @param id A unique identifier associated with the promise.\n */\nexport function addIdToPromise<T>(promise: Promise<T>, id: string): Promise<IdPromise<T>> {\n return promise\n .then((value) => ({ id, status: 'fulfilled', value }) as IdPromise<T>)\n .catch((reason) => ({ id, status: 'rejected', reason }) as IdPromise<T>);\n}\n\n/**\n * The settleAllIdPromises function processes an array of promises that return IdPromise objects,\n * resolving all of them and organizing the results into a record.\n *\n * Usage:\n * Use {@link addIdToPromise} to get array of promises\n *\n * @param promises An array of promises that each resolve to an IdPromise<T> object.\n */\nexport async function settleAllIdPromises<T>(promises: Promise<IdPromise<T>>[]): Promise<Record<Id, T | Error>> {\n return await Promise.allSettled(promises).then((results) => {\n return results.reduce(\n (acc, result) => {\n if (result.status === 'fulfilled') {\n const value = result.value;\n if (value.status === 'fulfilled') {\n acc[value.id] = value.value;\n } else {\n acc[value.id] = { error: value.reason };\n }\n }\n return acc;\n },\n {} as Record<Id, T | Error>,\n );\n });\n}\n","import { CurrencyCode } from '@avalabs/glacier-sdk';\nimport type { BalanceServiceInterface, TokenId } from '../../handlers/get-balances/balance-service-interface';\nimport {\n type ERC20Token,\n type Network,\n type NetworkContractToken,\n type NetworkTokenWithBalance,\n type Storage,\n type Error,\n TokenType,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n} from '@avalabs/vm-module-types';\nimport { addIdToPromise, settleAllIdPromises } from '../../utils/id-promise';\nimport type { VsCurrencyType } from '@avalabs/core-coingecko-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { ethers } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport { getProvider } from '../../utils/get-provider';\nimport { TokenService } from '@internal/utils';\nimport { getTokens } from '../../handlers/get-tokens/get-tokens';\nimport { isERC20Token } from '../../utils/type-utils';\n\nexport class RpcService implements BalanceServiceInterface {\n #network: Network;\n #storage: Storage | undefined;\n #proxyApiUrl: string;\n #customTokens: NetworkContractToken[];\n\n constructor({\n network,\n storage,\n proxyApiUrl,\n customTokens,\n }: {\n network: Network;\n storage?: Storage;\n proxyApiUrl: string;\n customTokens: NetworkContractToken[];\n }) {\n this.#network = network;\n this.#storage = storage;\n this.#proxyApiUrl = proxyApiUrl;\n this.#customTokens = customTokens;\n }\n\n async isNetworkSupported(): Promise<boolean> {\n return true;\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n }): Promise<NetworkTokenWithBalance> {\n const provider = await getProvider({\n chainId,\n chainName: this.#network.chainName,\n rpcUrl: this.#network.rpcUrl,\n multiContractAddress: this.#network.utilityAddresses?.multicall,\n });\n\n const coingeckoTokenId = this.#network.pricingProviders?.coingecko.nativeTokenId;\n const networkToken = this.#network.networkToken;\n const tokenService = new TokenService({ storage: this.#storage, proxyApiUrl: this.#proxyApiUrl });\n const simplePriceResponse = coingeckoTokenId\n ? await tokenService.getSimplePrice({\n coinIds: [coingeckoTokenId],\n currencies: [currency] as unknown as VsCurrencyType[],\n })\n : {};\n\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? undefined;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? undefined;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? undefined;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? undefined;\n\n const balance = await provider.getBalance(address);\n const balanceUnit = new TokenUnit(balance, networkToken.decimals, networkToken.symbol);\n const balanceInCurrency = priceInCurrency !== undefined ? balanceUnit.mul(priceInCurrency) : undefined;\n\n return {\n ...networkToken,\n coingeckoId: coingeckoTokenId ?? '',\n type: TokenType.NATIVE,\n balance,\n balanceDisplayValue: balanceUnit.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n marketCap,\n vol24,\n change24,\n };\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n pageSize: number;\n pageToken?: string;\n }): Promise<Record<TokenId, TokenWithBalanceEVM | Error>> {\n const provider = await getProvider({\n chainId,\n chainName: this.#network.chainName,\n rpcUrl: this.#network.rpcUrl,\n multiContractAddress: this.#network.utilityAddresses?.multicall,\n });\n\n const coingeckoPlatformId = this.#network.pricingProviders?.coingecko.assetPlatformId;\n const coingeckoTokenId = this.#network.pricingProviders?.coingecko.nativeTokenId;\n const tokens = await getTokens({ chainId: Number(chainId), proxyApiUrl: this.#proxyApiUrl });\n const tokenAddresses = tokens.map((token) => token.address);\n const erc20TokenList = [...tokens, ...this.#customTokens].filter(isERC20Token);\n\n const getTokenWithBalance = async (token: ERC20Token) => {\n const contract = new ethers.Contract(token.address, ERC20.abi, provider);\n const balanceBig: bigint = await contract.balanceOf?.(address);\n const balance = balanceBig || 0n;\n\n return {\n ...token,\n balance,\n };\n };\n\n const tokenBalancePromises = erc20TokenList.map((token) => {\n return addIdToPromise(getTokenWithBalance(token), token.address);\n });\n const tokenBalancesResults = await settleAllIdPromises(tokenBalancePromises);\n const tokenService = new TokenService({ storage: this.#storage, proxyApiUrl: this.#proxyApiUrl });\n const simplePriceResponse =\n (coingeckoPlatformId &&\n (await tokenService.getPricesByAddresses(\n tokenAddresses,\n coingeckoPlatformId,\n currency as unknown as VsCurrencyType,\n ))) ||\n {};\n\n const tokenIds = Object.keys(tokenBalancesResults);\n const erc20TokenBalances: Record<TokenId, TokenWithBalanceEVM | Error> = {};\n for (let i = 0; i < tokenIds.length; i++) {\n const tokenId = tokenIds[i];\n if (tokenId === undefined) continue;\n const tokenBalance = tokenBalancesResults[tokenId];\n if (tokenBalance === undefined || 'error' in tokenBalance) {\n erc20TokenBalances[tokenId] = {\n error: `rpcService:getTokenWithBalance failed: ${tokenBalance?.error ?? 'unknown error'}`,\n };\n continue;\n }\n\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? undefined;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? undefined;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? undefined;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? undefined;\n\n const balance = new TokenUnit(tokenBalance.balance, tokenBalance.decimals, tokenBalance.symbol);\n const balanceInCurrency = priceInCurrency !== undefined ? balance.mul(priceInCurrency) : undefined;\n\n erc20TokenBalances[tokenBalance.address.toLowerCase()] = {\n ...tokenBalance,\n type: TokenType.ERC20,\n balance: tokenBalance.balance,\n balanceDisplayValue: balance.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n marketCap,\n change24,\n vol24,\n reputation: null,\n };\n }\n return erc20TokenBalances;\n }\n\n async listNftBalances(): Promise<Record<string, NftTokenWithBalance | Error>> {\n // the token list does not maintain a list of NFTs\n return {};\n }\n}\n","import { Environment } from '@avalabs/vm-module-types';\n\ntype Env = {\n glacierApiUrl: string;\n proxyApiUrl: string;\n};\n\nexport const prodEnv: Env = {\n glacierApiUrl: 'https://glacier-api.avax.network',\n proxyApiUrl: 'https://proxy-api.avax.network',\n};\n\nexport const devEnv: Env = {\n glacierApiUrl: 'https://glacier-api-dev.avax.network',\n proxyApiUrl: 'https://proxy-api-dev.avax.network',\n};\n\nexport const getEnv = (environment: Environment): Env => {\n switch (environment) {\n case Environment.PRODUCTION:\n return prodEnv;\n case Environment.DEV:\n return devEnv;\n }\n};\n","import {\n CurrencyCode,\n Erc1155Token,\n Erc1155TokenBalance,\n Erc20TokenBalance,\n Erc721Token,\n Erc721TokenBalance,\n Glacier,\n} from '@avalabs/glacier-sdk';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport {\n type Error,\n type ERC20Token,\n type NetworkTokenWithBalance,\n TokenType,\n type TokenWithBalanceERC20,\n type TokenWithBalanceEVM,\n type NftTokenWithBalance,\n} from '@avalabs/vm-module-types';\n\nimport type { BalanceServiceInterface } from '@src/handlers/get-balances/balance-service-interface';\nimport { DEFAULT_DECIMALS } from '@src/constants';\nimport { ChainId } from '@avalabs/core-chains-sdk';\nimport { getSmallImageForNFT } from '@src/utils/get-small-image-for-nft';\nimport { GlacierFetchHttpRequest } from '@internal/utils';\n\nclass GlacierUnhealthyError extends Error {\n override message = 'Glacier is unhealthy. Try again later.';\n}\n\nconst CHAINS_TO_FILTER = [ChainId.ETHEREUM_HOMESTEAD];\n\nexport class EvmGlacierService implements BalanceServiceInterface {\n glacierSdk: Glacier;\n isGlacierHealthy = true;\n supportedChainIds: string[] = [];\n\n constructor({ glacierApiUrl, headers }: { glacierApiUrl: string; headers?: Record<string, string> }) {\n this.glacierSdk = new Glacier({ BASE: glacierApiUrl, HEADERS: headers }, GlacierFetchHttpRequest);\n /**\n * This is for performance, basically we just cache the health of glacier every 5 seconds and\n * go off of that instead of every request\n */\n this.getSupportedChainIds().catch(() => {\n // Noop. It will be retried by .isSupportedNetwork calls upon unlocking if necessary.\n });\n }\n\n isHealthy = (): boolean => this.isGlacierHealthy;\n\n setGlacierToUnhealthy(): void {\n this.isGlacierHealthy = false;\n setTimeout(\n () => {\n this.isGlacierHealthy = true;\n },\n 5 * 60 * 1000,\n ); // 5 minutes\n }\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n const chainIds = await this.getSupportedChainIds();\n return chainIds.some((id) => id === chainId.toString());\n }\n\n async getSupportedChainIds(): Promise<string[]> {\n if (this.supportedChainIds.length) {\n return this.supportedChainIds;\n }\n\n try {\n const supportedChains = await this.glacierSdk.evmChains.supportedChains({});\n /*\n * https://ava-labs.atlassian.net/browse/CP-9855\n * We are removing the support for Ethereum chain, so it is queried from DeBank instead\n */\n this.supportedChainIds = supportedChains.chains\n .map((chain) => chain.chainId)\n .filter((chainId) => !CHAINS_TO_FILTER.includes(Number(chainId)));\n return this.supportedChainIds;\n } catch {\n return [];\n }\n }\n\n async reindexNft({\n address,\n chainId,\n tokenId,\n }: {\n address: string;\n chainId: string;\n tokenId: string;\n }): Promise<void> {\n try {\n await this.glacierSdk.nfTs.reindexNft({\n address,\n chainId,\n tokenId,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async getTokenDetails({\n address,\n chainId,\n tokenId,\n }: {\n address: string;\n chainId: string;\n tokenId: string;\n }): Promise<Erc721Token | Erc1155Token> {\n try {\n return this.glacierSdk.nfTs.getTokenDetails({\n address,\n chainId,\n tokenId,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n coingeckoId,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n coingeckoId?: string;\n }): Promise<NetworkTokenWithBalance> {\n try {\n const nativeBalance = await this.glacierSdk.evmBalances.getNativeBalance({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n });\n\n const nativeTokenBalance = nativeBalance.nativeTokenBalance;\n const balance = new TokenUnit(nativeTokenBalance.balance, nativeTokenBalance.decimals, nativeTokenBalance.symbol);\n const priceInCurrency = nativeTokenBalance.price?.value;\n const balanceInCurrency = priceInCurrency !== undefined ? balance.mul(priceInCurrency) : undefined;\n\n return {\n name: nativeTokenBalance.name,\n symbol: nativeTokenBalance.symbol,\n decimals: nativeTokenBalance.decimals,\n type: TokenType.NATIVE,\n logoUri: nativeTokenBalance.logoUri,\n balance: balance.toSubUnit(),\n balanceDisplayValue: balance.toDisplay(),\n balanceInCurrency: balanceInCurrency?.toDisplay({ fixedDp: 2, asNumber: true }),\n balanceCurrencyDisplayValue: balanceInCurrency?.toDisplay({ fixedDp: 2 }),\n priceInCurrency,\n coingeckoId: coingeckoId ?? '',\n };\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listErc721Balances({ chainId, address }: { chainId: string; address: string }): Promise<Erc721TokenBalance[]> {\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n const tokens: Erc721TokenBalance[] = [];\n do {\n try {\n const response = await this.glacierSdk.evmBalances.listErc721Balances({\n chainId,\n address,\n pageSize: 100,\n pageToken: nextPageToken,\n });\n\n tokens.push(...response.erc721TokenBalances);\n\n nextPageToken = response.nextPageToken;\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n } while (nextPageToken);\n\n return tokens;\n }\n\n async listErc1155Balances({\n chainId,\n address,\n }: {\n chainId: string;\n address: string;\n }): Promise<Erc1155TokenBalance[]> {\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n const tokens: Erc1155TokenBalance[] = [];\n do {\n try {\n const response = await this.glacierSdk.evmBalances.listErc1155Balances({\n chainId,\n address,\n pageSize: 100,\n pageToken: nextPageToken,\n });\n\n tokens.push(...response.erc1155TokenBalances);\n\n nextPageToken = response.nextPageToken;\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n } while (nextPageToken);\n\n return tokens;\n }\n\n async listNftBalances({\n chainId,\n address,\n }: {\n chainId: number;\n address: string;\n }): Promise<Record<string, NftTokenWithBalance | Error>> {\n const balances = await Promise.allSettled([\n this.listErc721Balances({ chainId: chainId.toString(), address }),\n this.listErc1155Balances({ chainId: chainId.toString(), address }),\n ]);\n\n const entries = balances\n .filter(\n (\n tokenlist,\n ): tokenlist is PromiseFulfilledResult<Erc721TokenBalance[]> | PromiseFulfilledResult<Erc1155TokenBalance[]> =>\n tokenlist.status === 'fulfilled',\n )\n .flatMap((tokenlist) => {\n return (\n tokenlist.value\n // We filter out erc1155s with 0 balance, which Glacier returns for some reason\n .filter((token) => token.ercType === Erc721Token.ercType.ERC_721 || BigInt(token.balance) > 0n)\n .map((token) => {\n return [\n `${token.address}-${token.tokenId}`,\n {\n address: token.address,\n description: token.metadata.description ?? '',\n logoUri: token.metadata.imageUri ?? '',\n logoSmall: getSmallImageForNFT(token.metadata.imageUri ?? ''),\n name: token.metadata.name ?? '',\n symbol: token.metadata.symbol ?? '',\n tokenId: token.tokenId,\n tokenUri: token.tokenUri,\n chainId,\n // glacier does not provide the collection name information\n collectionName: 'Unknown',\n balance: token.ercType === Erc1155Token.ercType.ERC_1155 ? BigInt(token.balance) : 1,\n balanceDisplayValue: token.ercType === Erc1155Token.ercType.ERC_1155 ? token.balance : '1',\n type: token.ercType === Erc721Token.ercType.ERC_721 ? TokenType.ERC721 : TokenType.ERC1155,\n metadata: {\n indexStatus: token.metadata.indexStatus,\n description: token.metadata.description,\n lastUpdatedTimestamp: token.metadata.metadataLastUpdatedTimestamp,\n properties:\n token.ercType === Erc721Token.ercType.ERC_721\n ? token.metadata.attributes\n : token.metadata.properties,\n },\n },\n ];\n })\n );\n });\n\n return Object.fromEntries(entries);\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n customTokens,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n customTokens: ERC20Token[];\n }): Promise<Record<string, TokenWithBalanceEVM | Error>> {\n try {\n const tokensWithBalance: TokenWithBalanceERC20[] = [];\n /**\n * Load all pages to make sure we have all the tokens with balances\n */\n let nextPageToken: string | undefined = undefined;\n do {\n const response = await this.glacierSdk.evmBalances.listErc20Balances({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n pageSize: 100, // glacier has a cap on page size of 100\n pageToken: nextPageToken,\n });\n\n tokensWithBalance.push(\n ...convertErc20TokenWithBalanceToTokenWithBalance(response.erc20TokenBalances, Number(chainId)),\n );\n nextPageToken = response.nextPageToken;\n } while (nextPageToken);\n\n /**\n * Glacier doesnt return tokens without balances so we need to polyfill that list\n * from our own list of tokens. We just set the balance to 0, these zero balance\n * tokens are only used for swap, bridge and tx parsing.\n */\n return [\n ...convertErc20TokenToTokenWithBalance(customTokens),\n ...tokensWithBalance, // this needs to be second in the list so it overwrites its zero balance counterpart if there is one\n ].reduce(\n (acc, token) => {\n return { ...acc, [token.address.toLowerCase()]: token };\n },\n {} as Record<string, TokenWithBalanceEVM>,\n );\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listTransactions({\n chainId,\n address,\n pageToken,\n pageSize,\n }: {\n chainId: string;\n address: string;\n pageToken?: string;\n pageSize?: number;\n }) {\n try {\n return this.glacierSdk.evmTransactions.listTransactionsV2({\n chainId,\n address,\n pageToken,\n pageSize,\n filterSpamTokens: true,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n}\n\nconst convertErc20TokenToTokenWithBalance = (tokens: ERC20Token[]): TokenWithBalanceERC20[] => {\n return tokens.map((token) => {\n return {\n ...token,\n decimals: token.decimals ?? DEFAULT_DECIMALS,\n type: TokenType.ERC20,\n balance: 0n,\n balanceInCurrency: 0,\n balanceDisplayValue: '0',\n balanceCurrencyDisplayValue: '0',\n priceInCurrency: 0,\n marketCap: 0,\n change24: 0,\n vol24: 0,\n reputation: null,\n };\n });\n};\n\nconst convertErc20TokenWithBalanceToTokenWithBalance = (\n tokenBalances: Erc20TokenBalance[],\n chainId: number,\n): TokenWithBalanceERC20[] => {\n return tokenBalances.map((token: Erc20TokenBalance): TokenWithBalanceERC20 => {\n const balance = new TokenUnit(token.balance, token.decimals, token.symbol);\n const balanceDisplayValue = balance.toDisplay();\n const balanceCurrencyDisplayValue = token.balanceValue?.value.toString();\n const priceInCurrency = token.price?.value;\n const balanceInCurrency =\n priceInCurrency !== undefined\n ? balance.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n return {\n chainId,\n address: token.address,\n name: token.name,\n symbol: token.symbol,\n decimals: token.decimals,\n logoUri: token.logoUri,\n balance: balance.toSubUnit(),\n balanceCurrencyDisplayValue,\n balanceDisplayValue,\n balanceInCurrency,\n priceInCurrency,\n reputation: token.tokenReputation,\n type: TokenType.ERC20,\n };\n });\n};\n","export const DEFAULT_DECIMALS = 18;\n\nexport const BLOCKAID_API_KEY = 'DUMMY_API_KEY'; // since we're using our own proxy and api key is handled there, we can use a dummy key here\n","import { ipfsResolver } from '@avalabs/core-utils-sdk';\n\nexport const IPFS_URL = 'https://ipfs.io';\n\nexport function ipfsResolverWithFallback(sourceUrl: string | undefined, desiredGatewayPrefix: string = IPFS_URL) {\n if (!sourceUrl) {\n return '';\n }\n\n try {\n return ipfsResolver(sourceUrl, desiredGatewayPrefix);\n } catch {\n return sourceUrl;\n }\n}\n","import { ipfsResolverWithFallback } from './ipsf-resolver-with-fallback';\n\nconst COVALENT_IMG_SIZER = 'https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image';\n\n/**\n * Covalent has an on the fly image resizer, it resolves image urls then resizes the image.\n *\n * This allows us to request smaller images depending on the UI needs\n *\n * @param imgUrl the url of the image to convert to size\n * @param imageSize the resize dimension\n * @returns The url to the image which is sized at the time of request\n */\nexport function getSmallImageForNFT(imgUrl: string, imageSize: '256' | '512' | '1024' = '256') {\n const url = ipfsResolverWithFallback(imgUrl);\n return `${COVALENT_IMG_SIZER}/width=${imageSize},fit/${url}`;\n}\n","import {\n type SigningData,\n type Network,\n type ApprovalController,\n type DisplayData,\n type RpcRequest,\n RpcMethod,\n type Alert,\n AlertType,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { toUtf8String } from 'ethers';\nimport { beautifySimpleMessage, beautifyComplexMessage } from './utils/beautify-message/beautify-message';\nimport { parseRequestParams } from './schemas/parse-request-params/parse-request-params';\nimport { isTypedDataV1 } from './utils/typeguards';\nimport { isTypedDataValid } from './utils/is-typed-data-valid';\nimport { processJsonRpcSimulation } from '../../utils/process-transaction-simulation';\nimport { textItem } from '@internal/utils/src/utils/detail-item';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSign = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n blockaid: Blockaid;\n}) => {\n const result = parseRequestParams({ method: request.method, params: request.params });\n\n if (!result.success) {\n console.error('invalid params', result.error);\n\n return {\n success: false,\n error: rpcErrors.invalidParams({ message: 'Params are invalid', data: { cause: result.error } }),\n };\n }\n const { method, data, address } = result.data;\n\n // validate typed data\n let typedDataValidationResult: ReturnType<typeof isTypedDataValid> | undefined;\n\n if (method === RpcMethod.SIGN_TYPED_DATA_V3 || method === RpcMethod.SIGN_TYPED_DATA_V4) {\n typedDataValidationResult = isTypedDataValid(data);\n }\n\n // generate display data and signing data\n let signingData: SigningData | undefined;\n let messageDetails: string | undefined;\n let alert: Alert | undefined;\n\n if (typedDataValidationResult && !typedDataValidationResult.isValid) {\n alert = {\n type: AlertType.INFO,\n details: {\n title: 'Warning: Verify Message Content',\n description: 'This message contains non-standard elements. Please verify message content!',\n detailedDescription: (typedDataValidationResult.error as Error).toString(),\n },\n };\n }\n\n if (method === RpcMethod.ETH_SIGN) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = data;\n } else if (method === RpcMethod.PERSONAL_SIGN) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = toUtf8String(data);\n } else if (method === RpcMethod.SIGN_TYPED_DATA || method === RpcMethod.SIGN_TYPED_DATA_V1) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n messageDetails = isTypedDataV1(data) ? beautifySimpleMessage(data) : beautifyComplexMessage(data);\n } else if (method === RpcMethod.SIGN_TYPED_DATA_V3 || method === RpcMethod.SIGN_TYPED_DATA_V4) {\n signingData = {\n type: method,\n account: address,\n data: data,\n };\n\n const { types, primaryType, ...messageToDisplay } = data;\n messageDetails = beautifyComplexMessage(messageToDisplay);\n }\n\n if (!signingData || !messageDetails) {\n return {\n success: false,\n error: rpcErrors.internal('Unable to generate signing data'),\n };\n }\n\n const simulationResult = await processJsonRpcSimulation({\n request,\n accountAddress: address,\n chainId: network.chainId,\n data: { method, params: request.params },\n dAppUrl: request.dappInfo.url,\n blockaid,\n });\n\n const displayData: DisplayData = {\n title: 'Sign Message',\n dAppInfo: {\n name: request.dappInfo.name,\n action: `${request.dappInfo.name} is requesting to sign the following message`,\n logoUri: request.dappInfo.icon,\n },\n network: {\n chainId: network.chainId,\n name: network.chainName,\n logoUri: network.logoUri,\n },\n account: address,\n details: [\n {\n title: 'Message',\n items: [textItem('Message', messageDetails, 'vertical')],\n },\n ],\n alert: simulationResult?.alert ?? alert,\n balanceChange: simulationResult?.balanceChange,\n tokenApprovals: simulationResult?.tokenApprovals,\n };\n\n // prompt user for approval\n const response = await approvalController.requestApproval({ request, displayData, signingData });\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n if (!('signedData' in response)) {\n return {\n error: rpcErrors.internal('No signed data returned'),\n };\n }\n\n return { result: response.signedData };\n};\n","/* eslint-disable @typescript-eslint/no-explicit-any */\nexport const beautifyComplexMessage = (data: { domain: Record<string, any>; message: Record<string, any> }) => {\n let result = '';\n\n result += 'Domain\\n';\n for (const key in data.domain) {\n result += ` ${key}: ${toUnicodeBold(String(data.domain[key]))}\\n`;\n }\n\n result += '\\nMessage\\n';\n for (const key in data.message) {\n if (typeof data.message[key] === 'object' && !Array.isArray(data.message[key])) {\n result += ` ${key}: \\n`;\n for (const subKey in data.message[key]) {\n if (Array.isArray(data.message[key][subKey])) {\n result += ` ${subKey}: \\n`;\n data.message[key][subKey].forEach((item: any, index: number) => {\n result += ` ${index}: ${toUnicodeBold(item)}\\n`;\n });\n } else {\n result += ` ${subKey}: ${toUnicodeBold(String(data.message[key][subKey]))}\\n`;\n }\n }\n } else if (Array.isArray(data.message[key])) {\n result += ` ${key}: \\n`;\n data.message[key].forEach((item: any, index: number) => {\n result += ` ${index}: \\n`;\n for (const subKey in item) {\n if (Array.isArray(item[subKey])) {\n result += ` ${subKey}: \\n`;\n item[subKey].forEach((item: any, index: number) => {\n result += ` ${index}: ${toUnicodeBold(item)}\\n`;\n });\n } else {\n result += ` ${subKey}: ${toUnicodeBold(String(item[subKey]))}\\n`;\n }\n }\n });\n } else {\n result += ` ${key}: ${toUnicodeBold(String(data.message[key]))}\\n`;\n }\n }\n\n return result;\n};\n\nexport const beautifySimpleMessage = (\n data: {\n name: string;\n type: string;\n value: any;\n }[],\n) => {\n let result = '';\n\n data.forEach((item) => {\n result += `${item.name}:\\n`;\n result += `${toUnicodeBold(String(item.value))}\\n\\n`;\n });\n\n return result;\n};\n\nconst toUnicodeBold = (str: string) => {\n const boldMap: Record<string, string> = {\n '0': '𝟬',\n '1': '𝟭',\n '2': '𝟮',\n '3': '𝟯',\n '4': '𝟰',\n '5': '𝟱',\n '6': '𝟲',\n '7': '𝟳',\n '8': '𝟴',\n '9': '𝟵',\n a: '𝗮',\n b: '𝗯',\n c: '𝗰',\n d: '𝗱',\n e: '𝗲',\n f: '𝗳',\n g: '𝗴',\n h: '𝗵',\n i: '𝗶',\n j: '𝗷',\n k: '𝗸',\n l: '𝗹',\n m: '𝗺',\n n: '𝗻',\n o: '𝗼',\n p: '𝗽',\n q: '𝗾',\n r: '𝗿',\n s: '𝘀',\n t: '𝘁',\n u: '𝘂',\n v: '𝘃',\n w: '𝘄',\n x: '𝘅',\n y: '𝘆',\n z: '𝘇',\n A: '𝗔',\n B: '𝗕',\n C: '𝗖',\n D: '𝗗',\n E: '𝗘',\n F: '𝗙',\n G: '𝗚',\n H: '𝗛',\n I: '𝗜',\n J: '𝗝',\n K: '𝗞',\n L: '𝗟',\n M: '𝗠',\n N: '𝗡',\n O: '𝗢',\n P: '𝗣',\n Q: '𝗤',\n R: '𝗥',\n S: '𝗦',\n T: '𝗧',\n U: '𝗨',\n V: '𝗩',\n W: '𝗪',\n X: '𝗫',\n Y: '𝗬',\n Z: '𝗭',\n };\n\n return str\n .split('')\n .map((char) => boldMap[char] || char)\n .join('');\n};\n","import { z } from 'zod';\nimport { ethSignSchema } from '../eth-sign';\nimport {\n combinedTypedDataSchema,\n ethSignTypedDataSchema,\n ethSignTypedDataV1Schema,\n ethSignTypedDataV3Schema,\n ethSignTypedDataV4Schema,\n typedDataSchema,\n} from '../eth-sign-typed-data';\nimport { personalSignSchema } from '../personal-sign';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\nconst paramsSchema = z\n .discriminatedUnion('method', [\n personalSignSchema,\n ethSignSchema,\n ethSignTypedDataSchema,\n ethSignTypedDataV1Schema,\n ethSignTypedDataV3Schema,\n ethSignTypedDataV4Schema,\n ])\n .transform((value, ctx) => {\n const { method, params } = value;\n\n switch (method) {\n case RpcMethod.PERSONAL_SIGN:\n return {\n data: params[0],\n address: params[1],\n method,\n };\n case RpcMethod.ETH_SIGN:\n return {\n data: params[1],\n address: params[0],\n method,\n };\n case RpcMethod.SIGN_TYPED_DATA:\n case RpcMethod.SIGN_TYPED_DATA_V1: {\n const address = params[0];\n const data = params[1];\n\n if (typeof data !== 'string') return { data, address, method };\n\n try {\n const parsed = JSON.parse(data);\n const result = combinedTypedDataSchema.parse(parsed);\n\n return {\n data: result,\n address,\n method,\n };\n } catch (e) {\n ctx.addIssue({\n code: z.ZodIssueCode.custom,\n message: 'param is not a valid json',\n });\n\n return z.NEVER;\n }\n }\n case RpcMethod.SIGN_TYPED_DATA_V3:\n case RpcMethod.SIGN_TYPED_DATA_V4: {\n const address = params[0];\n const data = params[1];\n\n if (typeof data !== 'string') return { data, address, method };\n\n try {\n const parsed = JSON.parse(data);\n const result = typedDataSchema.parse(parsed);\n\n return {\n data: result,\n address,\n method,\n };\n } catch (e) {\n ctx.addIssue({\n code: z.ZodIssueCode.custom,\n message: 'param is not a valid json',\n });\n\n return z.NEVER;\n }\n }\n }\n });\n\nexport function parseRequestParams(params: { method: RpcMethod; params: unknown }) {\n return paramsSchema.safeParse(params);\n}\n","import { z } from 'zod';\nimport { addressSchema, messageSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n// https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sign\nexport const ethSignSchema = z.object({\n method: z.literal(RpcMethod.ETH_SIGN),\n params: z.tuple([addressSchema, messageSchema]),\n});\n","import { z } from 'zod';\n\nexport const messageSchema = z.string().describe('message');\n\nexport const addressSchema = z.string().describe('address');\n","import { z } from 'zod';\nimport { addressSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n/**\n * For the different eth_signTypedData methods, the payload differs based on the version.\n *\n * V1 is based upon [an early version of EIP-712](https://github.com/ethereum/EIPs/pull/712/commits/21abe254fe0452d8583d5b132b1d7be87c0439ca)\n * that lacked some later security improvements, and should generally be neglected in favor of\n * later versions.\n *\n * V3 is based on [EIP-712](https://eips.ethereum.org/EIPS/eip-712), except that arrays and\n * recursive data structures are not supported.\n *\n * V4 is based on [EIP-712](https://eips.ethereum.org/EIPS/eip-712), and includes full support of\n * arrays and recursive data structures.\n *\n * References:\n * - https://eips.ethereum.org/EIPS/eip-712#specification-of-the-eth_signtypeddata-json-rpc\n * - https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4\n */\nconst messageTypeSchema = z.object({ name: z.string(), type: z.string() });\n\nexport const typedDataSchema = z.object({\n types: z.object({ EIP712Domain: z.array(messageTypeSchema) }).catchall(z.array(messageTypeSchema)),\n primaryType: z.string(),\n domain: z.record(z.any()),\n message: z.record(z.any()),\n});\n\nexport const typedDataV1Schema = z\n .array(\n z.object({\n type: z.string(),\n name: z.string(),\n value: z.union([z.string(), z.number(), z.boolean(), z.object({}).passthrough(), z.array(z.unknown()), z.null()]),\n }),\n )\n .nonempty();\n\nexport const combinedTypedDataSchema = typedDataSchema.or(typedDataV1Schema);\n\nconst dataSchema = z.union([z.string().describe('data string'), typedDataSchema]);\n\nexport const combinedDataSchema = z.union([z.string().describe('data string'), combinedTypedDataSchema]);\n\nexport const ethSignTypedDataSchema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA),\n params: z.tuple([addressSchema, combinedDataSchema]),\n});\n\nexport const ethSignTypedDataV1Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V1),\n params: z.tuple([addressSchema, combinedDataSchema]),\n});\n\nexport const ethSignTypedDataV3Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V3),\n params: z.tuple([addressSchema, dataSchema]),\n});\n\nexport const ethSignTypedDataV4Schema = z.object({\n method: z.literal(RpcMethod.SIGN_TYPED_DATA_V4),\n params: z.tuple([addressSchema, dataSchema]),\n});\n","import { z } from 'zod';\nimport { addressSchema, messageSchema } from './shared';\nimport { RpcMethod } from '@avalabs/vm-module-types';\n\n// https://github.com/ethereum/go-ethereum/pull/2940\nexport const personalSignSchema = z.object({\n method: z.literal(RpcMethod.PERSONAL_SIGN),\n params: z.union([\n z.tuple([messageSchema, addressSchema]),\n z.tuple([messageSchema, addressSchema, z.string().optional().describe('password')]),\n ]),\n});\n","import type { TypedDataV1, TypedData, MessageTypes } from '@avalabs/vm-module-types';\nimport { typedDataSchema, typedDataV1Schema } from '../schemas/eth-sign-typed-data';\n\nexport const isTypedDataV1 = (data: unknown): data is TypedDataV1 => {\n return typedDataV1Schema.safeParse(data).success;\n};\n\nexport const isTypedData = (data: unknown): data is TypedData<MessageTypes> => {\n return typedDataSchema.safeParse(data).success;\n};\n","import { type TypedData, type MessageTypes } from '@avalabs/vm-module-types';\nimport { TypedDataEncoder } from 'ethers';\n\ntype Result = { isValid: true } | { isValid: false; error: unknown };\n\nexport const isTypedDataValid = (data: TypedData<MessageTypes>): Result => {\n try {\n // getPayload verifies the types and the content of the message throwing an error if the data is not valid.\n // We don't want to immediately reject the request even if there are errors for compatiblity reasons.\n // dApps tend to make small mistakes in the message format like leaving the verifyingContract emptry,\n // in which cases we should be able to continue just like other wallets do (even if it's technically incorrect).\n\n // remove EIP712Domain from types since ethers.js handles it separately\n const { EIP712Domain, ...types } = data.types;\n TypedDataEncoder.getPayload(data.domain, types, data.message);\n\n return {\n isValid: true,\n };\n } catch (e) {\n return {\n isValid: false,\n error: e,\n };\n }\n};\n","import type { Network, RpcRequest } from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getProvider } from '../../utils/get-provider';\n\nexport const forwardToRpcNode = async (request: RpcRequest, network: Network) => {\n try {\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n const response = await provider.send(request.method, request.params as unknown[]);\n return { result: response };\n } catch (error) {\n // extracting the error message based on the error object structure from ethers lib\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const message = (error as any).info?.error?.message || (error as any).error?.message || (error as Error).message;\n return { error: rpcErrors.internal(message) };\n }\n};\n","import { NetworkVMType, WalletType, type GetAddressParams, type GetAddressResponse } from '@avalabs/vm-module-types';\nimport { getAddressFromXPub, getEvmAddressFromPubKey } from '@avalabs/core-wallets-sdk';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\ntype GetAddress = Omit<GetAddressParams, 'isTestnet' | 'xpubXP'>;\n\nexport const getAddress = async ({ accountIndex, xpub, walletType }: GetAddress): Promise<GetAddressResponse> => {\n switch (walletType) {\n case WalletType.Mnemonic:\n case WalletType.Ledger:\n case WalletType.Keystone: {\n return {\n [NetworkVMType.EVM]: getAddressFromXPub(xpub, accountIndex),\n };\n }\n case WalletType.LedgerLive:\n case WalletType.Seedless: {\n const pubKeyBuffer = Buffer.from(xpub, 'hex');\n return {\n [NetworkVMType.EVM]: getEvmAddressFromPubKey(pubKeyBuffer),\n };\n }\n default:\n throw rpcErrors.invalidParams(`Unsupported wallet type: ${walletType}`);\n }\n};\n","import {\n NetworkVMType,\n type ApprovalController,\n type DeriveAddressParams,\n type DeriveAddressResponse,\n} from '@avalabs/vm-module-types';\nimport { computeAddress } from 'ethers';\n\nimport { hasDerivationDetails } from '@internal/utils/src/utils/address-derivation';\nimport { buildDerivationPath } from '../build-derivation-path/build-derivation-path';\n\nexport const deriveAddress = async (\n params: DeriveAddressParams & { approvalController: ApprovalController },\n): Promise<DeriveAddressResponse> => {\n const { secretId, approvalController } = params;\n\n // When dealing with single-account private keys, we don't need the derivation path any more.\n const derivationPath = hasDerivationDetails(params) ? buildDerivationPath(params).EVM : undefined;\n const publicKeyHex = await approvalController.requestPublicKey({\n curve: 'secp256k1',\n secretId,\n derivationPath,\n });\n\n return {\n [NetworkVMType.EVM]: computeAddress(`0x${publicKeyHex}`), // ApprovalController does not return the 0x prefix\n };\n};\n","import type { DeriveAddressParams, DetailedDeriveAddressParams } from '@avalabs/vm-module-types';\n\nexport const hasDerivationDetails = (params: DeriveAddressParams): params is DetailedDeriveAddressParams =>\n 'derivationPathType' in params &&\n 'accountIndex' in params &&\n typeof params.accountIndex === 'number' &&\n typeof params.derivationPathType === 'string';\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport {\n NetworkVMType,\n type BuildDerivationPathParams,\n type BuildDerivationPathResponse,\n} from '@avalabs/vm-module-types';\n\nexport const buildDerivationPath = ({\n accountIndex,\n derivationPathType,\n}: BuildDerivationPathParams): Pick<BuildDerivationPathResponse, NetworkVMType.EVM> => {\n if (accountIndex < 0) {\n throw rpcErrors.invalidParams('Account index must be a non-negative integer');\n }\n\n switch (derivationPathType) {\n case 'bip44':\n return {\n [NetworkVMType.EVM]: `m/44'/60'/0'/0/${accountIndex}`,\n };\n\n case 'ledger_live':\n return {\n [NetworkVMType.EVM]: `m/44'/60'/${accountIndex}'/0/0`,\n };\n\n default:\n throw rpcErrors.invalidParams(`Unsupported derivation path type: ${derivationPathType}`);\n }\n};\n","import { CurrencyCode, NftTokenMetadataStatus } from '@avalabs/glacier-sdk';\nimport type { BalanceServiceInterface, TokenId } from '@src/handlers/get-balances/balance-service-interface';\nimport { TokenUnit } from '@avalabs/core-utils-sdk';\nimport { isHexString } from 'ethers';\nimport {\n type Error,\n type NetworkTokenWithBalance,\n type NftTokenWithBalance,\n TokenType,\n type TokenWithBalanceEVM,\n} from '@avalabs/vm-module-types';\nimport { DeBank, type DeBankChainInfo, type DeBankNftToken } from './de-bank';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getExchangeRates } from '@internal/utils';\nimport { getSmallImageForNFT } from '@src/utils/get-small-image-for-nft';\n\nexport class DeBankService implements BalanceServiceInterface {\n #deBank: DeBank;\n\n constructor({ proxyApiUrl }: { proxyApiUrl: string }) {\n this.#deBank = new DeBank(`${proxyApiUrl}/proxy/debank`);\n }\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n return this.#deBank.isNetworkSupported(chainId);\n }\n\n async #getChainInfo(chainId: number): Promise<{ chainInfo: DeBankChainInfo; chainIdString: string }> {\n const chainList = await this.#deBank.getChainList();\n const chainIdString = chainList.find((value) => value.community_id === chainId)?.id;\n if (!chainIdString) {\n throw rpcErrors.invalidParams('getNativeBalance: not valid chainId: ' + chainId);\n }\n\n const chainInfo = await this.#deBank.getChainInfo({ chainId: chainIdString });\n\n if (!chainInfo) {\n throw rpcErrors.invalidParams('getNativeBalance: not valid chainId: ' + chainId);\n }\n\n return {\n chainInfo,\n chainIdString,\n };\n }\n\n async getNativeBalance({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n }): Promise<NetworkTokenWithBalance> {\n if (!isHexString(address)) throw rpcErrors.invalidParams('getNativeBalance: not valid address: ' + address);\n const { chainInfo, chainIdString } = await this.#getChainInfo(chainId);\n\n const tokenId = chainInfo.native_token_id;\n const nativeTokenBalance = await this.#deBank.getTokenBalance({ address, chainId: chainIdString, tokenId });\n const tokenUnit = new TokenUnit(\n nativeTokenBalance.raw_amount,\n nativeTokenBalance.decimals,\n nativeTokenBalance.symbol,\n );\n const balanceDisplayValue = tokenUnit.toDisplay();\n const exchangeRates = await getExchangeRates();\n const usdToCurrencyRate = exchangeRates.usd[currency.toLowerCase()];\n const priceInCurrency = usdToCurrencyRate ? usdToCurrencyRate * nativeTokenBalance.price : undefined;\n const balanceCurrencyDisplayValue = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2 })\n : undefined;\n const balanceInCurrency = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n return {\n name: nativeTokenBalance.name,\n symbol: nativeTokenBalance.symbol,\n decimals: nativeTokenBalance.decimals,\n type: TokenType.NATIVE,\n logoUri: nativeTokenBalance.logo_url,\n balance: tokenUnit.toSubUnit(),\n balanceDisplayValue,\n balanceInCurrency,\n balanceCurrencyDisplayValue,\n priceInCurrency,\n } as NetworkTokenWithBalance;\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n }: {\n chainId: number;\n address: string;\n currency: CurrencyCode;\n pageSize: number;\n pageToken?: string;\n }): Promise<Record<TokenId, TokenWithBalanceEVM | Error>> {\n if (!isHexString(address)) throw rpcErrors.invalidParams('listErc20Balances: not valid address');\n const { chainInfo, chainIdString } = await this.#getChainInfo(chainId);\n\n const tokenBalances = await this.#deBank.getTokensBalanceOnChain({ chainId: chainIdString, address });\n const exchangeRates = await getExchangeRates();\n\n const erc20TokenBalances: Record<TokenId, TokenWithBalanceEVM | Error> = {};\n for (const tokenBalance of tokenBalances) {\n // skip native token or tokens which are not core tokens\n if (tokenBalance.id === chainInfo.native_token_id || tokenBalance.is_core === false) {\n continue;\n }\n\n const tokenUnit = new TokenUnit(tokenBalance.raw_amount, tokenBalance.decimals, tokenBalance.symbol);\n const balanceDisplayValue = tokenUnit.toDisplay();\n const usdToCurrencyRate = exchangeRates.usd[currency.toLowerCase()];\n const priceInCurrency = usdToCurrencyRate ? usdToCurrencyRate * tokenBalance.price : undefined;\n const balanceCurrencyDisplayValue = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2 })\n : undefined;\n const balanceInCurrency = priceInCurrency\n ? tokenUnit.mul(priceInCurrency).toDisplay({ fixedDp: 2, asNumber: true })\n : undefined;\n\n erc20TokenBalances[tokenBalance.id] = {\n chainId: chainInfo.community_id,\n address: tokenBalance.id,\n name: tokenBalance.name,\n symbol: tokenBalance.symbol,\n decimals: tokenBalance.decimals,\n logoUri: tokenBalance.logo_url,\n balance: tokenUnit.toSubUnit(),\n balanceCurrencyDisplayValue,\n balanceDisplayValue,\n balanceInCurrency,\n priceInCurrency,\n type: TokenType.ERC20,\n reputation: null,\n };\n }\n\n return erc20TokenBalances;\n }\n\n #mapNftList(deBankNftList: DeBankNftToken[], chainId: number): Record<string, NftTokenWithBalance | Error> {\n return deBankNftList.reduce(\n (accumulator, token) => ({\n ...accumulator,\n [`${token.contract_id}-${token.id}`]: {\n chainId,\n address: token.contract_id,\n description: token.description ?? '',\n logoUri: token.thumbnail_url,\n logoSmall: getSmallImageForNFT(token.content),\n name: token.name,\n symbol: '',\n tokenId: `${token.inner_id}`,\n tokenUri: token.detail_url,\n collectionName: token.collection_name,\n balance: BigInt(token.amount),\n balanceDisplayValue: `${token.amount}`,\n type: token.is_erc721 ? TokenType.ERC721 : TokenType.ERC1155,\n metadata: {\n indexStatus: NftTokenMetadataStatus.UNINDEXED,\n description: token.description ?? '',\n properties: '',\n },\n },\n }),\n {} as Record<string, NftTokenWithBalance>,\n );\n }\n\n async listNftBalances({\n chainId,\n address,\n }: {\n chainId: number;\n address: string;\n }): Promise<Record<string, NftTokenWithBalance | Error>> {\n if (!isHexString(address)) {\n throw rpcErrors.invalidParams('listNftBalances: not valid address');\n }\n const { chainIdString } = await this.#getChainInfo(chainId);\n\n const nftList = await this.#deBank.getNftList({ chainId: chainIdString, address });\n return this.#mapNftList(nftList, chainId);\n }\n}\n","import type { Hex } from '@avalabs/vm-module-types';\nimport { isHexString } from 'ethers';\nimport { fetchAndVerify } from '@internal/utils';\nimport { z } from 'zod';\n\nexport class DeBank {\n #supportedChainIds: DeBankChainInfo[] = [];\n\n constructor(private baseUrl: string) {}\n\n async isNetworkSupported(chainId: number): Promise<boolean> {\n const chainList: DeBankChainInfo[] = await this.getChainList();\n const chainIds = chainList.map((value) => value.community_id);\n return chainIds.some((id) => id === chainId);\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n */\n async getChainInfo({ chainId }: { chainId: string }): Promise<DeBankChainInfo | undefined> {\n const chainList = await this.getChainList();\n return chainList.find((chain) => chain.id === chainId);\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n * @param tokenId - The address of the token contract or a native token id (eth, matic, bsc)\n */\n async getTokenBalance({\n chainId,\n address,\n tokenId,\n }: {\n chainId: string;\n address: Hex;\n tokenId: Hex | string;\n }): Promise<DeBankToken> {\n const tokenBalanceResponse = await fetch(\n `${this.baseUrl}/v1/user/token?id=${address}&chain_id=${chainId}&token_id=${tokenId}`,\n );\n if (tokenBalanceResponse.ok) {\n return await tokenBalanceResponse.json();\n } else {\n throw new Error(`${tokenBalanceResponse.status}:${tokenBalanceResponse.statusText}`);\n }\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getTokensBalanceOnChain({ chainId, address }: { chainId: string; address: Hex }): Promise<DeBankToken[]> {\n const tokenBalanceResponse = await fetch(`${this.baseUrl}/v1/user/token_list?id=${address}&chain_id=${chainId}`);\n if (tokenBalanceResponse.ok) {\n return await tokenBalanceResponse.json();\n } else {\n throw new Error(`${tokenBalanceResponse.status}:${tokenBalanceResponse.statusText}`);\n }\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getTokenList({ chainId, address }: { chainId: string; address: Hex }): Promise<DeBankToken[]> {\n const response = await fetch(`${this.baseUrl}/v1/user/token_list?id=${address}&chain_id=${chainId}`);\n if (response.ok) {\n return await response.json();\n } else {\n throw new Error(`${response.status}:${response.statusText}`);\n }\n }\n\n /**\n * @returns Cached chain list if there is any, or tries to fetch it from API\n */\n async getChainList(): Promise<DeBankChainInfo[]> {\n if (this.#supportedChainIds.length === 0) {\n const chainListResponse = await fetch(`${this.baseUrl}/v1/chain/list`);\n if (chainListResponse.ok) {\n this.#supportedChainIds = await chainListResponse.json();\n } else {\n throw new Error(`${chainListResponse.status}:${chainListResponse.statusText}`);\n }\n }\n return this.#supportedChainIds;\n }\n\n /**\n * @param chainId - DeBank chain id (ex. \"base\", \"eth\")\n * @param address - account address\n */\n async getNftList({ address, chainId }: { chainId: string; address: Hex }): Promise<DeBankNftToken[]> {\n return fetchAndVerify<typeof DeBankNftTokenListSchema>(\n [`${this.baseUrl}/v1/user/nft_list?id=${address}&chain_id=${chainId}`],\n DeBankNftTokenListSchema,\n );\n }\n}\n\nconst BaseDeBankTokenSchema = z.object({\n id: z.union([z.custom<string>(isHexString), z.string()]), //address or native token id (eth, matic, bsc)\n chain: z.string(),\n name: z.string(),\n});\n\ntype BaseDeBankToken = z.infer<typeof BaseDeBankTokenSchema>;\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"0x19225f002b65eefb22950b9739fc8b448d900d44\",\n * \"chain\": \"base\",\n * \"name\": \"BASE\",\n * \"symbol\": \"Collect on: swap-based.com\",\n * \"display_symbol\": null,\n * \"optimized_symbol\": \"Collect on: swap-based.com\",\n * \"decimals\": 0,\n * \"logo_url\": null,\n * \"protocol_id\": \"\",\n * \"price\": 0,\n * \"price_24h_change\": null,\n * \"is_verified\": false,\n * \"is_core\": false,\n * \"is_wallet\": false,\n * \"time_at\": 1712662101,\n * \"amount\": 250000,\n * \"raw_amount\": 250000,\n * \"raw_amount_hex_str\": \"0x3d090\"\n * }\n * ```\n */\n\n// TODO: create zod schema\nexport type DeBankToken = BaseDeBankToken & {\n symbol: string;\n optimized_symbol: string;\n decimals: number;\n logo_url: string;\n protocol_id: string;\n price: number;\n is_core: boolean;\n is_wallet: boolean;\n time_at: number;\n amount: bigint;\n raw_amount: bigint;\n};\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"eth\",\n * \"community_id\": 1,\n * \"name\": \"Ethereum\",\n * \"native_token_id\": \"eth\",\n * \"logo_url\": \"https://static.debank.com/image/chain/logo_url/eth/42ba589cd077e7bdd97db6480b0ff61d.png\",\n * \"wrapped_token_id\": \"0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2\",\n * \"is_support_pre_exec\": true\n * }\n * ```\n */\n// TODO: create zod schema\nexport type DeBankChainInfo = {\n id: string;\n community_id: number;\n name: string;\n native_token_id: string;\n logo_url: string;\n wrapped_token_id: Hex;\n is_support_pre_exec: boolean;\n};\n\n/**\n * Example:\n * ```json\n * {\n * \"id\": \"defc948fbe6d3b138b49bf981e276f0b\",\n * \"contract_id\": \"0x495f947276749ce646f68ac8c248420045cb7b5e\",\n * \"inner_id\": \"55575360221028374465659771733000318579577403829328624053715758637886677712897\",\n * \"chain\": \"eth\",\n * \"name\": \"A New Era has begun\",\n * \"description\": \"3 of 9\\n\\nOn February 8, 2021, one of the most influential men in the world decided to invest in Bitcoin. Elon Musk, owner of Tesl\",\n * \"content_type\": \"image_url\",\n * \"content\": \"https://lh3.googleusercontent.com/WQnK8JxSSPj5YIxegh9iaprMaMmv-JswrcnTp9Mi5PXKDWmigkOzTBBIAkhdXtLPe7EwIe6Q1gi2gdtLzV08d2y67rMVTHx0Ei0S\",\n * \"detail_url\": \"https://opensea.io/assets/0x495f947276749ce646f68ac8c248420045cb7b5e/55575360221028374465659771733000318579577403829328624053715758637886677712897\",\n * \"contract_name\": \"OpenSea Shared Storefront\",\n * \"is_erc1155\": true,\n * \"amount\": 1,\n * \"protocol\": {\n * \"id\": \"opensea\",\n * \"chain\": \"eth\",\n * \"name\": \"OpenSea\",\n * \"site_url\": \"https://opensea.io\",\n * \"logo_url\": \"https://static.debank.com/image/project/logo_url/opensea/4b23246fac2d4ce53bd8e8079844821c.png\",\n * \"has_supported_portfolio\": false,\n * \"tvl\": 114295.77061458935\n * },\n * \"pay_token\": {\n * \"id\": \"eth\",\n * \"chain\": \"eth\",\n * \"name\": \"ETH\",\n * \"symbol\": \"ETH\",\n * \"display_symbol\": null,\n * \"optimized_symbol\": \"ETH\",\n * \"decimals\": 18,\n * \"logo_url\": \"https://static.debank.com/image/token/logo_url/eth/935ae4e4d1d12d59a99717a24f2540b5.png\",\n * \"protocol_id\": \"\",\n * \"price\": 2510.46,\n * \"is_verified\": true,\n * \"is_core\": true,\n * \"is_wallet\": true,\n * \"time_at\": 1628248886,\n * \"amount\": 0.0178,\n * \"date_at\": \"2021-08-06\"\n * },\n * \"attributes\": [\n * {\n * \"trait_type\": \"Artist\",\n * \"value\": \"SpaceTurtleShip\"\n * },\n * {\n * \"trait_type\": \"Edition\",\n * \"value\": \"1\"\n * }\n * ],\n * \"usd_price\": 51.492552,\n * \"collection_id\": null\n * }\n * ```\n * */\n\nconst DeBankNftTokenSchema = BaseDeBankTokenSchema.merge(\n z.object({\n contract_id: z.string(),\n description: z.string().nullable(),\n content_type: z.string().nullable(),\n content: z.string(),\n thumbnail_url: z.string(),\n total_supply: z.number(),\n detail_url: z.string(),\n collection_id: z.string(),\n is_core: z.boolean(),\n inner_id: z.string(),\n collection_name: z.string(),\n contract_name: z.string(),\n amount: z.number(),\n usd_price: z.number().optional(),\n is_erc721: z.boolean().optional(),\n is_erc1155: z.boolean().optional(),\n }),\n);\n\nconst DeBankNftTokenListSchema = z.array(DeBankNftTokenSchema);\n\nexport type DeBankNftToken = z.infer<typeof DeBankNftTokenSchema>;\n","import {\n type Network,\n type RpcRequest,\n type DisplayData,\n type BatchApprovalController,\n type Hex,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\n\nimport { getProvider } from '../../utils/get-provider';\n\nimport { parseRequestParams } from './schema';\nimport { ensureProperNonces } from './utils/ensure-proper-nonces';\nimport { buildTxApprovalRequest } from '../../utils/build-tx-approval-request';\nimport { getTxHash } from '../../utils/get-tx-hash';\nimport { waitForTransactionReceipt } from '../../utils/wait-for-transaction-receipt';\nimport { getTxBatchUpdater } from '../../utils/evm-tx-batch-updater';\nimport { simulateTransactionBatch } from './utils/process-transaction-batch-simulation';\nimport { addressItem, linkItem, networkItem } from '@internal/utils/src/utils/detail-item';\nimport type Blockaid from '@blockaid/client';\n\nexport const ethSendTransactionBatch = async ({\n request,\n network,\n approvalController,\n blockaid,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: BatchApprovalController;\n blockaid: Blockaid;\n}) => {\n const { params } = request;\n const { data: parsedParams, success, error } = parseRequestParams(params);\n\n if (!success) {\n console.error('invalid params', error);\n return {\n error: rpcErrors.invalidParams({ message: 'Transaction params are invalid', data: { cause: error } }),\n };\n }\n\n const { transactions: transactionRequests, options = {} } = parsedParams;\n const { skipIntermediateTxs = false } = options;\n\n const provider = await getProvider({\n chainId: network.chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n pollingInterval: 1000,\n });\n\n // Ensure all transactions in the batch have the nonce set\n try {\n await ensureProperNonces(transactionRequests, provider);\n } catch (error) {\n console.error('Unable to calculate nonce', error);\n return {\n error: rpcErrors.internal({\n message: 'Unable to calculate nonce',\n data: {\n originalError: error,\n },\n }),\n };\n }\n\n const allTransactionPayloadsHaveGas = transactionRequests.every((tx) => Number(tx.gas) > 0);\n const { alert, balanceChange, isSimulationSuccessful, tokenApprovals, scans } = await simulateTransactionBatch({\n rpcMethod: request.method,\n chainId: network.chainId,\n params: transactionRequests,\n dAppUrl: request.dappInfo.url,\n provider,\n populateMissingGas: !allTransactionPayloadsHaveGas,\n blockaid,\n });\n const allTransactionsHaveGas = scans.every(({ transaction }) => Number(transaction.gas) > 0);\n\n if (!allTransactionsHaveGas) {\n console.error('Gas limit is missing in some transactions');\n return {\n error: rpcErrors.internal({\n message: 'Gas limit is missing in some transactions',\n }),\n };\n }\n\n const displayData: DisplayData = {\n title: 'Do you approve these transactions?',\n details: [\n {\n title: 'Transaction Details',\n items: [\n addressItem('Account', transactionRequests[0].from),\n networkItem('Network', {\n name: network.chainName,\n logoUri: network.logoUri,\n }),\n linkItem('Website', request.dappInfo),\n ],\n },\n ],\n isSimulationSuccessful,\n balanceChange,\n alert,\n tokenApprovals,\n networkFeeSelector: true,\n };\n\n const signingRequests = scans.map((scan) => buildTxApprovalRequest(request, network, scan.transaction, scan));\n const { cleanup, updateTx } = getTxBatchUpdater(request.requestId, signingRequests, displayData);\n\n const response = await approvalController.requestBatchApproval({ request, signingRequests, displayData, updateTx });\n\n cleanup();\n\n if ('error' in response) {\n return {\n error: response.error,\n };\n }\n\n if (response.result.length !== transactionRequests.length) {\n return {\n error: rpcErrors.internal({\n message: `Invalid number of signatures. Expected ${transactionRequests.length}, got ${response.result.length}`,\n }),\n };\n }\n\n const txHashes: Hex[] = [];\n\n for (const [index, result] of response.result.entries()) {\n const isLast = index === response.result.length - 1;\n const txHash = await getTxHash(provider, result);\n\n // For the last tx: fire-and-forget receipt tracking (for UI callbacks)\n if (isLast) {\n waitForTransactionReceipt({\n explorerUrl: network.explorerUrl ?? '',\n provider,\n txHash,\n onTransactionPending: approvalController.onTransactionPending,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n request,\n });\n }\n\n // For intermediate txs: wait for receipt before broadcasting next (unless skipIntermediateTxs)\n if (!isLast && !skipIntermediateTxs) {\n const isSuccess = await waitForTransactionReceipt({\n explorerUrl: network.explorerUrl ?? '',\n provider,\n txHash,\n onTransactionPending: approvalController.onTransactionPending,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n request,\n });\n\n if (!isSuccess) {\n return {\n error: rpcErrors.internal({\n message: `Transaction ${index + 1} failed! Batch execution stopped`,\n }),\n };\n }\n }\n\n txHashes.push(txHash);\n }\n\n return {\n result: txHashes,\n };\n};\n","import { z } from 'zod';\nimport { transactionSchema } from '../../utils/transaction-schema';\n\nexport const transactionArraySchema = z\n .tuple([transactionSchema, transactionSchema])\n .rest(transactionSchema)\n .refine((transactions) => areAllEqualByProp(transactions, 'chainId'), {\n message: 'All transactions must use the same \"chainId\"',\n })\n .refine((transactions) => areAllEqualByProp(transactions, 'from'), {\n message: 'All transactions must use the same \"from\" address',\n })\n .refine((transactions) => areAllEmptyByProp(transactions, 'nonce') || areAllDifferentByProp(transactions, 'nonce'), {\n message: `Each transaction needs a different \"nonce\". Make them different, or leave them empty for all transactions.`,\n });\n\nexport const batchOptionsSchema = z.object({\n /**\n * When true, skips waiting for intermediate transaction receipts.\n * All transactions are broadcasted immediately without waiting for each to confirm.\n *\n * This is useful for one-click swaps (approve + swap) where we want fast response.\n * The network guarantees ordering via sequential nonces.\n *\n * Default: false (wait for each transaction's receipt before broadcasting the next)\n */\n skipIntermediateTxs: z.boolean().optional(),\n});\n\nexport type BatchOptions = z.infer<typeof batchOptionsSchema>;\n\n/**\n * Support both legacy array format and new object format with options:\n * - Legacy: [tx1, tx2, ...]\n * - New: { transactions: [tx1, tx2, ...], options?: { onlyWaitForLastTx?: boolean } }\n */\nexport const transactionBatchSchema = z.union([\n transactionArraySchema,\n z.object({\n transactions: transactionArraySchema,\n options: batchOptionsSchema.optional(),\n }),\n]);\n\nfunction areAllDifferentByProp<T>(elements: T[], prop: keyof T) {\n return !areAllEqualByProp(elements, prop);\n}\n\nfunction areAllEmptyByProp<T>(elements: T[], prop: keyof T) {\n return elements.every((el) => el[prop] === undefined || el[prop] === '');\n}\n\nfunction areAllEqualByProp<T>(elements: T[], prop: keyof T) {\n const uniqValues = new Set(elements.map((el) => el[prop]));\n\n return uniqValues.size === 1;\n}\n\n/** Input type for batch transaction params - either array or object format */\nexport type TransactionBatchInput = z.input<typeof transactionBatchSchema>;\n\n/** Normalized output after parsing - always has transactions, options is optional */\nexport type ParsedParams = {\n transactions: z.infer<typeof transactionArraySchema>;\n options?: BatchOptions;\n};\n\nexport const parseRequestParams = (params: unknown): z.SafeParseReturnType<TransactionBatchInput, ParsedParams> => {\n const result = transactionBatchSchema.safeParse(params);\n\n if (!result.success) {\n return result;\n }\n\n // Normalize to the new format\n if (Array.isArray(result.data)) {\n return {\n success: true,\n data: {\n transactions: result.data,\n options: {},\n },\n };\n }\n\n return {\n success: true,\n data: {\n transactions: result.data.transactions,\n options: result.data.options ?? {},\n },\n };\n};\n","import type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport { getNonce } from '../../../utils/get-nonce';\nimport type { TransactionBatchParams } from '../../../types';\n\nexport const ensureProperNonces = async (transactions: TransactionBatchParams, provider: JsonRpcBatchInternal) => {\n // The parameters schema ensures all transactions have empty or different nonces,\n // so if we find the first one is populated, we know the rest should be both populated & unique.\n if (transactions[0].nonce) {\n return;\n }\n const nonce = await getNonce({\n from: transactions[0].from,\n provider,\n });\n\n return Promise.all(\n transactions.map(async (transaction, index) => {\n transaction.nonce = String(nonce + index);\n }),\n );\n};\n","import { rpcErrors } from '@metamask/rpc-errors';\nimport type { DisplayData, EvmTxBatchUpdateFn, SigningData_EthSendTx } from '@avalabs/vm-module-types';\n\ntype SigningRequests = {\n displayData: DisplayData;\n signingData: SigningData_EthSendTx;\n}[];\n\nconst requests = new Map<\n string,\n {\n signingRequests: SigningRequests;\n displayData: DisplayData;\n }\n>();\n\nexport const getTxBatchUpdater = (\n requestId: string,\n signingRequests: SigningRequests,\n displayData: DisplayData,\n): { updateTx: EvmTxBatchUpdateFn; cleanup: () => void } => {\n requests.set(requestId, { signingRequests, displayData });\n\n return {\n updateTx: ({ maxFeeRate, maxTipRate }, txIndex) => {\n const request = requests.get(requestId);\n\n if (!request) {\n throw rpcErrors.resourceNotFound();\n }\n\n const { signingRequests } = request;\n\n const newSigningRequests = signingRequests.map((signingRequest, index) => {\n if (txIndex === undefined || txIndex === index) {\n const { signingData } = signingRequest;\n const newSigningData = {\n ...signingData,\n data: {\n ...signingData.data,\n maxFeePerGas: maxFeeRate ?? signingData.data.maxFeePerGas,\n maxPriorityFeePerGas: maxTipRate ?? signingData.data.maxPriorityFeePerGas,\n },\n };\n\n return { signingData: newSigningData, displayData: signingRequest.displayData };\n }\n\n return signingRequest;\n });\n\n const updatedRequest = { signingRequests: newSigningRequests, displayData };\n\n requests.set(requestId, updatedRequest);\n return updatedRequest;\n },\n cleanup: () => requests.delete(requestId),\n };\n};\n","import {\n AlertType,\n RpcMethod,\n type Alert,\n type BalanceChange,\n type TokenApprovals,\n type TokenDiff,\n type TransactionSimulationResult,\n} from '@avalabs/vm-module-types';\nimport type { JsonRpcBatchInternal } from '@avalabs/core-wallets-sdk';\n\nimport type { TransactionBatchParams, TransactionParams } from '../../../types';\nimport { isEmpty, isNetworkToken } from '../../../utils/type-utils';\nimport { processTransactionSimulation } from '../../../utils/process-transaction-simulation';\nimport { scanTransactionBatch } from '../../../utils/scan-transaction';\nimport { transactionAlerts } from '../../../utils/transaction-alerts';\nimport type Blockaid from '@blockaid/client';\n\nexport const simulateTransactionBatch = async ({\n rpcMethod,\n dAppUrl,\n params,\n chainId,\n provider,\n populateMissingGas,\n blockaid,\n}: {\n rpcMethod: RpcMethod;\n dAppUrl?: string;\n params: TransactionBatchParams;\n chainId: number;\n provider: JsonRpcBatchInternal;\n populateMissingGas?: boolean;\n blockaid: Blockaid;\n}): Promise<\n TransactionSimulationResult & { scans: (TransactionSimulationResult & { transaction: TransactionParams })[] }\n> => {\n let simulationResults;\n\n try {\n simulationResults = await scanTransactionBatch({\n chainId,\n params,\n domain: dAppUrl,\n withGasEstimation: populateMissingGas,\n blockaid,\n });\n } catch (error) {\n console.error('simulateTransactionBatch error', error);\n return {\n isSimulationSuccessful: false,\n alert: transactionAlerts[AlertType.WARNING],\n scans: [],\n };\n }\n\n const scans = await Promise.all(\n simulationResults.map(async (simulationResult, index) => {\n const matchingTx = params[index]!;\n const processedResult = await processTransactionSimulation({\n chainId,\n params: params[index]!,\n provider,\n rpcMethod,\n simulationResult,\n });\n\n if (populateMissingGas && !matchingTx.gas && processedResult.estimatedGasLimit) {\n matchingTx.gas = `0x${processedResult.estimatedGasLimit.toString(16)}`;\n }\n\n return {\n transaction: matchingTx,\n ...processedResult,\n };\n }),\n );\n\n // Make sure `isSimulatioSuccessful` is only true for a batch if all nested transactions were simulated successfully\n const isSimulationSuccessful = scans.every((scan) => scan.isSimulationSuccessful);\n // Make sure the alert in the aggregated view is of the upmost priority (DANGER -> WARNING -> INFO)\n const alert = getHighestAlert(scans);\n const balanceChange = aggregateBalanceChange(scans);\n const tokenApprovals = aggregateTokenApprovals(scans);\n\n return {\n isSimulationSuccessful,\n alert,\n balanceChange,\n tokenApprovals,\n // For now, we're disabling the editing of approvals in the aggregated view since\n // it could disrupt the execution of the entire batch.\n scans: scans.map((scan) => {\n if (!scan.tokenApprovals) {\n return scan;\n }\n\n scan.tokenApprovals.isEditable = false;\n return scan;\n }),\n };\n};\n\nconst getHighestAlert = (scans: { alert?: Alert }[]): Alert | undefined => {\n const alertPrio = {\n [AlertType.INFO]: 0,\n [AlertType.WARNING]: 1,\n [AlertType.DANGER]: 2,\n };\n\n return scans.reduce(\n (highest, { alert }) => {\n if (!alert) return highest;\n if (!highest || alertPrio[alert.type] > alertPrio[highest.type]) {\n return alert;\n }\n return highest;\n },\n undefined as Alert | undefined,\n );\n};\n\nconst aggregateTokenApprovals = (scans: TransactionSimulationResult[]): TokenApprovals | undefined => {\n const tokenApprovals = scans.reduce(\n (aggregatedTokenApprovals, { tokenApprovals }) => {\n if (!tokenApprovals) {\n return aggregatedTokenApprovals;\n }\n\n tokenApprovals.approvals.forEach((appr) => {\n const aggregatedApproval = aggregatedTokenApprovals.approvals.find(\n (aggrApproval) =>\n appr.token.address === aggrApproval.token.address && appr.spenderAddress === aggrApproval.spenderAddress,\n );\n\n if (!aggregatedApproval) {\n aggregatedTokenApprovals.approvals.push({ ...appr });\n return;\n }\n\n // Replace with new values\n aggregatedApproval.value = appr.value;\n aggregatedApproval.usdPrice = appr.usdPrice;\n });\n\n return aggregatedTokenApprovals;\n },\n {\n approvals: [],\n isEditable: false,\n } as TokenApprovals,\n );\n\n clearSpentApprovals(tokenApprovals, scans);\n\n return isEmpty(tokenApprovals) ? undefined : tokenApprovals;\n};\n\nconst aggregateBalanceChange = (scans: TransactionSimulationResult[]): BalanceChange | undefined => {\n const aggregated = scans.reduce(\n (aggregatedBalanceChange, { balanceChange }) => {\n if (!balanceChange) {\n return aggregatedBalanceChange;\n }\n\n balanceChange.ins.forEach((diff) => {\n const aggregatedDiff = findTokenDiff(aggregatedBalanceChange.ins, diff);\n\n if (aggregatedDiff) {\n aggregatedDiff.items.push(...diff.items);\n } else {\n aggregatedBalanceChange.ins.push(diff);\n }\n });\n\n balanceChange.outs.forEach((diff) => {\n const aggregatedDiff = findTokenDiff(aggregatedBalanceChange.outs, diff);\n\n if (aggregatedDiff) {\n aggregatedDiff.items.push(...diff.items);\n } else {\n aggregatedBalanceChange.outs.push({ ...diff });\n }\n });\n\n return aggregatedBalanceChange;\n },\n {\n ins: [],\n outs: [],\n } as BalanceChange,\n );\n\n return isEmpty(aggregated) ? undefined : aggregated;\n};\n\nconst findTokenDiff = (diffs: TokenDiff[], lookupDiff: TokenDiff): TokenDiff | undefined =>\n diffs.find((diff) => {\n if (isNetworkToken(diff.token) && isNetworkToken(lookupDiff.token)) {\n return diff.token.symbol === lookupDiff.token.symbol;\n }\n\n if (!isNetworkToken(diff.token) && !isNetworkToken(lookupDiff.token)) {\n return diff.token.address === lookupDiff.token.address;\n }\n\n return false;\n });\n\nconst clearSpentApprovals = (tokenApprovals: TokenApprovals, scans: TransactionSimulationResult[]) => {\n // Go through each transaction. If the approvals were spent, remove them from the aggregated token approvals:\n for (const approval of tokenApprovals.approvals) {\n for (const scan of scans) {\n if (!scan.isSimulationSuccessful || !scan.balanceChange) {\n continue;\n }\n\n const touchesApproval = scan.balanceChange.outs.some((diff) => {\n return (\n !isNetworkToken(diff.token) &&\n !isNetworkToken(approval.token) &&\n diff.token.address === approval.token.address\n );\n });\n\n if (!touchesApproval) {\n continue;\n }\n\n if (!scan.tokenApprovals) {\n // Allowance was completely spent, remove the approval from aggregated view\n //\n // NOTE: With the current implementation of the BlockAid API, this is just an assumption on our part!\n // It's possible that the next transaction simply does not touch the approval at all.\n // Since we're initially be using this handler for the Swap feature, though, most of the time the allowance\n // will be spent completely.\n //\n // The BlockAid team is working to update their API to provide more accurate data.\n tokenApprovals.approvals = tokenApprovals.approvals.filter(\n (appr) => appr.spenderAddress !== approval.spenderAddress && approval.token.address !== appr.token.address,\n );\n } else {\n const latestApprovalState = scan.tokenApprovals.approvals.find(\n (appr) => appr.spenderAddress === approval.spenderAddress && appr.token.address === approval.token.address,\n );\n\n if (!latestApprovalState) {\n continue;\n }\n approval.usdPrice = latestApprovalState.usdPrice;\n approval.value = latestApprovalState.value;\n }\n }\n }\n};\n"]}