@avalabs/evm-module 0.0.13 → 0.0.15

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.
Files changed (41) hide show
  1. package/.turbo/turbo-build.log +10 -10
  2. package/.turbo/turbo-lint.log +1 -1
  3. package/.turbo/turbo-test.log +11 -6
  4. package/CHANGELOG.md +18 -0
  5. package/dist/index.cjs +10 -7
  6. package/dist/index.cjs.map +1 -1
  7. package/dist/index.d.cts +6 -5
  8. package/dist/index.d.ts +6 -5
  9. package/dist/index.js +10 -8
  10. package/dist/index.js.map +1 -1
  11. package/package.json +17 -10
  12. package/src/contracts/openzeppelin/ERC1155.ts +440 -0
  13. package/src/contracts/openzeppelin/ERC20.ts +330 -0
  14. package/src/contracts/openzeppelin/ERC721.ts +420 -0
  15. package/src/contracts/openzeppelin/common.ts +131 -0
  16. package/src/contracts/openzeppelin/factories/ERC1155__factory.ts +388 -0
  17. package/src/contracts/openzeppelin/factories/ERC20__factory.ts +353 -0
  18. package/src/contracts/openzeppelin/factories/ERC721__factory.ts +413 -0
  19. package/src/contracts/openzeppelin/factories/index.ts +6 -0
  20. package/src/contracts/openzeppelin/index.ts +10 -0
  21. package/src/handlers/eth-send-transaction/eth-send-transaction.test.ts +2 -2
  22. package/src/handlers/eth-send-transaction/eth-send-transaction.ts +1 -1
  23. package/src/handlers/get-balances/evm-balance-service/get-erc20-balances.test.ts +78 -0
  24. package/src/handlers/get-balances/evm-balance-service/get-erc20-balances.ts +85 -0
  25. package/src/handlers/get-balances/evm-balance-service/get-native-token-balances.test.ts +102 -0
  26. package/src/handlers/get-balances/evm-balance-service/get-native-token-balances.ts +54 -0
  27. package/src/handlers/get-balances/get-balances.test.ts +252 -0
  28. package/src/handlers/get-balances/get-balances.ts +109 -0
  29. package/src/handlers/get-balances/glacier-balance-service/get-erc20-balances.test.ts +76 -0
  30. package/src/handlers/get-balances/glacier-balance-service/get-erc20-balances.ts +107 -0
  31. package/src/handlers/get-balances/glacier-balance-service/get-native-token-balances.test.ts +61 -0
  32. package/src/handlers/get-balances/glacier-balance-service/get-native-token-balances.ts +46 -0
  33. package/src/handlers/get-network-fee/get-network-fee.test.ts +0 -1
  34. package/src/handlers/get-network-fee/get-network-fee.ts +0 -1
  35. package/src/handlers/get-transaction-history/converters/evm-transaction-converter/get-transaction-from-glacier.test.ts +72 -59
  36. package/src/handlers/get-transaction-history/converters/evm-transaction-converter/get-transactions-from-glacier.ts +4 -11
  37. package/src/handlers/get-transaction-history/get-transaction-history.test.ts +7 -2
  38. package/src/handlers/get-transaction-history/get-transaction-history.ts +13 -3
  39. package/src/module.ts +17 -7
  40. package/src/services/glacier-service/glacier-service.ts +259 -0
  41. package/tsconfig.json +1 -0
package/dist/index.js CHANGED
@@ -1,16 +1,18 @@
1
1
  import { rpcErrors } from '@metamask/rpc-errors';
2
- import { TransactionType, parseManifest, RpcMethod, Environment, SigningDataType, TokenType } from '@avalabs/vm-module-types';
2
+ import { TransactionType, parseManifest, RpcMethod, SigningDataType, Environment, TokenType } from '@avalabs/vm-module-types';
3
3
  import { JsonRpcBatchInternal } from '@avalabs/wallets-sdk';
4
- import { Network } from 'ethers';
5
- import { balanceToDisplayValue, ipfsResolver } from '@avalabs/utils-sdk';
6
- import { BN } from 'bn.js';
4
+ import { Network, ethers } from 'ethers';
5
+ import { numberToBN, bnToBig, balanceToDisplayValue, bigintToBig, bigToBN, ipfsResolver } from '@avalabs/utils-sdk';
6
+ import he, { BN } from 'bn.js';
7
7
  import { getNormalTxs, getErc20Txs } from '@avalabs/etherscan-sdk';
8
- import { Glacier } from '@avalabs/glacier-sdk';
9
- import St from 'lodash.startcase';
8
+ import qe from 'lodash.startcase';
10
9
  import { z as z$1 } from 'zod';
10
+ import '@avalabs/coingecko-sdk';
11
+ import { TokenService } from '@internal/utils';
12
+ import { Glacier } from '@avalabs/glacier-sdk';
11
13
 
12
- var H=(t,e,o)=>{if(!e.has(t))throw TypeError("Cannot "+o)};var N=(t,e,o)=>(H(t,e,"read from private field"),o?o.call(t):e.get(t)),P=(t,e,o)=>{if(e.has(t))throw TypeError("Cannot add the same private member more than once");e instanceof WeakSet?e.add(t):e.set(t,o);},I=(t,e,o,n)=>(H(t,e,"write to private field"),n?n.call(t,o):e.set(t,o),o);async function M({chainId:t,proxyApiUrl:e}){let o=await fetch(`${e}/tokens?evmChainId=${t}`);if(!o.ok)throw rpcErrors.internal(`Failed to fetch tokens for chainId ${t}`);return o.json()}var yt=process.env.GLACIER_API_KEY,b=t=>{let{chainId:e,chainName:o,rpcUrl:n,multiContractAddress:a,pollingInterval:s=2e3}=t,r=new JsonRpcBatchInternal({maxCalls:40,multiContractAddress:a},Et(n,yt),new Network(o,e));return r.pollingInterval=s,r},ht=["glacier-api.avax.network","proxy-api.avax.network"];function Et(t,e){let o=new URL(t);if(e&&ht.includes(o.hostname)){let n=o.searchParams;return n.set("token",e),o.search=n.toString(),o.toString()}return t}var F={LOW:1n,MEDIUM:4n,HIGH:6n},U=500000000n;async function _({chainId:t,chainName:e,rpcUrl:o,multiContractAddress:n}){let a=b({chainId:t,chainName:e,rpcUrl:o,multiContractAddress:n}),{maxFeePerGas:s}=await a.getFeeData();if(!s)throw rpcErrors.internal("Pre-EIP-1559 networks are not supported");let r=U*F.LOW,i=U*F.MEDIUM,c=U*F.HIGH;return {baseFee:s,low:{maxFeePerGas:s+r,maxPriorityFeePerGas:r},medium:{maxFeePerGas:s+i,maxPriorityFeePerGas:i},high:{maxFeePerGas:s+c,maxPriorityFeePerGas:c},isFixedFee:!1}}function E(t,e,o="tx"){return `${t}/${o}/${e}`}var B=({tx:t,networkToken:e,chainId:o,explorerUrl:n,address:a})=>{let s=t.from.toLowerCase()===a.toLowerCase(),r=parseInt(t.timeStamp)*1e3,i=e.decimals,c=new BN(t.value),m=balanceToDisplayValue(c,i),u=s?TransactionType.SEND:TransactionType.RECEIVE,{from:p,to:l,gasPrice:d,gasUsed:f,hash:g}=t,h=E(n,g);return {isIncoming:!s,isOutgoing:s,isContractCall:wt(t),timestamp:r,hash:g,isSender:s,from:p,to:l,tokens:[{decimal:i.toString(),name:e.name,symbol:e.symbol,amount:m,type:TokenType.NATIVE}],gasUsed:f,gasPrice:d,chainId:o.toString(),txType:u,explorerLink:h}};function wt(t){return t.input!=="0x"}function V({tx:t,address:e,explorerUrl:o,chainId:n}){let a=t.from.toLowerCase()===e.toLowerCase(),s=parseInt(t.timeStamp)*1e3,r=parseInt(t.tokenDecimal),i=new BN(t.value),c=balanceToDisplayValue(i,r),{from:m,to:u,gasPrice:p,gasUsed:l,hash:d,tokenDecimal:f,tokenName:g,tokenSymbol:h}=t,R=a?TransactionType.SEND:TransactionType.RECEIVE,A=E(o,d);return {isIncoming:!a,isOutgoing:a,isContractCall:!1,timestamp:s,hash:d,isSender:a,from:m,to:u,tokens:[{decimal:f,name:g,symbol:h,type:TokenType.ERC20,amount:c}],gasUsed:l,gasPrice:p,chainId:n.toString(),txType:R,explorerLink:A}}var W=async({isTestnet:t,networkToken:e,explorerUrl:o,chainId:n,address:a,nextPageToken:s,offset:r})=>{let i=s?JSON.parse(s):void 0,c=i?.page||1,m=i?.queries||["normal","erc20"],u=(m.includes("normal")?await getNormalTxs(a,!t,{page:c,offset:r}):[]).map(g=>B({tx:g,chainId:n,networkToken:e,explorerUrl:o,address:a})),p=(m.includes("erc20")?await getErc20Txs(a,!t,void 0,{page:c,offset:r}):[]).map(g=>V({tx:g,address:a,explorerUrl:o,chainId:n})),l=p.map(g=>g.hash),d=u.filter(g=>!l.includes(g.hash)),f={queries:[],page:c+1};return u.length&&f.queries.push("normal"),p.length&&f.queries.push("erc20"),{transactions:[...d,...p],nextPageToken:f.queries.length?JSON.stringify(f):""}};var q=t=>t===1||t===5||t===4||t===11155111;var $=({nativeTransaction:t,erc20Transfers:e,erc721Transfers:o},n,a)=>{let s=!e&&!o,r=At(t.method?.methodName),i=n.toLowerCase(),c=r.toLowerCase().includes("swap"),m=s&&t.from.address.toLowerCase()===i,u=s&&t.to.address.toLowerCase()===i,p=r==="Market Buy Orders With Eth"||r==="Buy NFT",l=r==="Approve",d=r.toLowerCase().includes("transfer"),f=r.toLowerCase().includes("airdrop"),g=r.toLowerCase().includes("unwrap"),h=r==="Fill Order",R=d&&!!a[0]&&j(a[0].type)&&a[0].from?.address.toLowerCase()===i,A=d&&!!a[0]&&j(a[0].type)&&a[0].to?.address.toLowerCase()===i;return c?TransactionType.SWAP:m?TransactionType.SEND:u?TransactionType.RECEIVE:p?TransactionType.NFT_BUY:l?TransactionType.APPROVE:R?TransactionType.NFT_SEND:A?TransactionType.NFT_RECEIVE:d?TransactionType.TRANSFER:f?TransactionType.AIRDROP:g?TransactionType.UNWRAP:h?TransactionType.FILL_ORDER:TransactionType.UNKNOWN};function j(t){return t===TokenType.ERC721||t===TokenType.ERC1155}var At=(t="")=>t.includes("(")?St(t.split("(",1)[0]):t;var Y=(t,{nativeTransaction:e,erc20Transfers:o,erc721Transfers:n},a)=>{let s=t===TransactionType.TRANSFER,r=t===TransactionType.SEND,i=t===TransactionType.RECEIVE,c=e?.from?.address,m=e?.to?.address;s&&o&&o[0]&&(c=o[0].from.address,m=o[0].to.address),s&&n&&n[0]&&(c=n[0].from.address,m=n[0].to.address);let u=r||s&&c.toLowerCase()===a.toLowerCase(),p=i||s&&m.toLowerCase()===a.toLowerCase();return {isOutgoing:u,isIncoming:p,isSender:c===a,from:c,to:m}};async function K(t){try{return t.then(e=>[e,null]).catch(e=>[null,e])}catch(e){return Promise.resolve([null,e])}}var Ut="https://cloudflare-ipfs.com";function v(t,e=Ut){if(!t)return "";try{return ipfsResolver(t,e)}catch{return t}}async function Dt(t,e=5e3){let o=new AbortController;return setTimeout(()=>o.abort(),e),fetch(t,{signal:o.signal})}async function z(t){let e={};if(t)if(t.startsWith("data:application/json;base64,")){let o=t.substring(29);try{let n=Buffer.from(o,"base64").toString();e=JSON.parse(n);}catch{e={};}}else e=await Dt(v(t)).then(o=>o.json()).catch(()=>({}));else return {};return e}var Lt="https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image";function Z(t,e="256"){let o=v(t);return `${Lt}/width=${e},fit/${o}`}var et=async({nativeTransaction:t,erc20Transfers:e,erc721Transfers:o,erc1155Transfers:n},a)=>{let s=[];if(t.value!=="0"){let r=a.decimals,i=new BN(t.value),c=balanceToDisplayValue(i,r);s.push({decimal:r.toString(),name:a.name,symbol:a.symbol,amount:c,from:t.from,to:t.to,type:TokenType.NATIVE});}return e?.forEach(r=>{let i=r.erc20Token.decimals,c=new BN(r.value),m=balanceToDisplayValue(c,i);s.push({decimal:i.toString(),name:r.erc20Token.name,symbol:r.erc20Token.symbol,amount:m,from:r.from,to:r.to,imageUri:r.erc20Token.logoUri,type:TokenType.ERC20});}),o&&await Promise.allSettled(o.map(async r=>{let i=r.erc721Token,c=await tt(i.tokenUri,i.metadata.imageUri);s.push({name:r.erc721Token.name,symbol:r.erc721Token.symbol,amount:"1",imageUri:c,from:r.from,to:r.to,collectableTokenId:r.erc721Token.tokenId,type:TokenType.ERC721});})),n&&await Promise.allSettled(n.map(async r=>{let i=r.erc1155Token,c=await tt(i.tokenUri,i.metadata.imageUri);s.push({name:r.erc1155Token.metadata.name??"",symbol:r.erc1155Token.metadata.symbol??"",amount:r.value,imageUri:c,from:r.from,to:r.to,collectableTokenId:r.erc1155Token.tokenId,type:TokenType.ERC1155});})),s},tt=async(t,e)=>{if(e)return e.startsWith("ipfs://")?v(e):e;{let[o,n]=await K(z(t));return n?"":o.image?Z(o.image):""}};var rt=[TransactionType.SEND,TransactionType.RECEIVE,TransactionType.TRANSFER];var ot=async({transactions:t,explorerUrl:e,networkToken:o,chainId:n,address:a})=>{let s=await et(t,o),r=$(t,a,s),{isOutgoing:i,isIncoming:c,isSender:m,from:u,to:p}=Y(r,t,a),{blockTimestamp:l,txHash:d,gasPrice:f,gasUsed:g}=t.nativeTransaction,h=E(e,d);return {isContractCall:!rt.includes(r),isIncoming:c,isOutgoing:i,isSender:m,timestamp:l*1e3,hash:d,from:u,to:p,tokens:s,gasPrice:f,gasUsed:g,chainId:n.toString(),txType:r,explorerLink:h}};var nt=async({chainId:t,explorerUrl:e,networkToken:o,address:n,nextPageToken:a,offset:s,glacierApiUrl:r})=>{if(!r)throw rpcErrors.invalidParams("Glacier API URL is required");let i=new Glacier({BASE:r});try{let c=await i.evmTransactions.listTransactions({chainId:t.toString(),address:n,pageToken:a,pageSize:s});return {transactions:(await Promise.all(c.transactions.filter(p=>p.nativeTransaction.txStatus==="1").map(p=>ot({transactions:p,explorerUrl:e,networkToken:o,chainId:t,address:n}).then(l=>l)))).filter(p=>p.tokens.find(l=>Number(l.amount)!==0)),nextPageToken:c.nextPageToken}}catch{return {transactions:[],nextPageToken:""}}};var at=async({chainId:t,isTestnet:e,networkToken:o,explorerUrl:n,address:a,nextPageToken:s,offset:r,glacierApiUrl:i})=>q(t)?W({isTestnet:e,networkToken:o,explorerUrl:n,chainId:t,address:a,nextPageToken:s,offset:r}):nt({networkToken:o,explorerUrl:n,chainId:t,address:a,nextPageToken:s,offset:r,glacierApiUrl:i});var st={name:"EVM",description:"",version:"0.0.1",sources:{module:{checksum:"",location:{npm:{filePath:"dist/bundle.js",packageName:"@avalabs/evm-module",registry:"https://registry.npmjs.org"}}},provider:{checksum:"",location:{npm:{filePath:"dist/provider.js",packageName:"@avalabs/evm-module",registry:"https://registry.npmjs.org"}}}},network:{chainIds:["eip155:1","eip155:43114","eip155:43113"],namespaces:["eip155"]},cointype:"60",permissions:{rpc:{dapps:!0,methods:["eth_sendTransaction","eth_*","getTransactionHistory"]}},manifestVersion:"0.0"};var Ot={glacierApiUrl:"https://glacier-api.avax.network",proxyApiUrl:"https://proxy-api.avax.network"},Bt={glacierApiUrl:"https://glacier-api-dev.avax.network",proxyApiUrl:"https://proxy-api-dev.avax.network"},ct=t=>{switch(t){case Environment.PRODUCTION:return Ot;case Environment.DEV:return Bt}};var Gt=z$1.object({from:z$1.string().length(42),to:z$1.string().length(42),data:z$1.string().optional(),value:z$1.string().startsWith("0x").optional(),gas:z$1.string().startsWith("0x").optional(),gasPrice:z$1.string().startsWith("0x").optional(),maxFeePerGas:z$1.string().startsWith("0x").optional(),maxPriorityFeePerGas:z$1.string().startsWith("0x").optional(),nonce:z$1.string().optional(),chainId:z$1.string().optional()}),Vt=z$1.array(Gt).length(1),mt=t=>Vt.safeParse(t);var pt=async({transactionParams:{from:t,to:e,data:o,value:n},provider:a})=>{let s=await a.getTransactionCount(t);return Number(await a.estimateGas({from:t,to:e,nonce:s,data:o,value:n}))};var lt=async({from:t,provider:e})=>e.getTransactionCount(t);var ut=async({request:t,network:e,approvalController:o})=>{let{dappInfo:n,params:a}=t,s=mt(a);if(!s.success)return console.error(s.error),{error:rpcErrors.invalidParams("Transaction params are invalid")};let r=s.data[0];if(!r)return {error:rpcErrors.invalidParams("Transaction params are invalid")};let i=b({chainId:e.chainId,chainName:e.chainName,rpcUrl:e.rpcUrl,multiContractAddress:e.utilityAddresses?.multicall,pollingInterval:1e3});if(!r.gas||Number(r.gas)<0)try{let l=await pt({transactionParams:{from:r.from,to:r.to,data:r.data,value:r.value},provider:i});r.gas="0x"+l.toString(16);}catch{return {error:rpcErrors.internal("Unable to calculate gas limit")}}if(!r.nonce)try{let l=await lt({from:r.from,provider:i});r.nonce=String(l);}catch{return {error:rpcErrors.internal("Unable to calculate nonce")}}let c={title:"Approve Transaction",network:{chainId:e.chainId,name:e.chainName,logoUrl:e.logoUrl},transactionDetails:{website:new URL(n.url).hostname,from:r.from,to:r.to,data:r.data},networkFeeSelector:!0},m={type:SigningDataType.EVM_TRANSACTION,account:r.from,chainId:e.chainId,data:{type:2,nonce:Number(r.nonce),gasLimit:Number(r.gas),to:r.to,from:r.from,data:r.data,value:r.value,chainId:r.chainId}},u=await o.requestApproval({request:t,displayData:c,signingData:m});if("error"in u)return {error:u.error};let p=await i.send("eth_sendRawTransaction",[u.result]);return qt({provider:i,txHash:p,onTransactionConfirmed:o.onTransactionConfirmed,onTransactionReverted:o.onTransactionReverted}),{result:p}},qt=async({provider:t,txHash:e,onTransactionConfirmed:o,onTransactionReverted:n})=>{(await t.waitForTransaction(e))?.status===1?o(e):n(e);};var k,x,w,gt=class{constructor({approvalController:e,environment:o}){P(this,k,void 0);P(this,x,void 0);P(this,w,void 0);let{glacierApiUrl:n,proxyApiUrl:a}=ct(o);I(this,k,n),I(this,x,a),I(this,w,e);}getAddress(){return Promise.resolve("EVM address")}getBalances(){return Promise.resolve("EVM balances")}getManifest(){let e=parseManifest(st);return e.success?e.data:void 0}getNetworkFee(e){let{chainId:o,chainName:n,rpcUrl:a,utilityAddresses:s}=e;return _({chainId:o,chainName:n,rpcUrl:a,multiContractAddress:s?.multicall,glacierApiUrl:N(this,k)})}getTransactionHistory(e){let{network:o,address:n,nextPageToken:a,offset:s}=e,{chainId:r,isTestnet:i,networkToken:c,explorerUrl:m=""}=o;return at({chainId:r,isTestnet:i,networkToken:c,explorerUrl:m,address:n,nextPageToken:a,offset:s,glacierApiUrl:N(this,k)})}getTokens(e){let{chainId:o}=e;return M({chainId:o,proxyApiUrl:N(this,x)})}async onRpcRequest(e,o){switch(e.method){case RpcMethod.ETH_SEND_TRANSACTION:return ut({request:e,network:o,approvalController:N(this,w)});default:return {error:rpcErrors.methodNotSupported(`Method ${e.method} not supported`)}}}};k=new WeakMap,x=new WeakMap,w=new WeakMap;
14
+ var xe=Object.defineProperty;var Ie=(t,e,a)=>e in t?xe(t,e,{enumerable:!0,configurable:!0,writable:!0,value:a}):t[e]=a;var E=(t,e,a)=>(Ie(t,typeof e!="symbol"?e+"":e,a),a),V=(t,e,a)=>{if(!e.has(t))throw TypeError("Cannot "+a)};var w=(t,e,a)=>(V(t,e,"read from private field"),a?a.call(t):e.get(t)),S=(t,e,a)=>{if(e.has(t))throw TypeError("Cannot add the same private member more than once");e instanceof WeakSet?e.add(t):e.set(t,a);},P=(t,e,a,n)=>(V(t,e,"write to private field"),n?n.call(t,a):e.set(t,a),a);async function A({chainId:t,proxyApiUrl:e}){let a=await fetch(`${e}/tokens?evmChainId=${t}`);if(!a.ok)throw rpcErrors.internal(`Failed to fetch tokens for chainId ${t}`);return a.json()}var Pe=process.env.GLACIER_API_KEY,C=t=>{let{chainId:e,chainName:a,rpcUrl:n,multiContractAddress:s,pollingInterval:o=2e3}=t,r=new JsonRpcBatchInternal({maxCalls:40,multiContractAddress:s},Ue(n,Pe),new Network(a,e));return r.pollingInterval=o,r},Ae=["glacier-api.avax.network","proxy-api.avax.network"];function Ue(t,e){let a=new URL(t);if(e&&Ae.includes(a.hostname)){let n=a.searchParams;return n.set("token",e),a.search=n.toString(),a.toString()}return t}var F={LOW:1n,MEDIUM:4n,HIGH:6n},D=500000000n;async function _({chainId:t,chainName:e,rpcUrl:a,multiContractAddress:n}){let s=C({chainId:t,chainName:e,rpcUrl:a,multiContractAddress:n}),{maxFeePerGas:o}=await s.getFeeData();if(!o)throw rpcErrors.internal("Pre-EIP-1559 networks are not supported");let r=D*F.LOW,i=D*F.MEDIUM,c=D*F.HIGH;return {baseFee:o,low:{maxFeePerGas:o+r,maxPriorityFeePerGas:r},medium:{maxFeePerGas:o+i,maxPriorityFeePerGas:i},high:{maxFeePerGas:o+c,maxPriorityFeePerGas:c},isFixedFee:!1}}function N(t,e,a="tx"){return `${t}/${a}/${e}`}var O=({tx:t,networkToken:e,chainId:a,explorerUrl:n,address:s})=>{let o=t.from.toLowerCase()===s.toLowerCase(),r=parseInt(t.timeStamp)*1e3,i=e.decimals,c=new BN(t.value),d=balanceToDisplayValue(c,i),p=o?TransactionType.SEND:TransactionType.RECEIVE,{from:m,to:l,gasPrice:b,gasUsed:f,hash:u}=t,y=N(n,u);return {isIncoming:!o,isOutgoing:o,isContractCall:Me(t),timestamp:r,hash:u,isSender:o,from:m,to:l,tokens:[{decimal:i.toString(),name:e.name,symbol:e.symbol,amount:d,type:TokenType.NATIVE}],gasUsed:f,gasPrice:b,chainId:a.toString(),txType:p,explorerLink:y}};function Me(t){return t.input!=="0x"}function J({tx:t,address:e,explorerUrl:a,chainId:n}){let s=t.from.toLowerCase()===e.toLowerCase(),o=parseInt(t.timeStamp)*1e3,r=parseInt(t.tokenDecimal),i=new BN(t.value),c=balanceToDisplayValue(i,r),{from:d,to:p,gasPrice:m,gasUsed:l,hash:b,tokenDecimal:f,tokenName:u,tokenSymbol:y}=t,g=s?TransactionType.SEND:TransactionType.RECEIVE,k=N(a,b);return {isIncoming:!s,isOutgoing:s,isContractCall:!1,timestamp:o,hash:b,isSender:s,from:d,to:p,tokens:[{decimal:f,name:u,symbol:y,type:TokenType.ERC20,amount:c}],gasUsed:l,gasPrice:m,chainId:n.toString(),txType:g,explorerLink:k}}var j=async({isTestnet:t,networkToken:e,explorerUrl:a,chainId:n,address:s,nextPageToken:o,offset:r})=>{let i=o?JSON.parse(o):void 0,c=i?.page||1,d=i?.queries||["normal","erc20"],p=(d.includes("normal")?await getNormalTxs(s,!t,{page:c,offset:r}):[]).map(u=>O({tx:u,chainId:n,networkToken:e,explorerUrl:a,address:s})),m=(d.includes("erc20")?await getErc20Txs(s,!t,void 0,{page:c,offset:r}):[]).map(u=>J({tx:u,address:s,explorerUrl:a,chainId:n})),l=m.map(u=>u.hash),b=p.filter(u=>!l.includes(u.hash)),f={queries:[],page:c+1};return p.length&&f.queries.push("normal"),m.length&&f.queries.push("erc20"),{transactions:[...b,...m],nextPageToken:f.queries.length?JSON.stringify(f):""}};var $=t=>t===1||t===5||t===4||t===11155111;var K=({nativeTransaction:t,erc20Transfers:e,erc721Transfers:a},n,s)=>{let o=!e&&!a,r=Je(t.method?.methodName),i=n.toLowerCase(),c=r.toLowerCase().includes("swap"),d=o&&t.from.address.toLowerCase()===i,p=o&&t.to.address.toLowerCase()===i,m=r==="Market Buy Orders With Eth"||r==="Buy NFT",l=r==="Approve",b=r.toLowerCase().includes("transfer"),f=r.toLowerCase().includes("airdrop"),u=r.toLowerCase().includes("unwrap"),y=r==="Fill Order",g=b&&!!s[0]&&z(s[0].type)&&s[0].from?.address.toLowerCase()===i,k=b&&!!s[0]&&z(s[0].type)&&s[0].to?.address.toLowerCase()===i;return c?TransactionType.SWAP:d?TransactionType.SEND:p?TransactionType.RECEIVE:m?TransactionType.NFT_BUY:l?TransactionType.APPROVE:g?TransactionType.NFT_SEND:k?TransactionType.NFT_RECEIVE:b?TransactionType.TRANSFER:f?TransactionType.AIRDROP:u?TransactionType.UNWRAP:y?TransactionType.FILL_ORDER:TransactionType.UNKNOWN};function z(t){return t===TokenType.ERC721||t===TokenType.ERC1155}var Je=(t="")=>t.includes("(")?qe(t.split("(",1)[0]):t;var X=(t,{nativeTransaction:e,erc20Transfers:a,erc721Transfers:n},s)=>{let o=t===TransactionType.TRANSFER,r=t===TransactionType.SEND,i=t===TransactionType.RECEIVE,c=e?.from?.address,d=e?.to?.address;o&&a&&a[0]&&(c=a[0].from.address,d=a[0].to.address),o&&n&&n[0]&&(c=n[0].from.address,d=n[0].to.address);let p=r||o&&c.toLowerCase()===s.toLowerCase(),m=i||o&&d.toLowerCase()===s.toLowerCase();return {isOutgoing:p,isIncoming:m,isSender:c===s,from:c,to:d}};async function Z(t){try{return t.then(e=>[e,null]).catch(e=>[null,e])}catch(e){return Promise.resolve([null,e])}}var $e="https://cloudflare-ipfs.com";function x(t,e=$e){if(!t)return "";try{return ipfsResolver(t,e)}catch{return t}}async function Ye(t,e=5e3){let a=new AbortController;return setTimeout(()=>a.abort(),e),fetch(t,{signal:a.signal})}async function Q(t){let e={};if(t)if(t.startsWith("data:application/json;base64,")){let a=t.substring(29);try{let n=Buffer.from(a,"base64").toString();e=JSON.parse(n);}catch{e={};}}else e=await Ye(x(t)).then(a=>a.json()).catch(()=>({}));else return {};return e}var ze="https://image-proxy.svc.prod.covalenthq.com/cdn-cgi/image";function ee(t,e="256"){let a=x(t);return `${ze}/width=${e},fit/${a}`}var ne=async({nativeTransaction:t,erc20Transfers:e,erc721Transfers:a,erc1155Transfers:n},s)=>{let o=[];if(t.value!=="0"){let r=s.decimals,i=new BN(t.value),c=balanceToDisplayValue(i,r);o.push({decimal:r.toString(),name:s.name,symbol:s.symbol,amount:c,from:t.from,to:t.to,type:TokenType.NATIVE});}return e?.forEach(r=>{let i=r.erc20Token.decimals,c=new BN(r.value),d=balanceToDisplayValue(c,i);o.push({decimal:i.toString(),name:r.erc20Token.name,symbol:r.erc20Token.symbol,amount:d,from:r.from,to:r.to,imageUri:r.erc20Token.logoUri,type:TokenType.ERC20});}),a&&await Promise.allSettled(a.map(async r=>{let i=r.erc721Token,c=await re(i.tokenUri,i.metadata.imageUri);o.push({name:r.erc721Token.name,symbol:r.erc721Token.symbol,amount:"1",imageUri:c,from:r.from,to:r.to,collectableTokenId:r.erc721Token.tokenId,type:TokenType.ERC721});})),n&&await Promise.allSettled(n.map(async r=>{let i=r.erc1155Token,c=await re(i.tokenUri,i.metadata.imageUri);o.push({name:r.erc1155Token.metadata.name??"",symbol:r.erc1155Token.metadata.symbol??"",amount:r.value,imageUri:c,from:r.from,to:r.to,collectableTokenId:r.erc1155Token.tokenId,type:TokenType.ERC1155});})),o},re=async(t,e)=>{if(e)return e.startsWith("ipfs://")?x(e):e;{let[a,n]=await Z(Q(t));return n?"":a.image?ee(a.image):""}};var oe=[TransactionType.SEND,TransactionType.RECEIVE,TransactionType.TRANSFER];var se=async({transactions:t,explorerUrl:e,networkToken:a,chainId:n,address:s})=>{let o=await ne(t,a),r=K(t,s,o),{isOutgoing:i,isIncoming:c,isSender:d,from:p,to:m}=X(r,t,s),{blockTimestamp:l,txHash:b,gasPrice:f,gasUsed:u}=t.nativeTransaction,y=N(e,b);return {isContractCall:!oe.includes(r),isIncoming:c,isOutgoing:i,isSender:d,timestamp:l*1e3,hash:b,from:p,to:m,tokens:o,gasPrice:f,gasUsed:u,chainId:n.toString(),txType:r,explorerLink:y}};var ie=async({chainId:t,explorerUrl:e,networkToken:a,address:n,nextPageToken:s,offset:o,glacierService:r})=>{try{let i=await r.listTransactions({chainId:t.toString(),address:n,pageToken:s,pageSize:o});return {transactions:(await Promise.all(i.transactions.filter(p=>p.nativeTransaction.txStatus==="1").map(p=>se({transactions:p,explorerUrl:e,networkToken:a,chainId:t,address:n}).then(m=>m)))).filter(p=>p.tokens.find(m=>Number(m.amount)!==0)),nextPageToken:i.nextPageToken}}catch{return {transactions:[],nextPageToken:""}}};var ce=async({chainId:t,isTestnet:e,networkToken:a,explorerUrl:n,address:s,nextPageToken:o,offset:r,glacierService:i})=>$(t)?j({isTestnet:e,networkToken:a,explorerUrl:n,chainId:t,address:s,nextPageToken:o,offset:r}):i.isHealthy()?ie({networkToken:a,explorerUrl:n,chainId:t,address:s,nextPageToken:o,offset:r,glacierService:i}):{transactions:[],nextPageToken:""};var le={name:"EVM",description:"",version:"0.0.1",sources:{module:{checksum:"",location:{npm:{filePath:"dist/bundle.js",packageName:"@avalabs/evm-module",registry:"https://registry.npmjs.org"}}},provider:{checksum:"",location:{npm:{filePath:"dist/provider.js",packageName:"@avalabs/evm-module",registry:"https://registry.npmjs.org"}}}},network:{chainIds:["eip155:1","eip155:43114","eip155:43113"],namespaces:["eip155"]},cointype:"60",permissions:{rpc:{dapps:!0,methods:["eth_sendTransaction","eth_*","getTransactionHistory"]}},manifestVersion:"0.0"};var Xe=z$1.object({from:z$1.string().length(42),to:z$1.string().length(42),data:z$1.string().optional(),value:z$1.string().startsWith("0x").optional(),gas:z$1.string().startsWith("0x").optional(),gasPrice:z$1.string().startsWith("0x").optional(),maxFeePerGas:z$1.string().startsWith("0x").optional(),maxPriorityFeePerGas:z$1.string().startsWith("0x").optional(),nonce:z$1.string().optional(),chainId:z$1.string().optional()}),Ze=z$1.array(Xe).length(1),pe=t=>Ze.safeParse(t);var me=async({transactionParams:{from:t,to:e,data:a,value:n},provider:s})=>{let o=await s.getTransactionCount(t);return Number(await s.estimateGas({from:t,to:e,nonce:o,data:a,value:n}))};var de=async({from:t,provider:e})=>e.getTransactionCount(t);var be=async({request:t,network:e,approvalController:a})=>{let{dappInfo:n,params:s}=t,o=pe(s);if(!o.success)return console.error(o.error),{error:rpcErrors.invalidParams("Transaction params are invalid")};let r=o.data[0];if(!r)return {error:rpcErrors.invalidParams("Transaction params are invalid")};let i=C({chainId:e.chainId,chainName:e.chainName,rpcUrl:e.rpcUrl,multiContractAddress:e.utilityAddresses?.multicall,pollingInterval:1e3});if(!r.gas||Number(r.gas)<0)try{let l=await me({transactionParams:{from:r.from,to:r.to,data:r.data,value:r.value},provider:i});r.gas="0x"+l.toString(16);}catch{return {error:rpcErrors.internal("Unable to calculate gas limit")}}if(!r.nonce)try{let l=await de({from:r.from,provider:i});r.nonce=String(l);}catch{return {error:rpcErrors.internal("Unable to calculate nonce")}}let c={title:"Approve Transaction",network:{chainId:e.chainId,name:e.chainName,logoUri:e.logoUri},transactionDetails:{website:new URL(n.url).hostname,from:r.from,to:r.to,data:r.data},networkFeeSelector:!0},d={type:SigningDataType.EVM_TRANSACTION,account:r.from,chainId:e.chainId,data:{type:2,nonce:Number(r.nonce),gasLimit:Number(r.gas),to:r.to,from:r.from,data:r.data,value:r.value,chainId:r.chainId}},p=await a.requestApproval({request:t,displayData:c,signingData:d});if("error"in p)return {error:p.error};let m=await i.send("eth_sendRawTransaction",[p.result]);return e0({provider:i,txHash:m,onTransactionConfirmed:a.onTransactionConfirmed,onTransactionReverted:a.onTransactionReverted}),{result:m}},e0=async({provider:t,txHash:e,onTransactionConfirmed:a,onTransactionReverted:n})=>{(await t.waitForTransaction(e))?.status===1?a(e):n(e);};var fe={_format:"hh-sol-artifact-1",contractName:"ERC20",sourceName:"contracts/token/ERC20/ERC20.sol",abi:[{inputs:[{internalType:"string",name:"name_",type:"string"},{internalType:"string",name:"symbol_",type:"string"}],stateMutability:"nonpayable",type:"constructor"},{anonymous:!1,inputs:[{indexed:!0,internalType:"address",name:"owner",type:"address"},{indexed:!0,internalType:"address",name:"spender",type:"address"},{indexed:!1,internalType:"uint256",name:"value",type:"uint256"}],name:"Approval",type:"event"},{anonymous:!1,inputs:[{indexed:!0,internalType:"address",name:"from",type:"address"},{indexed:!0,internalType:"address",name:"to",type:"address"},{indexed:!1,internalType:"uint256",name:"value",type:"uint256"}],name:"Transfer",type:"event"},{inputs:[{internalType:"address",name:"owner",type:"address"},{internalType:"address",name:"spender",type:"address"}],name:"allowance",outputs:[{internalType:"uint256",name:"",type:"uint256"}],stateMutability:"view",type:"function"},{inputs:[{internalType:"address",name:"spender",type:"address"},{internalType:"uint256",name:"amount",type:"uint256"}],name:"approve",outputs:[{internalType:"bool",name:"",type:"bool"}],stateMutability:"nonpayable",type:"function"},{inputs:[{internalType:"address",name:"account",type:"address"}],name:"balanceOf",outputs:[{internalType:"uint256",name:"",type:"uint256"}],stateMutability:"view",type:"function"},{inputs:[],name:"decimals",outputs:[{internalType:"uint8",name:"",type:"uint8"}],stateMutability:"view",type:"function"},{inputs:[{internalType:"address",name:"spender",type:"address"},{internalType:"uint256",name:"subtractedValue",type:"uint256"}],name:"decreaseAllowance",outputs:[{internalType:"bool",name:"",type:"bool"}],stateMutability:"nonpayable",type:"function"},{inputs:[{internalType:"address",name:"spender",type:"address"},{internalType:"uint256",name:"addedValue",type:"uint256"}],name:"increaseAllowance",outputs:[{internalType:"bool",name:"",type:"bool"}],stateMutability:"nonpayable",type:"function"},{inputs:[],name:"name",outputs:[{internalType:"string",name:"",type:"string"}],stateMutability:"view",type:"function"},{inputs:[],name:"symbol",outputs:[{internalType:"string",name:"",type:"string"}],stateMutability:"view",type:"function"},{inputs:[],name:"totalSupply",outputs:[{internalType:"uint256",name:"",type:"uint256"}],stateMutability:"view",type:"function"},{inputs:[{internalType:"address",name:"to",type:"address"},{internalType:"uint256",name:"amount",type:"uint256"}],name:"transfer",outputs:[{internalType:"bool",name:"",type:"bool"}],stateMutability:"nonpayable",type:"function"},{inputs:[{internalType:"address",name:"from",type:"address"},{internalType:"address",name:"to",type:"address"},{internalType:"uint256",name:"amount",type:"uint256"}],name:"transferFrom",outputs:[{internalType:"bool",name:"",type:"bool"}],stateMutability:"nonpayable",type:"function"}],bytecode:"0x60806040523480156200001157600080fd5b5060405162000aed38038062000aed8339810160408190526200003491620001db565b81516200004990600390602085019062000068565b5080516200005f90600490602084019062000068565b50505062000281565b828054620000769062000245565b90600052602060002090601f0160209004810192826200009a5760008555620000e5565b82601f10620000b557805160ff1916838001178555620000e5565b82800160010185558215620000e5579182015b82811115620000e5578251825591602001919060010190620000c8565b50620000f3929150620000f7565b5090565b5b80821115620000f35760008155600101620000f8565b634e487b7160e01b600052604160045260246000fd5b600082601f8301126200013657600080fd5b81516001600160401b03808211156200015357620001536200010e565b604051601f8301601f19908116603f011681019082821181831017156200017e576200017e6200010e565b816040528381526020925086838588010111156200019b57600080fd5b600091505b83821015620001bf5785820183015181830184015290820190620001a0565b83821115620001d15760008385830101525b9695505050505050565b60008060408385031215620001ef57600080fd5b82516001600160401b03808211156200020757600080fd5b620002158683870162000124565b935060208501519150808211156200022c57600080fd5b506200023b8582860162000124565b9150509250929050565b600181811c908216806200025a57607f821691505b6020821081036200027b57634e487b7160e01b600052602260045260246000fd5b50919050565b61085c80620002916000396000f3fe608060405234801561001057600080fd5b50600436106100a95760003560e01c80633950935111610071578063395093511461012357806370a082311461013657806395d89b411461015f578063a457c2d714610167578063a9059cbb1461017a578063dd62ed3e1461018d57600080fd5b806306fdde03146100ae578063095ea7b3146100cc57806318160ddd146100ef57806323b872dd14610101578063313ce56714610114575b600080fd5b6100b66101a0565b6040516100c3919061069a565b60405180910390f35b6100df6100da36600461070b565b610232565b60405190151581526020016100c3565b6002545b6040519081526020016100c3565b6100df61010f366004610735565b61024a565b604051601281526020016100c3565b6100df61013136600461070b565b61026e565b6100f3610144366004610771565b6001600160a01b031660009081526020819052604090205490565b6100b6610290565b6100df61017536600461070b565b61029f565b6100df61018836600461070b565b61031f565b6100f361019b366004610793565b61032d565b6060600380546101af906107c6565b80601f01602080910402602001604051908101604052809291908181526020018280546101db906107c6565b80156102285780601f106101fd57610100808354040283529160200191610228565b820191906000526020600020905b81548152906001019060200180831161020b57829003601f168201915b5050505050905090565b600033610240818585610358565b5060019392505050565b60003361025885828561047c565b6102638585856104f6565b506001949350505050565b600033610240818585610281838361032d565b61028b9190610800565b610358565b6060600480546101af906107c6565b600033816102ad828661032d565b9050838110156103125760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b60648201526084015b60405180910390fd5b6102638286868403610358565b6000336102408185856104f6565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b6001600160a01b0383166103ba5760405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608401610309565b6001600160a01b03821661041b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610309565b6001600160a01b0383811660008181526001602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6000610488848461032d565b905060001981146104f057818110156104e35760405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606401610309565b6104f08484848403610358565b50505050565b6001600160a01b03831661055a5760405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f206164604482015264647265737360d81b6064820152608401610309565b6001600160a01b0382166105bc5760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610309565b6001600160a01b038316600090815260208190526040902054818110156106345760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610309565b6001600160a01b03848116600081815260208181526040808320878703905593871680835291849020805487019055925185815290927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef910160405180910390a36104f0565b600060208083528351808285015260005b818110156106c7578581018301518582016040015282016106ab565b818111156106d9576000604083870101525b50601f01601f1916929092016040019392505050565b80356001600160a01b038116811461070657600080fd5b919050565b6000806040838503121561071e57600080fd5b610727836106ef565b946020939093013593505050565b60008060006060848603121561074a57600080fd5b610753846106ef565b9250610761602085016106ef565b9150604084013590509250925092565b60006020828403121561078357600080fd5b61078c826106ef565b9392505050565b600080604083850312156107a657600080fd5b6107af836106ef565b91506107bd602084016106ef565b90509250929050565b600181811c908216806107da57607f821691505b6020821081036107fa57634e487b7160e01b600052602260045260246000fd5b50919050565b6000821982111561082157634e487b7160e01b600052601160045260246000fd5b50019056fea2646970667358221220e7dc257596a973556752e9db5f465b21aad78d068ae819d0395e1f99a95065a964736f6c634300080d0033",deployedBytecode:"0x608060405234801561001057600080fd5b50600436106100a95760003560e01c80633950935111610071578063395093511461012357806370a082311461013657806395d89b411461015f578063a457c2d714610167578063a9059cbb1461017a578063dd62ed3e1461018d57600080fd5b806306fdde03146100ae578063095ea7b3146100cc57806318160ddd146100ef57806323b872dd14610101578063313ce56714610114575b600080fd5b6100b66101a0565b6040516100c3919061069a565b60405180910390f35b6100df6100da36600461070b565b610232565b60405190151581526020016100c3565b6002545b6040519081526020016100c3565b6100df61010f366004610735565b61024a565b604051601281526020016100c3565b6100df61013136600461070b565b61026e565b6100f3610144366004610771565b6001600160a01b031660009081526020819052604090205490565b6100b6610290565b6100df61017536600461070b565b61029f565b6100df61018836600461070b565b61031f565b6100f361019b366004610793565b61032d565b6060600380546101af906107c6565b80601f01602080910402602001604051908101604052809291908181526020018280546101db906107c6565b80156102285780601f106101fd57610100808354040283529160200191610228565b820191906000526020600020905b81548152906001019060200180831161020b57829003601f168201915b5050505050905090565b600033610240818585610358565b5060019392505050565b60003361025885828561047c565b6102638585856104f6565b506001949350505050565b600033610240818585610281838361032d565b61028b9190610800565b610358565b6060600480546101af906107c6565b600033816102ad828661032d565b9050838110156103125760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b60648201526084015b60405180910390fd5b6102638286868403610358565b6000336102408185856104f6565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b6001600160a01b0383166103ba5760405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608401610309565b6001600160a01b03821661041b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610309565b6001600160a01b0383811660008181526001602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6000610488848461032d565b905060001981146104f057818110156104e35760405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606401610309565b6104f08484848403610358565b50505050565b6001600160a01b03831661055a5760405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f206164604482015264647265737360d81b6064820152608401610309565b6001600160a01b0382166105bc5760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610309565b6001600160a01b038316600090815260208190526040902054818110156106345760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610309565b6001600160a01b03848116600081815260208181526040808320878703905593871680835291849020805487019055925185815290927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef910160405180910390a36104f0565b600060208083528351808285015260005b818110156106c7578581018301518582016040015282016106ab565b818111156106d9576000604083870101525b50601f01601f1916929092016040019392505050565b80356001600160a01b038116811461070657600080fd5b919050565b6000806040838503121561071e57600080fd5b610727836106ef565b946020939093013593505050565b60008060006060848603121561074a57600080fd5b610753846106ef565b9250610761602085016106ef565b9150604084013590509250925092565b60006020828403121561078357600080fd5b61078c826106ef565b9392505050565b600080604083850312156107a657600080fd5b6107af836106ef565b91506107bd602084016106ef565b90509250929050565b600181811c908216806107da57607f821691505b6020821081036107fa57634e487b7160e01b600052602260045260246000fd5b50919050565b6000821982111561082157634e487b7160e01b600052601160045260246000fd5b50019056fea2646970667358221220e7dc257596a973556752e9db5f465b21aad78d068ae819d0395e1f99a95065a964736f6c634300080d0033",linkReferences:{},deployedLinkReferences:{}};var c0=18,ue=async({provider:t,tokenService:e,address:a,currency:n,tokens:s,network:o})=>{let r=o.pricingProviders?.coingecko.assetPlatformId,i=o.pricingProviders?.coingecko.nativeTokenId,c=s.map(m=>m.address),d=await Promise.allSettled(s.map(async m=>{let b=await new ethers.Contract(m.address,fe.abi,t).balanceOf?.(a),f=new he(b)||numberToBN(0,m.decimals||c0);return {...m,balance:f}})).then(m=>m.reduce((l,b)=>b.status==="fulfilled"&&!b.value.balance.isZero()?[...l,b.value]:l,[]));if(!d.length)return {};let p=r&&await e.getPricesByAddresses(c,r,n)||{};return d.reduce((m,l)=>{let b=p?.[i??""]?.[n]?.price??0,f=p?.[i??""]?.[n]?.marketCap??0,u=p?.[i??""]?.[n]?.vol24??0,y=p?.[i??""]?.[n]?.change24??0,g=bnToBig(l.balance,l.decimals).mul(b).toNumber(),k=balanceToDisplayValue(l.balance,l.decimals),Ne=g.toFixed(2);return {...m,[l.address.toLowerCase()]:{...l,type:TokenType.ERC20,balance:l.balance,balanceDisplayValue:k,balanceInCurrency:g,balanceCurrencyDisplayValue:Ne,priceInCurrency:b,marketCap:f,change24:y,vol24:u}}},{})};var ye=async({provider:t,tokenService:e,address:a,currency:n,network:s})=>{let o=s.pricingProviders?.coingecko.nativeTokenId,r=s.networkToken,i=o?await e.getSimplePrice({coinIds:[o],currencies:[n]}):{},c=i?.[o??""]?.[n]?.price??0,d=i?.[o??""]?.[n]?.marketCap??0,p=i?.[o??""]?.[n]?.vol24??0,m=i?.[o??""]?.[n]?.change24??0,l=await t.getBalance(a),b=bigintToBig(l,r.decimals),f=bigToBN(b,r.decimals),u=balanceToDisplayValue(f,r.decimals),y=bnToBig(f,r.decimals).mul(c).toNumber(),g=y.toFixed(2);return {...r,coingeckoId:o??"",type:TokenType.NATIVE,balance:f,balanceDisplayValue:u,balanceInCurrency:y,balanceCurrencyDisplayValue:g,priceInCurrency:c,marketCap:d,vol24:p,change24:m}};var ge=async({address:t,currency:e,chainId:a,glacierService:n})=>{let o=(await n.getNativeBalance({chainId:a.toString(),address:t,currency:e.toLocaleLowerCase()})).nativeTokenBalance,r=new BN(o.balance),i=balanceToDisplayValue(r,o.decimals),c=o.price?.value??0,d=bnToBig(r,o.decimals).mul(c).toNumber(),p=d.toFixed(2);return {name:o.name,symbol:o.symbol,decimals:o.decimals,type:TokenType.NATIVE,logoUri:o.logoUri,balance:r,balanceDisplayValue:i,balanceInCurrency:d,balanceCurrencyDisplayValue:p,priceInCurrency:c,marketCap:0,vol24:0,change24:0,coingeckoId:""}};var ve=async({glacierService:t,currency:e,chainId:a,address:n,customTokens:s})=>{let o=[],r;do{let i=await t.listErc20Balances({chainId:a.toString(),address:n,currency:e.toLocaleLowerCase(),pageSize:100,pageToken:r});o.push(...k0(i.erc20TokenBalances,Number(a))),r=i.nextPageToken;}while(r);return [...v0(s),...o].reduce((i,c)=>({...i,[c.address.toLowerCase()]:c}),{})},v0=t=>t.map(e=>({...e,type:TokenType.ERC20,balance:new he(0),balanceInCurrency:0,balanceDisplayValue:"0",balanceCurrencyDisplayValue:"0",priceInCurrency:0,marketCap:0,change24:0,vol24:0})),k0=(t,e)=>t.map(a=>{let n=new he(a.balance),s=balanceToDisplayValue(n,a.decimals),o=a.balanceValue?.value.toString()??"0",r=a.price?.value??0,i=bnToBig(n,a.decimals).mul(r).toNumber();return {chainId:e,address:a.address,name:a.name,symbol:a.symbol,decimals:a.decimals,logoUri:a.logoUri,balance:n,balanceCurrencyDisplayValue:o,balanceDisplayValue:s,balanceInCurrency:i,priceInCurrency:r,contractType:"ERC-20",type:TokenType.ERC20,change24:0,marketCap:0,vol24:0}});var ke=async({addresses:t,currency:e,network:a,proxyApiUrl:n,customTokens:s=[],storage:o,glacierService:r})=>{let i=a.chainId,c=await r.isNetworkSupported(a.chainId),d=r.isHealthy(),p=[];if(d&&c)p=await Promise.allSettled(t.map(async l=>{let b=await ge({address:l,currency:e,chainId:i,glacierService:r}),f=await ve({customTokens:s,glacierService:r,currency:e,chainId:i,address:l});return {address:l,balances:{[b.symbol]:b,...f}}}));else {let l=new TokenService({storage:o,proxyApiUrl:n}),f=[...await A({chainId:Number(i),proxyApiUrl:n}),...s],u=C({chainId:i,chainName:a.chainName,rpcUrl:a.rpcUrl,multiContractAddress:a.utilityAddresses?.multicall});p=await Promise.allSettled(t.map(async y=>{let g=await ye({network:a,tokenService:l,address:y,currency:e,provider:u}),k=await ue({provider:u,network:a,tokenService:l,address:y,currency:e,tokens:f});return {address:y,balances:{[g.symbol]:g,...k}}}));}return p.reduce((l,b)=>b.status==="rejected"?l:{...l,[b.value.address]:b.value.balances},{})};var w0={glacierApiUrl:"https://glacier-api.avax.network",proxyApiUrl:"https://proxy-api.avax.network"},C0={glacierApiUrl:"https://glacier-api-dev.avax.network",proxyApiUrl:"https://proxy-api-dev.avax.network"},we=t=>{switch(t){case Environment.PRODUCTION:return w0;case Environment.DEV:return C0}};var v=class extends Error{constructor(){super(...arguments);E(this,"message","Glacier is unhealthy. Try again later.");}},L=class{constructor({glacierApiUrl:e}){E(this,"glacierSdk");E(this,"isGlacierHealthy",!0);E(this,"supportedChainIds",[]);E(this,"isHealthy",()=>this.isGlacierHealthy);this.glacierSdk=new Glacier({BASE:e}),this.getSupportedChainIds().catch(()=>{});}setGlacierToUnhealthy(){this.isGlacierHealthy=!1,setTimeout(()=>{this.isGlacierHealthy=!0;},5*60*1e3);}async isNetworkSupported(e){return (await this.getSupportedChainIds()).some(n=>n===e.toString())}async getSupportedChainIds(){if(this.supportedChainIds.length)return this.supportedChainIds;try{let e=await this.glacierSdk.evmChains.supportedChains({});return this.supportedChainIds=e.chains.map(a=>a.chainId),this.supportedChainIds}catch{return []}}async reindexNft({address:e,chainId:a,tokenId:n}){try{await this.glacierSdk.nfTs.reindexNft({address:e,chainId:a,tokenId:n});}catch(s){throw s instanceof v&&this.setGlacierToUnhealthy(),s}}async getTokenDetails({address:e,chainId:a,tokenId:n}){try{return this.glacierSdk.nfTs.getTokenDetails({address:e,chainId:a,tokenId:n})}catch(s){throw s instanceof v&&this.setGlacierToUnhealthy(),s}}async getChainBalance(e){try{return this.glacierSdk.primaryNetworkBalances.getBalancesByAddresses(e)}catch(a){throw a instanceof v&&this.setGlacierToUnhealthy(),a}}async getNativeBalance({chainId:e,address:a,currency:n}){try{return this.glacierSdk.evmBalances.getNativeBalance({chainId:e,address:a,currency:n.toLocaleLowerCase()})}catch(s){throw s instanceof v&&this.setGlacierToUnhealthy(),s}}async listErc721Balances({chainId:e,address:a,pageSize:n,pageToken:s}){try{return this.glacierSdk.evmBalances.listErc721Balances({chainId:e,address:a,pageSize:n,pageToken:s})}catch(o){throw o instanceof v&&this.setGlacierToUnhealthy(),o}}async listErc1155Balances({chainId:e,address:a,pageSize:n,pageToken:s}){try{return this.glacierSdk.evmBalances.listErc1155Balances({chainId:e,address:a,pageSize:n,pageToken:s})}catch(o){throw o instanceof v&&this.setGlacierToUnhealthy(),o}}async listErc20Balances({chainId:e,address:a,currency:n,pageSize:s,pageToken:o}){try{return this.glacierSdk.evmBalances.listErc20Balances({chainId:e,address:a,currency:n.toLocaleLowerCase(),pageSize:s,pageToken:o})}catch(r){throw r instanceof v&&this.setGlacierToUnhealthy(),r}}async listTransactions({chainId:e,address:a,pageToken:n,pageSize:s}){try{return this.glacierSdk.evmTransactions.listTransactions({chainId:e,address:a,pageToken:n,pageSize:s})}catch(o){throw o instanceof v&&this.setGlacierToUnhealthy(),o}}};var I,B,R,Ce=class{constructor({approvalController:e,environment:a}){S(this,I,void 0);S(this,B,void 0);S(this,R,void 0);let{glacierApiUrl:n,proxyApiUrl:s}=we(a);P(this,I,new L({glacierApiUrl:n})),P(this,B,s),P(this,R,e);}getAddress(){return Promise.resolve("EVM address")}getBalances({addresses:e,network:a,currency:n,customTokens:s}){return ke({addresses:e,currency:n,network:a,proxyApiUrl:w(this,B),customTokens:s,glacierService:w(this,I)})}getManifest(){let e=parseManifest(le);return e.success?e.data:void 0}getNetworkFee(e){let{chainId:a,chainName:n,rpcUrl:s,utilityAddresses:o}=e;return _({chainId:a,chainName:n,rpcUrl:s,multiContractAddress:o?.multicall})}getTransactionHistory(e){let{network:a,address:n,nextPageToken:s,offset:o}=e,{chainId:r,isTestnet:i,networkToken:c,explorerUrl:d=""}=a;return ce({chainId:r,isTestnet:i,networkToken:c,explorerUrl:d,address:n,nextPageToken:s,offset:o,glacierService:w(this,I)})}getTokens(e){let{chainId:a}=e;return A({chainId:a,proxyApiUrl:w(this,B)})}async onRpcRequest(e,a){switch(e.method){case RpcMethod.ETH_SEND_TRANSACTION:return be({request:e,network:a,approvalController:w(this,R)});default:return {error:rpcErrors.methodNotSupported(`Method ${e.method} not supported`)}}}};I=new WeakMap,B=new WeakMap,R=new WeakMap;
13
15
 
14
- export { gt as EvmModule, rt as NonContractCallTypes };
16
+ export { Ce as EvmModule, oe as NonContractCallTypes };
15
17
  //# sourceMappingURL=out.js.map
16
18
  //# sourceMappingURL=index.js.map
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/module.ts","../src/handlers/get-tokens/get-tokens.ts","../src/utils/get-provider.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-transactions-from-glacier.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/get-transaction-history.ts","../manifest.json","../src/env.ts","../src/handlers/eth-send-transaction/eth-send-transaction.ts","../src/handlers/eth-send-transaction/schema.ts","../src/utils/estimate-gas-limit.ts","../src/utils/get-nonce.ts"],"names":["rpcErrors","RpcMethod","parseManifest","getTokens","chainId","proxyApiUrl","response","JsonRpcBatchInternal","EVMNetwork","GLACIER_API_KEY","getProvider","params","chainName","rpcUrl","multiContractAddress","pollingInterval","provider","addGlacierAPIKeyIfNeeded","knownHosts","url","glacierApiKey","urlObj","search_params","DEFAULT_PRESETS","BASE_PRIORITY_FEE_WEI","getNetworkFee","maxFeePerGasInWei","lowMaxTip","mediumMaxTip","highMaxTip","TokenType","TransactionType","balanceToDisplayValue","BN","getExplorerAddressByNetwork","explorerUrl","hash","hashType","convertTransactionNormal","tx","networkToken","address","isSender","timestamp","decimals","amountBN","amountDisplayValue","txType","from","to","gasPrice","gasUsed","explorerLink","isContractCall","convertTransactionERC20","tokenDecimal","tokenName","tokenSymbol","getErc20Txs","getNormalTxs","getTransactionFromEtherscan","isTestnet","nextPageToken","offset","parsedPageToken","page","queries","normalHist","erc20Hist","erc20TxHashes","filteredNormalTxs","next","isEthereumChainId","Glacier","startCase","getTxType","nativeTransaction","erc20Transfers","erc721Transfers","userAddress","tokens","nativeOnly","method","parseRawMethod","isSwap","isNativeSend","isNativeReceive","isNFTPurchase","isApprove","isTransfer","isAirdrop","isUnwrap","isFillOrder","isNFTSend","isNFT","isNFTReceive","tokenType","getSenderInfo","isOutgoing","isIncoming","resolve","promise","res","err","ipfsResolver","CLOUDFLARE_IPFS_URL","ipfsResolverWithFallback","sourceUrl","desiredGatewayPrefix","fetchWithTimeout","uri","timeout","controller","getNftMetadata","tokenUri","data","value","json","r","COVALENT_IMG_SIZER","getSmallImageForNFT","imgUrl","imageSize","erc1155Transfers","result","decimal","erc20Transfer","erc721Transfer","token","imageUri","getImageUri","erc1155Transfer","metadata","error","NonContractCallTypes","convertTransaction","transactions","blockTimestamp","getTransactionsFromGlacier","glacierApiUrl","glacierSdk","tranasaction","transaction","getTransactionHistory","manifest_default","Environment","prodEnv","devEnv","getEnv","environment","SigningDataType","z","transactionSchema","paramsSchema","parseRequestParams","estimateGasLimit","nonce","getNonce","ethSendTransaction","request","network","approvalController","dappInfo","gasLimit","displayData","signingData","txHash","waitForTransactionReceipt","onTransactionConfirmed","onTransactionReverted","_glacierApiUrl","_proxyApiUrl","_approvalController","EvmModule","__privateAdd","__privateSet","utilityAddresses","__privateGet"],"mappings":"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","sourcesContent":["import type {\n Module,\n Manifest,\n NetworkFees,\n GetTransactionHistory,\n RpcRequest,\n Environment,\n Network,\n ApprovalController,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { RpcMethod, parseManifest } from '@avalabs/vm-module-types';\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 { getEnv } from './env';\nimport { ethSendTransaction } from './handlers/eth-send-transaction/eth-send-transaction';\n\nexport class EvmModule implements Module {\n #glacierApiUrl: string;\n #proxyApiUrl: string;\n #approvalController: ApprovalController;\n\n constructor({\n approvalController,\n environment,\n }: {\n approvalController: ApprovalController;\n environment: Environment;\n }) {\n const { glacierApiUrl, proxyApiUrl } = getEnv(environment);\n this.#glacierApiUrl = glacierApiUrl;\n this.#proxyApiUrl = proxyApiUrl;\n this.#approvalController = approvalController;\n }\n\n getAddress(): Promise<string> {\n return Promise.resolve('EVM address');\n }\n\n getBalances(): Promise<string> {\n return Promise.resolve('EVM balances');\n }\n\n getManifest(): Manifest | undefined {\n const result = parseManifest(ManifestJson);\n return result.success ? result.data : undefined;\n }\n\n getNetworkFee(network: Network): Promise<NetworkFees> {\n const { chainId, chainName, rpcUrl, utilityAddresses } = network;\n return getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress: utilityAddresses?.multicall,\n glacierApiUrl: this.#glacierApiUrl,\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 glacierApiUrl: this.#glacierApiUrl,\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 });\n default:\n return { error: rpcErrors.methodNotSupported(`Method ${request.method} not supported`) };\n }\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/wallets-sdk';\nimport { Network as EVMNetwork } from 'ethers';\n\n// this key is only needed in development to bypass rate limit\n// it should never be used in production\nconst GLACIER_API_KEY = process.env.GLACIER_API_KEY;\n\ntype ProviderParams = {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n multiContractAddress?: string;\n pollingInterval?: number;\n};\n\nexport const getProvider = (params: ProviderParams): JsonRpcBatchInternal => {\n const { chainId, chainName, rpcUrl, multiContractAddress, pollingInterval = 2000 } = params;\n\n const provider = new JsonRpcBatchInternal(\n { maxCalls: 40, multiContractAddress },\n addGlacierAPIKeyIfNeeded(rpcUrl, GLACIER_API_KEY),\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, glacierApiKey?: string): string {\n const urlObj = new URL(url);\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 type { NetworkFees } from '@avalabs/vm-module-types';\nimport { getProvider } from '../../utils/get-provider';\nimport { rpcErrors } from '@metamask/rpc-errors';\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\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}: {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n glacierApiUrl: string;\n multiContractAddress?: string;\n}): Promise<NetworkFees> {\n const provider = getProvider({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n });\n\n const { maxFeePerGas: maxFeePerGasInWei } = await provider.getFeeData();\n if (!maxFeePerGasInWei) {\n throw rpcErrors.internal('Pre-EIP-1559 networks are not supported');\n }\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: maxFeePerGasInWei,\n low: {\n maxFeePerGas: maxFeePerGasInWei + lowMaxTip,\n maxPriorityFeePerGas: lowMaxTip,\n },\n medium: {\n maxFeePerGas: maxFeePerGasInWei + mediumMaxTip,\n maxPriorityFeePerGas: mediumMaxTip,\n },\n high: {\n maxFeePerGas: maxFeePerGasInWei + highMaxTip,\n maxPriorityFeePerGas: highMaxTip,\n },\n isFixedFee: false,\n };\n}\n","import type { NormalTx } from '@avalabs/etherscan-sdk';\nimport { TokenType, TransactionType, type NetworkToken, type Transaction } from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { BN } from 'bn.js';\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 amountBN = new BN(tx.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\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/etherscan-sdk';\nimport { TokenType, TransactionType, type Transaction } from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\nimport { BN } from 'bn.js';\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 decimals = parseInt(tx.tokenDecimal);\n const amountBN = new BN(tx.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\n const { from, to, gasPrice, gasUsed, hash, tokenDecimal, tokenName, tokenSymbol } = 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 },\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/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 { NetworkToken, TransactionHistoryResponse } from '@avalabs/vm-module-types';\nimport { Glacier } from '@avalabs/glacier-sdk';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { convertTransaction } from './convert-transaction';\n\nexport const getTransactionsFromGlacier = async ({\n chainId,\n explorerUrl,\n networkToken,\n address,\n nextPageToken,\n offset,\n glacierApiUrl,\n}: {\n chainId: number;\n explorerUrl: string;\n networkToken: NetworkToken;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierApiUrl: string;\n}): Promise<TransactionHistoryResponse> => {\n if (!glacierApiUrl) {\n throw rpcErrors.invalidParams('Glacier API URL is required');\n }\n\n const glacierSdk = new Glacier({ BASE: glacierApiUrl });\n\n try {\n const response = await glacierSdk.evmTransactions.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 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 }: TransactionDetails,\n userAddress: string,\n tokens: TxToken[],\n): TransactionType => {\n const nativeOnly = !erc20Transfers && !erc721Transfers;\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 =\n isTransfer && !!tokens[0] && isNFT(tokens[0].type) && tokens[0].from?.address.toLowerCase() === address;\n const isNFTReceive =\n isTransfer && !!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 { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { BN } from 'bn.js';\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 amountBN = new BN(nativeTransaction.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimal);\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 amountBN = new BN(erc20Transfer.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\n\n result.push({\n decimal: decimals.toString(),\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 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 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/utils-sdk';\n\nexport const CLOUDFLARE_IPFS_URL = 'https://cloudflare-ipfs.com';\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 { TransactionType } from '@avalabs/vm-module-types';\n\nexport const NonContractCallTypes = [TransactionType.SEND, TransactionType.RECEIVE, TransactionType.TRANSFER];\n\nexport type TransactionParams = {\n from: string;\n to: string;\n data?: string;\n value?: string;\n gas?: string;\n gasPrice?: string;\n maxFeePerGas?: string;\n maxPriorityFeePerGas?: string;\n nonce?: string;\n chainId?: string;\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 { 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';\n\nexport const getTransactionHistory = async ({\n chainId,\n isTestnet,\n networkToken,\n explorerUrl,\n address,\n nextPageToken,\n offset,\n glacierApiUrl,\n}: {\n chainId: number;\n isTestnet?: boolean;\n networkToken: NetworkToken;\n explorerUrl: string;\n address: string;\n nextPageToken?: string;\n offset?: number;\n glacierApiUrl: string;\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 return getTransactionsFromGlacier({\n networkToken,\n explorerUrl,\n chainId,\n address,\n nextPageToken,\n offset,\n glacierApiUrl,\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/bundle.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.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n }\n },\n \"network\": {\n \"chainIds\": [\"eip155:1\", \"eip155:43114\", \"eip155:43113\"],\n \"namespaces\": [\"eip155\"]\n },\n \"cointype\": \"60\",\n \"permissions\": {\n \"rpc\": {\n \"dapps\": true,\n \"methods\": [\"eth_sendTransaction\", \"eth_*\", \"getTransactionHistory\"]\n }\n },\n \"manifestVersion\": \"0.0\"\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 type Network,\n type Hex,\n type RpcRequest,\n type ApprovalController,\n type DisplayData,\n type SigningData,\n SigningDataType,\n} from '@avalabs/vm-module-types';\nimport { parseRequestParams } from './schema';\nimport { estimateGasLimit } from '../../utils/estimate-gas-limit';\nimport { getNonce } from '../../utils/get-nonce';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getProvider } from '../../utils/get-provider';\nimport type { JsonRpcBatchInternal } from '@avalabs/wallets-sdk';\n\nexport const ethSendTransaction = async ({\n request,\n network,\n approvalController,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n}) => {\n const { dappInfo, params } = request;\n\n // validate params\n const result = parseRequestParams(params);\n\n if (!result.success) {\n console.error(result.error);\n return {\n error: rpcErrors.invalidParams('Transaction params are invalid'),\n };\n }\n\n const transaction = result.data[0];\n\n if (!transaction) {\n return {\n error: rpcErrors.invalidParams('Transaction params are invalid'),\n };\n }\n\n const provider = 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 },\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 // TODO: validate + simulate transaction\n // https://ava-labs.atlassian.net/browse/CP-8870\n\n // generate display and signing data\n // TODO adjust title for different transaction types\n // https://ava-labs.atlassian.net/browse/CP-8870\n const displayData: DisplayData = {\n title: 'Approve Transaction',\n network: {\n chainId: network.chainId,\n name: network.chainName,\n logoUrl: network.logoUrl,\n },\n transactionDetails: {\n website: new URL(dappInfo.url).hostname,\n from: transaction.from,\n to: transaction.to,\n data: transaction.data,\n },\n networkFeeSelector: true,\n };\n\n const signingData: SigningData = {\n type: SigningDataType.EVM_TRANSACTION,\n account: transaction.from,\n chainId: network.chainId,\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 to: transaction.to,\n from: transaction.from,\n data: transaction.data,\n value: transaction.value,\n chainId: transaction.chainId,\n },\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 // broadcast the signed transaction\n const txHash = await provider.send('eth_sendRawTransaction', [response.result]);\n\n waitForTransactionReceipt({\n provider,\n txHash,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n });\n\n return { result: txHash };\n};\n\nconst waitForTransactionReceipt = async ({\n provider,\n txHash,\n onTransactionConfirmed,\n onTransactionReverted,\n}: {\n provider: JsonRpcBatchInternal;\n txHash: Hex;\n onTransactionConfirmed: (txHash: Hex) => void;\n onTransactionReverted: (txHash: Hex) => void;\n}) => {\n const receipt = await provider.waitForTransaction(txHash);\n\n const success = receipt?.status === 1; // 1 indicates success, 0 indicates revert\n\n if (success) {\n onTransactionConfirmed(txHash);\n } else {\n onTransactionReverted(txHash);\n }\n};\n","import { z } from 'zod';\n\nconst transactionSchema = z.object({\n from: z.string().length(42),\n to: z.string().length(42),\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(),\n});\n\nconst paramsSchema = z.array(transactionSchema).length(1);\n\nexport const parseRequestParams = (params: unknown) => {\n return paramsSchema.safeParse(params);\n};\n","import { JsonRpcBatchInternal } from '@avalabs/wallets-sdk';\nimport type { BigNumberish } from 'ethers';\n\nexport const estimateGasLimit = async ({\n transactionParams: { from, to, data, value },\n provider,\n}: {\n transactionParams: {\n from: string;\n to: string;\n data?: string;\n value?: BigNumberish;\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 }),\n );\n};\n","import { JsonRpcBatchInternal } from '@avalabs/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"]}
1
+ {"version":3,"sources":["../src/module.ts","../src/handlers/get-tokens/get-tokens.ts","../src/utils/get-provider.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/handlers/eth-send-transaction/schema.ts","../src/utils/estimate-gas-limit.ts","../src/utils/get-nonce.ts","../src/handlers/get-balances/evm-balance-service/get-erc20-balances.ts","../../../node_modules/.pnpm/@openzeppelin+contracts@4.9.3/node_modules/@openzeppelin/contracts/build/contracts/ERC20.json","../src/handlers/get-balances/get-balances.ts","../src/handlers/get-balances/evm-balance-service/get-native-token-balances.ts","../src/handlers/get-balances/glacier-balance-service/get-native-token-balances.ts","../src/handlers/get-balances/glacier-balance-service/get-erc20-balances.ts","../src/env.ts","../src/services/glacier-service/glacier-service.ts"],"names":["rpcErrors","RpcMethod","parseManifest","getTokens","chainId","proxyApiUrl","response","JsonRpcBatchInternal","EVMNetwork","GLACIER_API_KEY","getProvider","params","chainName","rpcUrl","multiContractAddress","pollingInterval","provider","addGlacierAPIKeyIfNeeded","knownHosts","url","glacierApiKey","urlObj","search_params","DEFAULT_PRESETS","BASE_PRIORITY_FEE_WEI","getNetworkFee","maxFeePerGasInWei","lowMaxTip","mediumMaxTip","highMaxTip","TokenType","TransactionType","balanceToDisplayValue","BN","getExplorerAddressByNetwork","explorerUrl","hash","hashType","convertTransactionNormal","tx","networkToken","address","isSender","timestamp","decimals","amountBN","amountDisplayValue","txType","from","to","gasPrice","gasUsed","explorerLink","isContractCall","convertTransactionERC20","tokenDecimal","tokenName","tokenSymbol","getErc20Txs","getNormalTxs","getTransactionFromEtherscan","isTestnet","nextPageToken","offset","parsedPageToken","page","queries","normalHist","erc20Hist","erc20TxHashes","filteredNormalTxs","next","isEthereumChainId","startCase","getTxType","nativeTransaction","erc20Transfers","erc721Transfers","userAddress","tokens","nativeOnly","method","parseRawMethod","isSwap","isNativeSend","isNativeReceive","isNFTPurchase","isApprove","isTransfer","isAirdrop","isUnwrap","isFillOrder","isNFTSend","isNFT","isNFTReceive","tokenType","getSenderInfo","isOutgoing","isIncoming","resolve","promise","res","err","ipfsResolver","CLOUDFLARE_IPFS_URL","ipfsResolverWithFallback","sourceUrl","desiredGatewayPrefix","fetchWithTimeout","uri","timeout","controller","getNftMetadata","tokenUri","data","value","json","r","COVALENT_IMG_SIZER","getSmallImageForNFT","imgUrl","imageSize","erc1155Transfers","result","decimal","erc20Transfer","erc721Transfer","token","imageUri","getImageUri","erc1155Transfer","metadata","error","NonContractCallTypes","convertTransaction","transactions","blockTimestamp","getTransactionsFromGlacier","glacierService","tranasaction","transaction","getTransactionHistory","manifest_default","SigningDataType","z","transactionSchema","paramsSchema","parseRequestParams","estimateGasLimit","nonce","getNonce","ethSendTransaction","request","network","approvalController","dappInfo","gasLimit","displayData","signingData","txHash","waitForTransactionReceipt","onTransactionConfirmed","onTransactionReverted","numberToBN","bnToBig","ethers","ERC20_default","DEFAULT_DECIMALS","getErc20Balances","tokenService","currency","coingeckoPlatformId","coingeckoTokenId","tokenAddresses","tokensBalances","balanceBig","balance","acc","simplePriceResponse","priceInCurrency","marketCap","vol24","change24","balanceInCurrency","balanceDisplayValue","balanceCurrencyDisplayValue","TokenService","bigToBN","bigintToBig","getNativeTokenBalances","balanceBigint","balaceBig","nativeTokenBalance","customTokens","tokensWithBalance","convertErc20ToTokenWithBalance","convertNetworkTokenToTokenWithBalance","tokenBalances","getBalances","addresses","storage","isNetworkSupported","isHealthy","balances","nativeToken","erc20Tokens","allTokens","Environment","prodEnv","devEnv","getEnv","environment","Glacier","GlacierUnhealthyError","__publicField","EvmGlacierService","glacierApiUrl","id","supportedChains","chain","tokenId","pageSize","pageToken","_glacierService","_proxyApiUrl","_approvalController","EvmModule","__privateAdd","__privateSet","__privateGet","utilityAddresses"],"mappings":"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","sourcesContent":["import type {\n Module,\n Manifest,\n NetworkFees,\n GetTransactionHistory,\n RpcRequest,\n Environment,\n Network,\n ApprovalController,\n GetBalancesParams,\n GetBalancesResponse,\n} from '@avalabs/vm-module-types';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { RpcMethod, parseManifest } from '@avalabs/vm-module-types';\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';\n\nexport class EvmModule implements Module {\n #glacierService: EvmGlacierService;\n #proxyApiUrl: string;\n #approvalController: ApprovalController;\n\n constructor({\n approvalController,\n environment,\n }: {\n approvalController: ApprovalController;\n environment: Environment;\n }) {\n const { glacierApiUrl, proxyApiUrl } = getEnv(environment);\n this.#glacierService = new EvmGlacierService({ glacierApiUrl });\n this.#proxyApiUrl = proxyApiUrl;\n this.#approvalController = approvalController;\n }\n\n getAddress(): Promise<string> {\n return Promise.resolve('EVM address');\n }\n\n getBalances({ addresses, network, currency, customTokens }: GetBalancesParams): Promise<GetBalancesResponse> {\n return getBalances({\n addresses,\n currency,\n network,\n proxyApiUrl: this.#proxyApiUrl,\n customTokens,\n glacierService: this.#glacierService,\n });\n }\n\n getManifest(): Manifest | undefined {\n const result = parseManifest(ManifestJson);\n return result.success ? result.data : undefined;\n }\n\n getNetworkFee(network: Network): Promise<NetworkFees> {\n const { chainId, chainName, rpcUrl, utilityAddresses } = network;\n return getNetworkFee({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress: utilityAddresses?.multicall,\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 });\n default:\n return { error: rpcErrors.methodNotSupported(`Method ${request.method} not supported`) };\n }\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/wallets-sdk';\nimport { Network as EVMNetwork } from 'ethers';\n\n// this key is only needed in development to bypass rate limit\n// it should never be used in production\nconst GLACIER_API_KEY = process.env.GLACIER_API_KEY;\n\ntype ProviderParams = {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n multiContractAddress?: string;\n pollingInterval?: number;\n};\n\nexport const getProvider = (params: ProviderParams): JsonRpcBatchInternal => {\n const { chainId, chainName, rpcUrl, multiContractAddress, pollingInterval = 2000 } = params;\n\n const provider = new JsonRpcBatchInternal(\n { maxCalls: 40, multiContractAddress },\n addGlacierAPIKeyIfNeeded(rpcUrl, GLACIER_API_KEY),\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, glacierApiKey?: string): string {\n const urlObj = new URL(url);\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 type { NetworkFees } from '@avalabs/vm-module-types';\nimport { getProvider } from '../../utils/get-provider';\nimport { rpcErrors } from '@metamask/rpc-errors';\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\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}: {\n chainId: number;\n chainName: string;\n rpcUrl: string;\n multiContractAddress?: string;\n}): Promise<NetworkFees> {\n const provider = getProvider({\n chainId,\n chainName,\n rpcUrl,\n multiContractAddress,\n });\n\n const { maxFeePerGas: maxFeePerGasInWei } = await provider.getFeeData();\n if (!maxFeePerGasInWei) {\n throw rpcErrors.internal('Pre-EIP-1559 networks are not supported');\n }\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: maxFeePerGasInWei,\n low: {\n maxFeePerGas: maxFeePerGasInWei + lowMaxTip,\n maxPriorityFeePerGas: lowMaxTip,\n },\n medium: {\n maxFeePerGas: maxFeePerGasInWei + mediumMaxTip,\n maxPriorityFeePerGas: mediumMaxTip,\n },\n high: {\n maxFeePerGas: maxFeePerGasInWei + highMaxTip,\n maxPriorityFeePerGas: highMaxTip,\n },\n isFixedFee: false,\n };\n}\n","import type { NormalTx } from '@avalabs/etherscan-sdk';\nimport { TokenType, TransactionType, type NetworkToken, type Transaction } from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { BN } from 'bn.js';\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 amountBN = new BN(tx.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\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/etherscan-sdk';\nimport { TokenType, TransactionType, type Transaction } from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { getExplorerAddressByNetwork } from '../../utils/get-explorer-address-by-network';\nimport { BN } from 'bn.js';\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 decimals = parseInt(tx.tokenDecimal);\n const amountBN = new BN(tx.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\n const { from, to, gasPrice, gasUsed, hash, tokenDecimal, tokenName, tokenSymbol } = 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 },\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/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 }: TransactionDetails,\n userAddress: string,\n tokens: TxToken[],\n): TransactionType => {\n const nativeOnly = !erc20Transfers && !erc721Transfers;\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 =\n isTransfer && !!tokens[0] && isNFT(tokens[0].type) && tokens[0].from?.address.toLowerCase() === address;\n const isNFTReceive =\n isTransfer && !!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 { balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { BN } from 'bn.js';\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 amountBN = new BN(nativeTransaction.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimal);\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 amountBN = new BN(erc20Transfer.value);\n const amountDisplayValue = balanceToDisplayValue(amountBN, decimals);\n\n result.push({\n decimal: decimals.toString(),\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 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 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/utils-sdk';\n\nexport const CLOUDFLARE_IPFS_URL = 'https://cloudflare-ipfs.com';\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 { TransactionType } from '@avalabs/vm-module-types';\n\nexport const NonContractCallTypes = [TransactionType.SEND, TransactionType.RECEIVE, TransactionType.TRANSFER];\n\nexport type TransactionParams = {\n from: string;\n to: string;\n data?: string;\n value?: string;\n gas?: string;\n gasPrice?: string;\n maxFeePerGas?: string;\n maxPriorityFeePerGas?: string;\n nonce?: string;\n chainId?: string;\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/bundle.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.js\",\n \"packageName\": \"@avalabs/evm-module\",\n \"registry\": \"https://registry.npmjs.org\"\n }\n }\n }\n },\n \"network\": {\n \"chainIds\": [\"eip155:1\", \"eip155:43114\", \"eip155:43113\"],\n \"namespaces\": [\"eip155\"]\n },\n \"cointype\": \"60\",\n \"permissions\": {\n \"rpc\": {\n \"dapps\": true,\n \"methods\": [\"eth_sendTransaction\", \"eth_*\", \"getTransactionHistory\"]\n }\n },\n \"manifestVersion\": \"0.0\"\n}\n","import {\n type Network,\n type Hex,\n type RpcRequest,\n type ApprovalController,\n type DisplayData,\n type SigningData,\n SigningDataType,\n} from '@avalabs/vm-module-types';\nimport { parseRequestParams } from './schema';\nimport { estimateGasLimit } from '../../utils/estimate-gas-limit';\nimport { getNonce } from '../../utils/get-nonce';\nimport { rpcErrors } from '@metamask/rpc-errors';\nimport { getProvider } from '../../utils/get-provider';\nimport type { JsonRpcBatchInternal } from '@avalabs/wallets-sdk';\n\nexport const ethSendTransaction = async ({\n request,\n network,\n approvalController,\n}: {\n request: RpcRequest;\n network: Network;\n approvalController: ApprovalController;\n}) => {\n const { dappInfo, params } = request;\n\n // validate params\n const result = parseRequestParams(params);\n\n if (!result.success) {\n console.error(result.error);\n return {\n error: rpcErrors.invalidParams('Transaction params are invalid'),\n };\n }\n\n const transaction = result.data[0];\n\n if (!transaction) {\n return {\n error: rpcErrors.invalidParams('Transaction params are invalid'),\n };\n }\n\n const provider = 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 },\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 // TODO: validate + simulate transaction\n // https://ava-labs.atlassian.net/browse/CP-8870\n\n // generate display and signing data\n // TODO adjust title for different transaction types\n // https://ava-labs.atlassian.net/browse/CP-8870\n const displayData: DisplayData = {\n title: 'Approve Transaction',\n network: {\n chainId: network.chainId,\n name: network.chainName,\n logoUri: network.logoUri,\n },\n transactionDetails: {\n website: new URL(dappInfo.url).hostname,\n from: transaction.from,\n to: transaction.to,\n data: transaction.data,\n },\n networkFeeSelector: true,\n };\n\n const signingData: SigningData = {\n type: SigningDataType.EVM_TRANSACTION,\n account: transaction.from,\n chainId: network.chainId,\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 to: transaction.to,\n from: transaction.from,\n data: transaction.data,\n value: transaction.value,\n chainId: transaction.chainId,\n },\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 // broadcast the signed transaction\n const txHash = await provider.send('eth_sendRawTransaction', [response.result]);\n\n waitForTransactionReceipt({\n provider,\n txHash,\n onTransactionConfirmed: approvalController.onTransactionConfirmed,\n onTransactionReverted: approvalController.onTransactionReverted,\n });\n\n return { result: txHash };\n};\n\nconst waitForTransactionReceipt = async ({\n provider,\n txHash,\n onTransactionConfirmed,\n onTransactionReverted,\n}: {\n provider: JsonRpcBatchInternal;\n txHash: Hex;\n onTransactionConfirmed: (txHash: Hex) => void;\n onTransactionReverted: (txHash: Hex) => void;\n}) => {\n const receipt = await provider.waitForTransaction(txHash);\n\n const success = receipt?.status === 1; // 1 indicates success, 0 indicates revert\n\n if (success) {\n onTransactionConfirmed(txHash);\n } else {\n onTransactionReverted(txHash);\n }\n};\n","import { z } from 'zod';\n\nconst transactionSchema = z.object({\n from: z.string().length(42),\n to: z.string().length(42),\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(),\n});\n\nconst paramsSchema = z.array(transactionSchema).length(1);\n\nexport const parseRequestParams = (params: unknown) => {\n return paramsSchema.safeParse(params);\n};\n","import { JsonRpcBatchInternal } from '@avalabs/wallets-sdk';\nimport type { BigNumberish } from 'ethers';\n\nexport const estimateGasLimit = async ({\n transactionParams: { from, to, data, value },\n provider,\n}: {\n transactionParams: {\n from: string;\n to: string;\n data?: string;\n value?: BigNumberish;\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 }),\n );\n};\n","import { JsonRpcBatchInternal } from '@avalabs/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 { numberToBN, bnToBig, balanceToDisplayValue } from '@avalabs/utils-sdk';\nimport { TokenType, type Network, type NetworkContractToken, type TokenWithBalance } from '@avalabs/vm-module-types';\nimport { ethers, type Provider } from 'ethers';\nimport ERC20 from '@openzeppelin/contracts/build/contracts/ERC20.json';\nimport type { TokenService } from '@internal/utils';\nimport { VsCurrencyType } from '@avalabs/coingecko-sdk';\nimport BN from 'bn.js';\n\nconst DEFAULT_DECIMALS = 18;\n\nexport const getErc20Balances = async ({\n provider,\n tokenService,\n address: userAddress,\n currency,\n tokens,\n network,\n}: {\n provider: Provider;\n tokenService: TokenService;\n address: string;\n currency: string;\n tokens: NetworkContractToken[];\n network: Network;\n}): Promise<Record<string, TokenWithBalance>> => {\n const coingeckoPlatformId = network.pricingProviders?.coingecko.assetPlatformId;\n const coingeckoTokenId = network.pricingProviders?.coingecko.nativeTokenId;\n const tokenAddresses = tokens.map((token) => token.address);\n\n const tokensBalances = await Promise.allSettled(\n tokens.map(async (token) => {\n const contract = new ethers.Contract(token.address, ERC20.abi, provider);\n const balanceBig = await contract.balanceOf?.(userAddress);\n const balance = new BN(balanceBig) || numberToBN(0, token.decimals || DEFAULT_DECIMALS);\n\n const tokenWithBalance = {\n ...token,\n balance,\n };\n\n return tokenWithBalance;\n }),\n ).then((res) => {\n return res.reduce<(NetworkContractToken & { balance: BN })[]>((acc, result) => {\n return result.status === 'fulfilled' && !result.value.balance.isZero() ? [...acc, result.value] : acc;\n }, []);\n });\n\n if (!tokensBalances.length) return {};\n\n const simplePriceResponse =\n (coingeckoPlatformId &&\n (await tokenService.getPricesByAddresses(tokenAddresses, coingeckoPlatformId, currency as VsCurrencyType))) ||\n {};\n\n return tokensBalances.reduce(\n (acc, token) => {\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? 0;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? 0;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? 0;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? 0;\n\n const balanceInCurrency = bnToBig(token.balance, token.decimals).mul(priceInCurrency).toNumber();\n const balanceDisplayValue = balanceToDisplayValue(token.balance, token.decimals);\n const balanceCurrencyDisplayValue = balanceInCurrency.toFixed(2);\n\n return {\n ...acc,\n [token.address.toLowerCase()]: {\n ...token,\n type: TokenType.ERC20,\n balance: token.balance,\n balanceDisplayValue,\n balanceInCurrency,\n balanceCurrencyDisplayValue,\n priceInCurrency,\n marketCap,\n change24,\n vol24,\n },\n };\n },\n {} as Record<string, TokenWithBalance>,\n );\n};\n","{\n \"_format\": \"hh-sol-artifact-1\",\n \"contractName\": \"ERC20\",\n \"sourceName\": \"contracts/token/ERC20/ERC20.sol\",\n \"abi\": [\n {\n \"inputs\": [\n {\n \"internalType\": \"string\",\n \"name\": \"name_\",\n \"type\": \"string\"\n },\n {\n \"internalType\": \"string\",\n \"name\": \"symbol_\",\n \"type\": \"string\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"constructor\"\n },\n {\n \"anonymous\": false,\n \"inputs\": [\n {\n \"indexed\": true,\n \"internalType\": \"address\",\n \"name\": \"owner\",\n \"type\": \"address\"\n },\n {\n \"indexed\": true,\n \"internalType\": \"address\",\n \"name\": \"spender\",\n \"type\": \"address\"\n },\n {\n \"indexed\": false,\n \"internalType\": \"uint256\",\n \"name\": \"value\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"Approval\",\n \"type\": \"event\"\n },\n {\n \"anonymous\": false,\n \"inputs\": [\n {\n \"indexed\": true,\n \"internalType\": \"address\",\n \"name\": \"from\",\n \"type\": \"address\"\n },\n {\n \"indexed\": true,\n \"internalType\": \"address\",\n \"name\": \"to\",\n \"type\": \"address\"\n },\n {\n \"indexed\": false,\n \"internalType\": \"uint256\",\n \"name\": \"value\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"Transfer\",\n \"type\": \"event\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"owner\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"address\",\n \"name\": \"spender\",\n \"type\": \"address\"\n }\n ],\n \"name\": \"allowance\",\n \"outputs\": [\n {\n \"internalType\": \"uint256\",\n \"name\": \"\",\n \"type\": \"uint256\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"spender\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"uint256\",\n \"name\": \"amount\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"approve\",\n \"outputs\": [\n {\n \"internalType\": \"bool\",\n \"name\": \"\",\n \"type\": \"bool\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"account\",\n \"type\": \"address\"\n }\n ],\n \"name\": \"balanceOf\",\n \"outputs\": [\n {\n \"internalType\": \"uint256\",\n \"name\": \"\",\n \"type\": \"uint256\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [],\n \"name\": \"decimals\",\n \"outputs\": [\n {\n \"internalType\": \"uint8\",\n \"name\": \"\",\n \"type\": \"uint8\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"spender\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"uint256\",\n \"name\": \"subtractedValue\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"decreaseAllowance\",\n \"outputs\": [\n {\n \"internalType\": \"bool\",\n \"name\": \"\",\n \"type\": \"bool\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"spender\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"uint256\",\n \"name\": \"addedValue\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"increaseAllowance\",\n \"outputs\": [\n {\n \"internalType\": \"bool\",\n \"name\": \"\",\n \"type\": \"bool\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [],\n \"name\": \"name\",\n \"outputs\": [\n {\n \"internalType\": \"string\",\n \"name\": \"\",\n \"type\": \"string\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [],\n \"name\": \"symbol\",\n \"outputs\": [\n {\n \"internalType\": \"string\",\n \"name\": \"\",\n \"type\": \"string\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [],\n \"name\": \"totalSupply\",\n \"outputs\": [\n {\n \"internalType\": \"uint256\",\n \"name\": \"\",\n \"type\": \"uint256\"\n }\n ],\n \"stateMutability\": \"view\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"to\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"uint256\",\n \"name\": \"amount\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"transfer\",\n \"outputs\": [\n {\n \"internalType\": \"bool\",\n \"name\": \"\",\n \"type\": \"bool\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"function\"\n },\n {\n \"inputs\": [\n {\n \"internalType\": \"address\",\n \"name\": \"from\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"address\",\n \"name\": \"to\",\n \"type\": \"address\"\n },\n {\n \"internalType\": \"uint256\",\n \"name\": \"amount\",\n \"type\": \"uint256\"\n }\n ],\n \"name\": \"transferFrom\",\n \"outputs\": [\n {\n \"internalType\": \"bool\",\n \"name\": \"\",\n \"type\": \"bool\"\n }\n ],\n \"stateMutability\": \"nonpayable\",\n \"type\": \"function\"\n }\n ],\n \"bytecode\": \"0x60806040523480156200001157600080fd5b5060405162000aed38038062000aed8339810160408190526200003491620001db565b81516200004990600390602085019062000068565b5080516200005f90600490602084019062000068565b50505062000281565b828054620000769062000245565b90600052602060002090601f0160209004810192826200009a5760008555620000e5565b82601f10620000b557805160ff1916838001178555620000e5565b82800160010185558215620000e5579182015b82811115620000e5578251825591602001919060010190620000c8565b50620000f3929150620000f7565b5090565b5b80821115620000f35760008155600101620000f8565b634e487b7160e01b600052604160045260246000fd5b600082601f8301126200013657600080fd5b81516001600160401b03808211156200015357620001536200010e565b604051601f8301601f19908116603f011681019082821181831017156200017e576200017e6200010e565b816040528381526020925086838588010111156200019b57600080fd5b600091505b83821015620001bf5785820183015181830184015290820190620001a0565b83821115620001d15760008385830101525b9695505050505050565b60008060408385031215620001ef57600080fd5b82516001600160401b03808211156200020757600080fd5b620002158683870162000124565b935060208501519150808211156200022c57600080fd5b506200023b8582860162000124565b9150509250929050565b600181811c908216806200025a57607f821691505b6020821081036200027b57634e487b7160e01b600052602260045260246000fd5b50919050565b61085c80620002916000396000f3fe608060405234801561001057600080fd5b50600436106100a95760003560e01c80633950935111610071578063395093511461012357806370a082311461013657806395d89b411461015f578063a457c2d714610167578063a9059cbb1461017a578063dd62ed3e1461018d57600080fd5b806306fdde03146100ae578063095ea7b3146100cc57806318160ddd146100ef57806323b872dd14610101578063313ce56714610114575b600080fd5b6100b66101a0565b6040516100c3919061069a565b60405180910390f35b6100df6100da36600461070b565b610232565b60405190151581526020016100c3565b6002545b6040519081526020016100c3565b6100df61010f366004610735565b61024a565b604051601281526020016100c3565b6100df61013136600461070b565b61026e565b6100f3610144366004610771565b6001600160a01b031660009081526020819052604090205490565b6100b6610290565b6100df61017536600461070b565b61029f565b6100df61018836600461070b565b61031f565b6100f361019b366004610793565b61032d565b6060600380546101af906107c6565b80601f01602080910402602001604051908101604052809291908181526020018280546101db906107c6565b80156102285780601f106101fd57610100808354040283529160200191610228565b820191906000526020600020905b81548152906001019060200180831161020b57829003601f168201915b5050505050905090565b600033610240818585610358565b5060019392505050565b60003361025885828561047c565b6102638585856104f6565b506001949350505050565b600033610240818585610281838361032d565b61028b9190610800565b610358565b6060600480546101af906107c6565b600033816102ad828661032d565b9050838110156103125760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b60648201526084015b60405180910390fd5b6102638286868403610358565b6000336102408185856104f6565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b6001600160a01b0383166103ba5760405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608401610309565b6001600160a01b03821661041b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610309565b6001600160a01b0383811660008181526001602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6000610488848461032d565b905060001981146104f057818110156104e35760405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606401610309565b6104f08484848403610358565b50505050565b6001600160a01b03831661055a5760405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f206164604482015264647265737360d81b6064820152608401610309565b6001600160a01b0382166105bc5760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610309565b6001600160a01b038316600090815260208190526040902054818110156106345760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610309565b6001600160a01b03848116600081815260208181526040808320878703905593871680835291849020805487019055925185815290927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef910160405180910390a36104f0565b600060208083528351808285015260005b818110156106c7578581018301518582016040015282016106ab565b818111156106d9576000604083870101525b50601f01601f1916929092016040019392505050565b80356001600160a01b038116811461070657600080fd5b919050565b6000806040838503121561071e57600080fd5b610727836106ef565b946020939093013593505050565b60008060006060848603121561074a57600080fd5b610753846106ef565b9250610761602085016106ef565b9150604084013590509250925092565b60006020828403121561078357600080fd5b61078c826106ef565b9392505050565b600080604083850312156107a657600080fd5b6107af836106ef565b91506107bd602084016106ef565b90509250929050565b600181811c908216806107da57607f821691505b6020821081036107fa57634e487b7160e01b600052602260045260246000fd5b50919050565b6000821982111561082157634e487b7160e01b600052601160045260246000fd5b50019056fea2646970667358221220e7dc257596a973556752e9db5f465b21aad78d068ae819d0395e1f99a95065a964736f6c634300080d0033\",\n \"deployedBytecode\": \"0x608060405234801561001057600080fd5b50600436106100a95760003560e01c80633950935111610071578063395093511461012357806370a082311461013657806395d89b411461015f578063a457c2d714610167578063a9059cbb1461017a578063dd62ed3e1461018d57600080fd5b806306fdde03146100ae578063095ea7b3146100cc57806318160ddd146100ef57806323b872dd14610101578063313ce56714610114575b600080fd5b6100b66101a0565b6040516100c3919061069a565b60405180910390f35b6100df6100da36600461070b565b610232565b60405190151581526020016100c3565b6002545b6040519081526020016100c3565b6100df61010f366004610735565b61024a565b604051601281526020016100c3565b6100df61013136600461070b565b61026e565b6100f3610144366004610771565b6001600160a01b031660009081526020819052604090205490565b6100b6610290565b6100df61017536600461070b565b61029f565b6100df61018836600461070b565b61031f565b6100f361019b366004610793565b61032d565b6060600380546101af906107c6565b80601f01602080910402602001604051908101604052809291908181526020018280546101db906107c6565b80156102285780601f106101fd57610100808354040283529160200191610228565b820191906000526020600020905b81548152906001019060200180831161020b57829003601f168201915b5050505050905090565b600033610240818585610358565b5060019392505050565b60003361025885828561047c565b6102638585856104f6565b506001949350505050565b600033610240818585610281838361032d565b61028b9190610800565b610358565b6060600480546101af906107c6565b600033816102ad828661032d565b9050838110156103125760405162461bcd60e51b815260206004820152602560248201527f45524332303a2064656372656173656420616c6c6f77616e63652062656c6f77604482015264207a65726f60d81b60648201526084015b60405180910390fd5b6102638286868403610358565b6000336102408185856104f6565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b6001600160a01b0383166103ba5760405162461bcd60e51b8152602060048201526024808201527f45524332303a20617070726f76652066726f6d20746865207a65726f206164646044820152637265737360e01b6064820152608401610309565b6001600160a01b03821661041b5760405162461bcd60e51b815260206004820152602260248201527f45524332303a20617070726f766520746f20746865207a65726f206164647265604482015261737360f01b6064820152608401610309565b6001600160a01b0383811660008181526001602090815260408083209487168084529482529182902085905590518481527f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925910160405180910390a3505050565b6000610488848461032d565b905060001981146104f057818110156104e35760405162461bcd60e51b815260206004820152601d60248201527f45524332303a20696e73756666696369656e7420616c6c6f77616e63650000006044820152606401610309565b6104f08484848403610358565b50505050565b6001600160a01b03831661055a5760405162461bcd60e51b815260206004820152602560248201527f45524332303a207472616e736665722066726f6d20746865207a65726f206164604482015264647265737360d81b6064820152608401610309565b6001600160a01b0382166105bc5760405162461bcd60e51b815260206004820152602360248201527f45524332303a207472616e7366657220746f20746865207a65726f206164647260448201526265737360e81b6064820152608401610309565b6001600160a01b038316600090815260208190526040902054818110156106345760405162461bcd60e51b815260206004820152602660248201527f45524332303a207472616e7366657220616d6f756e7420657863656564732062604482015265616c616e636560d01b6064820152608401610309565b6001600160a01b03848116600081815260208181526040808320878703905593871680835291849020805487019055925185815290927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef910160405180910390a36104f0565b600060208083528351808285015260005b818110156106c7578581018301518582016040015282016106ab565b818111156106d9576000604083870101525b50601f01601f1916929092016040019392505050565b80356001600160a01b038116811461070657600080fd5b919050565b6000806040838503121561071e57600080fd5b610727836106ef565b946020939093013593505050565b60008060006060848603121561074a57600080fd5b610753846106ef565b9250610761602085016106ef565b9150604084013590509250925092565b60006020828403121561078357600080fd5b61078c826106ef565b9392505050565b600080604083850312156107a657600080fd5b6107af836106ef565b91506107bd602084016106ef565b90509250929050565b600181811c908216806107da57607f821691505b6020821081036107fa57634e487b7160e01b600052602260045260246000fd5b50919050565b6000821982111561082157634e487b7160e01b600052601160045260246000fd5b50019056fea2646970667358221220e7dc257596a973556752e9db5f465b21aad78d068ae819d0395e1f99a95065a964736f6c634300080d0033\",\n \"linkReferences\": {},\n \"deployedLinkReferences\": {}\n}\n","import type { GetBalancesResponse, GetBalancesParams, Storage } from '@avalabs/vm-module-types';\nimport { getErc20Balances } from './evm-balance-service/get-erc20-balances';\nimport { TokenService } from '@internal/utils';\nimport { getProvider } from '../../utils/get-provider';\nimport { getTokens } from '../get-tokens/get-tokens';\nimport { getNativeTokenBalances } from './evm-balance-service/get-native-token-balances';\nimport { getNativeTokenBalances as getNativeTokenBalancesFromGlacier } from './glacier-balance-service/get-native-token-balances';\nimport { getErc20Balances as getErc20BalancesFromGlacier } from './glacier-balance-service/get-erc20-balances';\nimport type { EvmGlacierService } from '../../services/glacier-service/glacier-service';\n\nexport const getBalances = async ({\n addresses,\n currency,\n network,\n proxyApiUrl,\n customTokens = [],\n storage,\n glacierService,\n}: GetBalancesParams & {\n proxyApiUrl: string;\n glacierService: EvmGlacierService;\n storage?: Storage;\n}): Promise<GetBalancesResponse> => {\n const chainId = network.chainId;\n const isNetworkSupported = await glacierService.isNetworkSupported(network.chainId);\n const isHealthy = glacierService.isHealthy();\n\n let balances = [];\n if (isHealthy && isNetworkSupported) {\n balances = await Promise.allSettled(\n addresses.map(async (address) => {\n const nativeToken = await getNativeTokenBalancesFromGlacier({\n address,\n currency,\n chainId,\n glacierService,\n });\n\n const erc20Tokens = await getErc20BalancesFromGlacier({\n customTokens,\n glacierService,\n currency,\n chainId,\n address,\n });\n\n return {\n address,\n balances: {\n [nativeToken.symbol]: nativeToken,\n ...erc20Tokens,\n },\n };\n }),\n );\n } else {\n const tokenService = new TokenService({ storage, proxyApiUrl });\n const tokens = await getTokens({ chainId: Number(chainId), proxyApiUrl });\n const allTokens = [...tokens, ...customTokens];\n const provider = getProvider({\n chainId,\n chainName: network.chainName,\n rpcUrl: network.rpcUrl,\n multiContractAddress: network.utilityAddresses?.multicall,\n });\n\n balances = await Promise.allSettled(\n addresses.map(async (address) => {\n const nativeToken = await getNativeTokenBalances({\n network,\n tokenService,\n address,\n currency,\n provider,\n });\n\n const erc20Tokens = await getErc20Balances({\n provider,\n network,\n tokenService,\n address,\n currency,\n tokens: allTokens,\n });\n\n return {\n address,\n balances: {\n [nativeToken.symbol]: nativeToken,\n ...erc20Tokens,\n },\n };\n }),\n );\n }\n\n const filterBalances = balances.reduce((acc, result) => {\n if (result.status === 'rejected') {\n return acc;\n }\n\n return {\n ...acc,\n [result.value.address]: result.value.balances,\n };\n }, {} as GetBalancesResponse);\n\n return filterBalances;\n};\n","import { balanceToDisplayValue, bigToBN, bigintToBig, bnToBig } from '@avalabs/utils-sdk';\nimport { TokenType, type Network, type NetworkTokenWithBalance } from '@avalabs/vm-module-types';\nimport type { VsCurrencyType } from '@avalabs/coingecko-sdk';\nimport { TokenService } from '@internal/utils';\nimport type { Provider } from 'ethers';\n\nexport const getNativeTokenBalances = async ({\n provider,\n tokenService,\n address,\n currency,\n network,\n}: {\n provider: Provider;\n tokenService: TokenService;\n address: string;\n currency: string;\n network: Network;\n}): Promise<NetworkTokenWithBalance> => {\n const coingeckoTokenId = network.pricingProviders?.coingecko.nativeTokenId;\n const networkToken = network.networkToken;\n const simplePriceResponse = coingeckoTokenId\n ? await tokenService.getSimplePrice({\n coinIds: [coingeckoTokenId],\n currencies: [currency] as VsCurrencyType[],\n })\n : {};\n\n const priceInCurrency = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.price ?? 0;\n const marketCap = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.marketCap ?? 0;\n const vol24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.vol24 ?? 0;\n const change24 = simplePriceResponse?.[coingeckoTokenId ?? '']?.[currency]?.change24 ?? 0;\n\n const balanceBigint = await provider.getBalance(address);\n const balaceBig = bigintToBig(balanceBigint, networkToken.decimals);\n const balance = bigToBN(balaceBig, networkToken.decimals);\n const balanceDisplayValue = balanceToDisplayValue(balance, networkToken.decimals);\n const balanceInCurrency = bnToBig(balance, networkToken.decimals).mul(priceInCurrency).toNumber();\n const balanceCurrencyDisplayValue = balanceInCurrency.toFixed(2);\n\n return {\n ...networkToken,\n coingeckoId: coingeckoTokenId ?? '',\n type: TokenType.NATIVE,\n balance,\n balanceDisplayValue,\n balanceInCurrency,\n balanceCurrencyDisplayValue,\n priceInCurrency,\n marketCap,\n vol24,\n change24,\n };\n};\n","import type { CurrencyCode } from '@avalabs/glacier-sdk';\nimport { TokenType, type NetworkTokenWithBalance } from '@avalabs/vm-module-types';\nimport { balanceToDisplayValue, bnToBig } from '@avalabs/utils-sdk';\nimport { BN } from 'bn.js';\nimport type { EvmGlacierService } from '../../../services/glacier-service/glacier-service';\n\nexport const getNativeTokenBalances = async ({\n address,\n currency,\n chainId,\n glacierService,\n}: {\n chainId: number;\n address: string;\n currency: string;\n glacierService: EvmGlacierService;\n}): Promise<NetworkTokenWithBalance> => {\n const nativeBalance = await glacierService.getNativeBalance({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n });\n const nativeTokenBalance = nativeBalance.nativeTokenBalance;\n const balance = new BN(nativeTokenBalance.balance);\n const balanceDisplayValue = balanceToDisplayValue(balance, nativeTokenBalance.decimals);\n const priceInCurrency = nativeTokenBalance.price?.value ?? 0;\n const balanceInCurrency = bnToBig(balance, nativeTokenBalance.decimals).mul(priceInCurrency).toNumber();\n const balanceCurrencyDisplayValue = balanceInCurrency.toFixed(2);\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,\n balanceDisplayValue,\n balanceInCurrency,\n balanceCurrencyDisplayValue,\n priceInCurrency,\n marketCap: 0,\n vol24: 0,\n change24: 0,\n coingeckoId: '',\n };\n};\n","import { balanceToDisplayValue, bnToBig } from '@avalabs/utils-sdk';\nimport {\n TokenType,\n type TokenWithBalanceERC20,\n type NetworkContractToken,\n type TokenWithBalance,\n} from '@avalabs/vm-module-types';\nimport { CurrencyCode, Erc20TokenBalance } from '@avalabs/glacier-sdk';\nimport BN from 'bn.js';\nimport type { EvmGlacierService } from '../../../services/glacier-service/glacier-service';\n\nexport const getErc20Balances = async ({\n glacierService,\n currency,\n chainId,\n address,\n customTokens,\n}: {\n glacierService: EvmGlacierService;\n address: string;\n currency: string;\n chainId: number;\n customTokens: NetworkContractToken[];\n}): Promise<Record<string, TokenWithBalance>> => {\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;\n do {\n const response = await glacierService.listErc20Balances({\n chainId: chainId.toString(),\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n // glacier has a cap on page size of 100\n pageSize: 100,\n pageToken: nextPageToken,\n });\n\n tokensWithBalance.push(...convertErc20ToTokenWithBalance(response.erc20TokenBalances, Number(chainId)));\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 ...convertNetworkTokenToTokenWithBalance(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, TokenWithBalance>,\n );\n};\n\nconst convertNetworkTokenToTokenWithBalance = (tokens: NetworkContractToken[]): TokenWithBalanceERC20[] => {\n return tokens.map((token) => {\n return {\n ...token,\n type: TokenType.ERC20,\n balance: new BN(0),\n balanceInCurrency: 0,\n balanceDisplayValue: '0',\n balanceCurrencyDisplayValue: '0',\n priceInCurrency: 0,\n marketCap: 0,\n change24: 0,\n vol24: 0,\n };\n });\n};\n\nconst convertErc20ToTokenWithBalance = (\n tokenBalances: Erc20TokenBalance[],\n chainId: number,\n): TokenWithBalanceERC20[] => {\n return tokenBalances.map((token: Erc20TokenBalance): TokenWithBalanceERC20 => {\n const balance = new BN(token.balance);\n const balanceDisplayValue = balanceToDisplayValue(balance, token.decimals);\n const balanceCurrencyDisplayValue = token.balanceValue?.value.toString() ?? '0';\n const priceInCurrency = token.price?.value ?? 0;\n const balanceInCurrency = bnToBig(balance, token.decimals).mul(priceInCurrency).toNumber();\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,\n balanceCurrencyDisplayValue,\n balanceDisplayValue,\n balanceInCurrency,\n priceInCurrency,\n contractType: 'ERC-20',\n type: TokenType.ERC20,\n change24: 0,\n marketCap: 0,\n vol24: 0,\n };\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 BlockchainId,\n CurrencyCode,\n Erc1155Token,\n Erc721Token,\n type GetNativeBalanceResponse,\n Glacier,\n type ListCChainAtomicBalancesResponse,\n type ListErc1155BalancesResponse,\n type ListErc20BalancesResponse,\n type ListErc721BalancesResponse,\n type ListPChainBalancesResponse,\n type ListXChainBalancesResponse,\n Network,\n} from '@avalabs/glacier-sdk';\n\nclass GlacierUnhealthyError extends Error {\n override message = 'Glacier is unhealthy. Try again later.';\n}\n\nexport class EvmGlacierService {\n glacierSdk: Glacier;\n isGlacierHealthy = true;\n supportedChainIds: string[] = [];\n\n constructor({ glacierApiUrl }: { glacierApiUrl: string }) {\n this.glacierSdk = new Glacier({ BASE: glacierApiUrl });\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 this.supportedChainIds = supportedChains.chains.map((chain) => chain.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 getChainBalance(params: {\n blockchainId: BlockchainId;\n network: Network;\n blockTimestamp?: number;\n addresses?: string;\n }): Promise<ListPChainBalancesResponse | ListXChainBalancesResponse | ListCChainAtomicBalancesResponse> {\n try {\n return this.glacierSdk.primaryNetworkBalances.getBalancesByAddresses(params);\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 }: {\n chainId: string;\n address: string;\n currency: CurrencyCode;\n }): Promise<GetNativeBalanceResponse> {\n try {\n return this.glacierSdk.evmBalances.getNativeBalance({\n chainId,\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listErc721Balances({\n chainId,\n address,\n pageSize,\n pageToken,\n }: {\n chainId: string;\n address: string;\n pageSize: number;\n pageToken?: string;\n }): Promise<ListErc721BalancesResponse> {\n try {\n return this.glacierSdk.evmBalances.listErc721Balances({\n chainId,\n address,\n pageSize,\n pageToken,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listErc1155Balances({\n chainId,\n address,\n pageSize,\n pageToken,\n }: {\n chainId: string;\n address: string;\n pageSize: number;\n pageToken?: string;\n }): Promise<ListErc1155BalancesResponse> {\n try {\n return this.glacierSdk.evmBalances.listErc1155Balances({\n chainId,\n address,\n pageSize,\n pageToken,\n });\n } catch (error) {\n if (error instanceof GlacierUnhealthyError) {\n this.setGlacierToUnhealthy();\n }\n throw error;\n }\n }\n\n async listErc20Balances({\n chainId,\n address,\n currency,\n pageSize,\n pageToken,\n }: {\n chainId: string;\n address: string;\n currency: CurrencyCode;\n pageSize: number;\n pageToken?: string;\n }): Promise<ListErc20BalancesResponse> {\n try {\n return this.glacierSdk.evmBalances.listErc20Balances({\n chainId,\n address,\n currency: currency.toLocaleLowerCase() as CurrencyCode,\n pageSize,\n pageToken,\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"]}