@avalabs/fusion-sdk 0.14.1 → 0.14.3
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.
- package/dist/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.cjs +1 -1
- package/dist/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.cjs.map +1 -1
- package/dist/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.js +1 -1
- package/dist/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.js.map +1 -1
- package/dist/transfer-service/markr/_handlers/estimate-native-fee.cjs +1 -1
- package/dist/transfer-service/markr/_handlers/estimate-native-fee.cjs.map +1 -1
- package/dist/transfer-service/markr/_handlers/estimate-native-fee.js +1 -1
- package/dist/transfer-service/markr/_handlers/estimate-native-fee.js.map +1 -1
- package/dist/transfer-service/markr/_handlers/transfer-asset.cjs +1 -1
- package/dist/transfer-service/markr/_handlers/transfer-asset.cjs.map +1 -1
- package/dist/transfer-service/markr/_handlers/transfer-asset.js +1 -1
- package/dist/transfer-service/markr/_handlers/transfer-asset.js.map +1 -1
- package/dist/transfer-service/markr/_utils.cjs +1 -1
- package/dist/transfer-service/markr/_utils.cjs.map +1 -1
- package/dist/transfer-service/markr/_utils.js +1 -1
- package/dist/transfer-service/markr/_utils.js.map +1 -1
- package/package.json +1 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
require(`../../../../_virtual/_rolldown/runtime.cjs`);const e=require(`../../../../errors.cjs`),t=require(`../../../_utils.cjs`);let n=require(`viem`),r=require(`@lombard.finance/sdk`);function i({bitcoinFunctions:e,config:n}){return({transfer:r,updateListener:i})=>{let s=new AbortController,c=()=>{s.abort()};return r.status===`completed`||r.status===`failed`?{cancel:c,result:Promise.resolve(r)}:{cancel:c,result:(async()=>{let c=structuredClone(r),l=t.getEvmClientForChain({chain:r.sourceChain});for(;!s.signal.aborted;){let r=c.status;switch(c.status){case`source-pending`:c=await a({currentTransfer:c,signal:s.signal,sourceClient:l}),s.signal.aborted||i(c);break;case`source-completed`:case`target-pending`:c=await o({bitcoinFunctions:e,config:n,currentTransfer:c,signal:s.signal}),s.signal.aborted||i(c);break;default:return c}c.status===r&&await t.waitForTimeoutOrAbort({timeoutMs:6e4,signal:s.signal})}return c})()}}}async function a({currentTransfer:r,signal:i,sourceClient:a}){let{confirmationCount:o,requiredConfirmationCount:s,txHash:c}=r.source;if(!(0,n.isHash)(c))return{...r,errorCode:e.ErrorCode.INVALID_PARAMS,errorReason:`Invalid source transaction hash`,failedAtMs:Date.now(),status:`failed`};if(o>=s)return{...r,status:`source-completed`};try{let n=await t.awaitOrAbort(a.waitForTransactionReceipt({
|
|
1
|
+
require(`../../../../_virtual/_rolldown/runtime.cjs`);const e=require(`../../../../errors.cjs`),t=require(`../../../_utils.cjs`);let n=require(`viem`),r=require(`@lombard.finance/sdk`);function i({bitcoinFunctions:e,config:n}){return({transfer:r,updateListener:i})=>{let s=new AbortController,c=()=>{s.abort()};return r.status===`completed`||r.status===`failed`?{cancel:c,result:Promise.resolve(r)}:{cancel:c,result:(async()=>{let c=structuredClone(r),l=t.getEvmClientForChain({chain:r.sourceChain});for(;!s.signal.aborted;){let r=c.status;switch(c.status){case`source-pending`:c=await a({currentTransfer:c,signal:s.signal,sourceClient:l}),s.signal.aborted||i(c);break;case`source-completed`:case`target-pending`:c=await o({bitcoinFunctions:e,config:n,currentTransfer:c,signal:s.signal}),s.signal.aborted||i(c);break;default:return c}c.status===r&&await t.waitForTimeoutOrAbort({timeoutMs:6e4,signal:s.signal})}return c})()}}}async function a({currentTransfer:r,signal:i,sourceClient:a}){let{confirmationCount:o,requiredConfirmationCount:s,txHash:c}=r.source;if(!(0,n.isHash)(c))return{...r,errorCode:e.ErrorCode.INVALID_PARAMS,errorReason:`Invalid source transaction hash`,failedAtMs:Date.now(),status:`failed`};if(o>=s)return{...r,status:`source-completed`};try{let n=await t.awaitOrAbort(a.waitForTransactionReceipt({hash:c}),i);if(n.status===`aborted`)return r;let l=n.value;if(l.status===`reverted`)return{...r,errorCode:e.ErrorCode.TRANSACTION_REVERTED,errorReason:`Source transaction was reverted`,failedAtMs:Date.now(),status:`failed`};let u=await t.awaitOrAbort(a.getTransactionConfirmations({transactionReceipt:l}).then(e=>Math.max(Number(e),0)).catch(()=>Math.max(o,0)),i);if(u.status===`aborted`)return r;let d=u.value;return d<s?{...r,source:{...r.source,confirmationCount:d}}:{...r,source:{...r.source,confirmationCount:d},status:`source-completed`}}catch{return r}}async function o({bitcoinFunctions:n,config:i,currentTransfer:a,signal:o}){switch(a.status){case`source-completed`:try{let e=await t.awaitOrAbort((0,r.getUnstakesByAddress)({address:a.fromAddress,env:i.env}),o);if(e.status===`aborted`)return a;let n=e.value.find(e=>e.txHash.toLowerCase()===a.source.txHash.toLowerCase());return n?.payoutTxHash?{...a,status:`target-pending`,target:{confirmationCount:0,requiredConfirmationCount:1,startedAtMs:Date.now(),txHash:n.payoutTxHash}}:a}catch{return a}default:if(!a.target?.txHash)return{...a,errorCode:e.ErrorCode.UNKNOWN,errorReason:`Missing target transaction hash`,failedAtMs:Date.now(),status:`failed`};try{let e=await t.awaitOrAbort(n.getTransaction(a.target.txHash),o);if(e.status===`aborted`)return a;let r=e.value.confirmations;return r<a.target.requiredConfirmationCount?{...a,target:{...a.target,confirmationCount:r}}:{...a,completedAtMs:Date.now(),target:{...a.target,confirmationCount:r},status:`completed`}}catch{return a}}}exports.trackTransferFactory=i;
|
|
2
2
|
//# sourceMappingURL=track-transfer.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"track-transfer.cjs","names":["getEvmClientForChain","waitForTimeoutOrAbort","ErrorCode","awaitOrAbort"],"sources":["../../../../../src/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.ts"],"sourcesContent":["import { isHash } from 'viem';\nimport { ErrorCode } from '../../../../errors';\nimport type { BitcoinFunctions } from '../../../../types/bitcoin';\nimport type { TransferService } from '../../../../types/service';\nimport type {\n CompletedTransfer,\n FailedTransfer,\n SourceCompletedTransfer,\n SourcePendingTransfer,\n TargetPendingTransfer,\n Transfer,\n} from '../../../../types/transfer';\nimport type { BtcbToBtcConfig } from '../../types';\nimport { awaitOrAbort, getEvmClientForChain, waitForTimeoutOrAbort } from '../../../_utils';\nimport { getUnstakesByAddress } from '@lombard.finance/sdk';\n\n// Polling interval for BTC redemptions (60 seconds - BTC confirmations are slow)\nconst REDEMPTION_POLLING_INTERVAL_MS = 1000 * 60;\n\nexport interface TrackTransferFactoryOptions {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n}\n\nexport function trackTransferFactory({\n bitcoinFunctions,\n config,\n}: TrackTransferFactoryOptions): TransferService['trackTransfer'] {\n return ({ transfer, updateListener }) => {\n const ac = new AbortController();\n const cancel = () => {\n ac.abort();\n };\n\n if (transfer.status === 'completed' || transfer.status === 'failed') {\n return {\n cancel,\n result: Promise.resolve(transfer),\n };\n }\n\n const executeTracking = async (): Promise<Transfer> => {\n let currentTransfer: Transfer = structuredClone(transfer);\n\n const sourceClient = getEvmClientForChain({ chain: transfer.sourceChain });\n\n while (!ac.signal.aborted) {\n const statusBeforeTracking = currentTransfer.status;\n\n switch (currentTransfer.status) {\n case 'source-pending': {\n currentTransfer = await _trackSourceTx({\n currentTransfer,\n signal: ac.signal,\n sourceClient,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n case 'source-completed':\n case 'target-pending': {\n currentTransfer = await _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal: ac.signal,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n default: {\n return currentTransfer;\n }\n }\n\n const hasStatusChanged = currentTransfer.status !== statusBeforeTracking;\n\n if (!hasStatusChanged) {\n await waitForTimeoutOrAbort({ timeoutMs: REDEMPTION_POLLING_INTERVAL_MS, signal: ac.signal });\n }\n }\n\n return currentTransfer;\n };\n\n return {\n cancel,\n result: executeTracking(),\n };\n };\n}\n\nexport interface TrackBaseParams {\n currentTransfer: Transfer;\n signal: AbortSignal;\n}\n\nexport interface TrackSourceTxParams extends TrackBaseParams {\n currentTransfer: SourcePendingTransfer;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n}\n\nexport interface TrackTargetTxParams extends TrackBaseParams {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n currentTransfer: SourceCompletedTransfer | TargetPendingTransfer;\n}\n\nexport async function _trackSourceTx({\n currentTransfer,\n signal,\n sourceClient,\n}: TrackSourceTxParams): Promise<SourcePendingTransfer | SourceCompletedTransfer | FailedTransfer> {\n const { confirmationCount, requiredConfirmationCount, txHash } = currentTransfer.source;\n\n if (!isHash(txHash)) {\n const failedTransfer: FailedTransfer = {\n ...currentTransfer,\n errorCode: ErrorCode.INVALID_PARAMS,\n errorReason: 'Invalid source transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n };\n\n return failedTransfer;\n }\n\n if (confirmationCount >= requiredConfirmationCount) {\n // Move to next stage\n\n const updatedTransfer: SourceCompletedTransfer = {\n ...currentTransfer,\n status: 'source-completed',\n };\n\n return updatedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n sourceClient.waitForTransactionReceipt({\n confirmations: Math.min(confirmationCount + 1, requiredConfirmationCount),\n hash: txHash,\n }),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transactionReceipt = receiptResult.value;\n\n if (transactionReceipt.status === 'reverted') {\n return {\n ...currentTransfer,\n errorCode: ErrorCode.TRANSACTION_REVERTED,\n errorReason: 'Source transaction was reverted',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n const newConfirmationCount: number = await sourceClient\n .getTransactionConfirmations({\n transactionReceipt,\n })\n .then((count) => Number(count))\n .catch(() => confirmationCount + 1);\n\n // If still not enough confirmations, update the count\n if (newConfirmationCount < requiredConfirmationCount) {\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n } satisfies SourcePendingTransfer;\n }\n\n // Move to next stage\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n status: 'source-completed',\n } satisfies SourceCompletedTransfer;\n } catch {\n return currentTransfer;\n }\n}\n\nexport async function _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal,\n}: TrackTargetTxParams): Promise<SourceCompletedTransfer | TargetPendingTransfer | FailedTransfer | CompletedTransfer> {\n switch (currentTransfer.status) {\n case 'source-completed': {\n try {\n const unstakesResult = await awaitOrAbort(\n getUnstakesByAddress({ address: currentTransfer.fromAddress, env: config.env }),\n signal,\n );\n\n if (unstakesResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transferUnstake = unstakesResult.value.find(\n (d) => d.txHash.toLowerCase() === currentTransfer.source.txHash.toLowerCase(),\n );\n\n if (transferUnstake?.payoutTxHash) {\n const updatedTransfer: TargetPendingTransfer = {\n ...currentTransfer,\n status: 'target-pending',\n target: {\n confirmationCount: 0,\n requiredConfirmationCount: 1,\n startedAtMs: Date.now(),\n txHash: transferUnstake.payoutTxHash,\n },\n };\n\n return updatedTransfer;\n }\n\n return currentTransfer;\n } catch {\n return currentTransfer;\n }\n }\n default: {\n // target-pending\n if (!currentTransfer.target?.txHash) {\n // Fail if no target tx hash, should not reach here\n return {\n ...currentTransfer,\n errorCode: ErrorCode.UNKNOWN,\n errorReason: 'Missing target transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n bitcoinFunctions.getTransaction(currentTransfer.target.txHash),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const targetConfirmationCount = receiptResult.value.confirmations;\n\n if (targetConfirmationCount < currentTransfer.target.requiredConfirmationCount) {\n return {\n ...currentTransfer,\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n } satisfies TargetPendingTransfer;\n }\n\n // Completed\n return {\n ...currentTransfer,\n completedAtMs: Date.now(),\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n status: 'completed',\n } satisfies CompletedTransfer;\n } catch {\n return currentTransfer;\n }\n }\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"track-transfer.cjs","names":["getEvmClientForChain","waitForTimeoutOrAbort","ErrorCode","awaitOrAbort"],"sources":["../../../../../src/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.ts"],"sourcesContent":["import { isHash } from 'viem';\nimport { ErrorCode } from '../../../../errors';\nimport type { BitcoinFunctions } from '../../../../types/bitcoin';\nimport type { TransferService } from '../../../../types/service';\nimport type {\n CompletedTransfer,\n FailedTransfer,\n SourceCompletedTransfer,\n SourcePendingTransfer,\n TargetPendingTransfer,\n Transfer,\n} from '../../../../types/transfer';\nimport type { BtcbToBtcConfig } from '../../types';\nimport { awaitOrAbort, getEvmClientForChain, waitForTimeoutOrAbort } from '../../../_utils';\nimport { getUnstakesByAddress } from '@lombard.finance/sdk';\n\n// Polling interval for BTC redemptions (60 seconds - BTC confirmations are slow)\nconst REDEMPTION_POLLING_INTERVAL_MS = 1000 * 60;\n\nexport interface TrackTransferFactoryOptions {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n}\n\nexport function trackTransferFactory({\n bitcoinFunctions,\n config,\n}: TrackTransferFactoryOptions): TransferService['trackTransfer'] {\n return ({ transfer, updateListener }) => {\n const ac = new AbortController();\n const cancel = () => {\n ac.abort();\n };\n\n if (transfer.status === 'completed' || transfer.status === 'failed') {\n return {\n cancel,\n result: Promise.resolve(transfer),\n };\n }\n\n const executeTracking = async (): Promise<Transfer> => {\n let currentTransfer: Transfer = structuredClone(transfer);\n\n const sourceClient = getEvmClientForChain({ chain: transfer.sourceChain });\n\n while (!ac.signal.aborted) {\n const statusBeforeTracking = currentTransfer.status;\n\n switch (currentTransfer.status) {\n case 'source-pending': {\n currentTransfer = await _trackSourceTx({\n currentTransfer,\n signal: ac.signal,\n sourceClient,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n case 'source-completed':\n case 'target-pending': {\n currentTransfer = await _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal: ac.signal,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n default: {\n return currentTransfer;\n }\n }\n\n const hasStatusChanged = currentTransfer.status !== statusBeforeTracking;\n\n if (!hasStatusChanged) {\n await waitForTimeoutOrAbort({ timeoutMs: REDEMPTION_POLLING_INTERVAL_MS, signal: ac.signal });\n }\n }\n\n return currentTransfer;\n };\n\n return {\n cancel,\n result: executeTracking(),\n };\n };\n}\n\nexport interface TrackBaseParams {\n currentTransfer: Transfer;\n signal: AbortSignal;\n}\n\nexport interface TrackSourceTxParams extends TrackBaseParams {\n currentTransfer: SourcePendingTransfer;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n}\n\nexport interface TrackTargetTxParams extends TrackBaseParams {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n currentTransfer: SourceCompletedTransfer | TargetPendingTransfer;\n}\n\nexport async function _trackSourceTx({\n currentTransfer,\n signal,\n sourceClient,\n}: TrackSourceTxParams): Promise<SourcePendingTransfer | SourceCompletedTransfer | FailedTransfer> {\n const { confirmationCount, requiredConfirmationCount, txHash } = currentTransfer.source;\n\n if (!isHash(txHash)) {\n const failedTransfer: FailedTransfer = {\n ...currentTransfer,\n errorCode: ErrorCode.INVALID_PARAMS,\n errorReason: 'Invalid source transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n };\n\n return failedTransfer;\n }\n\n if (confirmationCount >= requiredConfirmationCount) {\n // Move to next stage\n\n const updatedTransfer: SourceCompletedTransfer = {\n ...currentTransfer,\n status: 'source-completed',\n };\n\n return updatedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n sourceClient.waitForTransactionReceipt({\n hash: txHash,\n }),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transactionReceipt = receiptResult.value;\n\n if (transactionReceipt.status === 'reverted') {\n return {\n ...currentTransfer,\n errorCode: ErrorCode.TRANSACTION_REVERTED,\n errorReason: 'Source transaction was reverted',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n const newConfirmationCountResult = await awaitOrAbort(\n sourceClient\n .getTransactionConfirmations({\n transactionReceipt,\n })\n .then((count) => Math.max(Number(count), 0))\n .catch(() => Math.max(confirmationCount, 0)),\n signal,\n );\n\n if (newConfirmationCountResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const newConfirmationCount = newConfirmationCountResult.value;\n\n // If still not enough confirmations, update the count\n if (newConfirmationCount < requiredConfirmationCount) {\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n } satisfies SourcePendingTransfer;\n }\n\n // Move to next stage\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n status: 'source-completed',\n } satisfies SourceCompletedTransfer;\n } catch {\n return currentTransfer;\n }\n}\n\nexport async function _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal,\n}: TrackTargetTxParams): Promise<SourceCompletedTransfer | TargetPendingTransfer | FailedTransfer | CompletedTransfer> {\n switch (currentTransfer.status) {\n case 'source-completed': {\n try {\n const unstakesResult = await awaitOrAbort(\n getUnstakesByAddress({ address: currentTransfer.fromAddress, env: config.env }),\n signal,\n );\n\n if (unstakesResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transferUnstake = unstakesResult.value.find(\n (d) => d.txHash.toLowerCase() === currentTransfer.source.txHash.toLowerCase(),\n );\n\n if (transferUnstake?.payoutTxHash) {\n const updatedTransfer: TargetPendingTransfer = {\n ...currentTransfer,\n status: 'target-pending',\n target: {\n confirmationCount: 0,\n requiredConfirmationCount: 1,\n startedAtMs: Date.now(),\n txHash: transferUnstake.payoutTxHash,\n },\n };\n\n return updatedTransfer;\n }\n\n return currentTransfer;\n } catch {\n return currentTransfer;\n }\n }\n default: {\n // target-pending\n if (!currentTransfer.target?.txHash) {\n // Fail if no target tx hash, should not reach here\n return {\n ...currentTransfer,\n errorCode: ErrorCode.UNKNOWN,\n errorReason: 'Missing target transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n bitcoinFunctions.getTransaction(currentTransfer.target.txHash),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const targetConfirmationCount = receiptResult.value.confirmations;\n\n if (targetConfirmationCount < currentTransfer.target.requiredConfirmationCount) {\n return {\n ...currentTransfer,\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n } satisfies TargetPendingTransfer;\n }\n\n // Completed\n return {\n ...currentTransfer,\n completedAtMs: Date.now(),\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n status: 'completed',\n } satisfies CompletedTransfer;\n } catch {\n return currentTransfer;\n }\n }\n }\n}\n"],"mappings":"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"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{ErrorCode as e}from"../../../../errors.js";import{awaitOrAbort as t,getEvmClientForChain as n,waitForTimeoutOrAbort as r}from"../../../_utils.js";import{isHash as i}from"viem";import{getUnstakesByAddress as a}from"@lombard.finance/sdk";function o({bitcoinFunctions:e,config:t}){return({transfer:i,updateListener:a})=>{let o=new AbortController,l=()=>{o.abort()};return i.status===`completed`||i.status===`failed`?{cancel:l,result:Promise.resolve(i)}:{cancel:l,result:(async()=>{let l=structuredClone(i),u=n({chain:i.sourceChain});for(;!o.signal.aborted;){let n=l.status;switch(l.status){case`source-pending`:l=await s({currentTransfer:l,signal:o.signal,sourceClient:u}),o.signal.aborted||a(l);break;case`source-completed`:case`target-pending`:l=await c({bitcoinFunctions:e,config:t,currentTransfer:l,signal:o.signal}),o.signal.aborted||a(l);break;default:return l}l.status===n&&await r({timeoutMs:6e4,signal:o.signal})}return l})()}}}async function s({currentTransfer:n,signal:r,sourceClient:a}){let{confirmationCount:o,requiredConfirmationCount:s,txHash:c}=n.source;if(!i(c))return{...n,errorCode:e.INVALID_PARAMS,errorReason:`Invalid source transaction hash`,failedAtMs:Date.now(),status:`failed`};if(o>=s)return{...n,status:`source-completed`};try{let i=await t(a.waitForTransactionReceipt({
|
|
1
|
+
import{ErrorCode as e}from"../../../../errors.js";import{awaitOrAbort as t,getEvmClientForChain as n,waitForTimeoutOrAbort as r}from"../../../_utils.js";import{isHash as i}from"viem";import{getUnstakesByAddress as a}from"@lombard.finance/sdk";function o({bitcoinFunctions:e,config:t}){return({transfer:i,updateListener:a})=>{let o=new AbortController,l=()=>{o.abort()};return i.status===`completed`||i.status===`failed`?{cancel:l,result:Promise.resolve(i)}:{cancel:l,result:(async()=>{let l=structuredClone(i),u=n({chain:i.sourceChain});for(;!o.signal.aborted;){let n=l.status;switch(l.status){case`source-pending`:l=await s({currentTransfer:l,signal:o.signal,sourceClient:u}),o.signal.aborted||a(l);break;case`source-completed`:case`target-pending`:l=await c({bitcoinFunctions:e,config:t,currentTransfer:l,signal:o.signal}),o.signal.aborted||a(l);break;default:return l}l.status===n&&await r({timeoutMs:6e4,signal:o.signal})}return l})()}}}async function s({currentTransfer:n,signal:r,sourceClient:a}){let{confirmationCount:o,requiredConfirmationCount:s,txHash:c}=n.source;if(!i(c))return{...n,errorCode:e.INVALID_PARAMS,errorReason:`Invalid source transaction hash`,failedAtMs:Date.now(),status:`failed`};if(o>=s)return{...n,status:`source-completed`};try{let i=await t(a.waitForTransactionReceipt({hash:c}),r);if(i.status===`aborted`)return n;let l=i.value;if(l.status===`reverted`)return{...n,errorCode:e.TRANSACTION_REVERTED,errorReason:`Source transaction was reverted`,failedAtMs:Date.now(),status:`failed`};let u=await t(a.getTransactionConfirmations({transactionReceipt:l}).then(e=>Math.max(Number(e),0)).catch(()=>Math.max(o,0)),r);if(u.status===`aborted`)return n;let d=u.value;return d<s?{...n,source:{...n.source,confirmationCount:d}}:{...n,source:{...n.source,confirmationCount:d},status:`source-completed`}}catch{return n}}async function c({bitcoinFunctions:n,config:r,currentTransfer:i,signal:o}){switch(i.status){case`source-completed`:try{let e=await t(a({address:i.fromAddress,env:r.env}),o);if(e.status===`aborted`)return i;let n=e.value.find(e=>e.txHash.toLowerCase()===i.source.txHash.toLowerCase());return n?.payoutTxHash?{...i,status:`target-pending`,target:{confirmationCount:0,requiredConfirmationCount:1,startedAtMs:Date.now(),txHash:n.payoutTxHash}}:i}catch{return i}default:if(!i.target?.txHash)return{...i,errorCode:e.UNKNOWN,errorReason:`Missing target transaction hash`,failedAtMs:Date.now(),status:`failed`};try{let e=await t(n.getTransaction(i.target.txHash),o);if(e.status===`aborted`)return i;let r=e.value.confirmations;return r<i.target.requiredConfirmationCount?{...i,target:{...i.target,confirmationCount:r}}:{...i,completedAtMs:Date.now(),target:{...i.target,confirmationCount:r},status:`completed`}}catch{return i}}}export{o as trackTransferFactory};
|
|
2
2
|
//# sourceMappingURL=track-transfer.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"track-transfer.js","names":[],"sources":["../../../../../src/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.ts"],"sourcesContent":["import { isHash } from 'viem';\nimport { ErrorCode } from '../../../../errors';\nimport type { BitcoinFunctions } from '../../../../types/bitcoin';\nimport type { TransferService } from '../../../../types/service';\nimport type {\n CompletedTransfer,\n FailedTransfer,\n SourceCompletedTransfer,\n SourcePendingTransfer,\n TargetPendingTransfer,\n Transfer,\n} from '../../../../types/transfer';\nimport type { BtcbToBtcConfig } from '../../types';\nimport { awaitOrAbort, getEvmClientForChain, waitForTimeoutOrAbort } from '../../../_utils';\nimport { getUnstakesByAddress } from '@lombard.finance/sdk';\n\n// Polling interval for BTC redemptions (60 seconds - BTC confirmations are slow)\nconst REDEMPTION_POLLING_INTERVAL_MS = 1000 * 60;\n\nexport interface TrackTransferFactoryOptions {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n}\n\nexport function trackTransferFactory({\n bitcoinFunctions,\n config,\n}: TrackTransferFactoryOptions): TransferService['trackTransfer'] {\n return ({ transfer, updateListener }) => {\n const ac = new AbortController();\n const cancel = () => {\n ac.abort();\n };\n\n if (transfer.status === 'completed' || transfer.status === 'failed') {\n return {\n cancel,\n result: Promise.resolve(transfer),\n };\n }\n\n const executeTracking = async (): Promise<Transfer> => {\n let currentTransfer: Transfer = structuredClone(transfer);\n\n const sourceClient = getEvmClientForChain({ chain: transfer.sourceChain });\n\n while (!ac.signal.aborted) {\n const statusBeforeTracking = currentTransfer.status;\n\n switch (currentTransfer.status) {\n case 'source-pending': {\n currentTransfer = await _trackSourceTx({\n currentTransfer,\n signal: ac.signal,\n sourceClient,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n case 'source-completed':\n case 'target-pending': {\n currentTransfer = await _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal: ac.signal,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n default: {\n return currentTransfer;\n }\n }\n\n const hasStatusChanged = currentTransfer.status !== statusBeforeTracking;\n\n if (!hasStatusChanged) {\n await waitForTimeoutOrAbort({ timeoutMs: REDEMPTION_POLLING_INTERVAL_MS, signal: ac.signal });\n }\n }\n\n return currentTransfer;\n };\n\n return {\n cancel,\n result: executeTracking(),\n };\n };\n}\n\nexport interface TrackBaseParams {\n currentTransfer: Transfer;\n signal: AbortSignal;\n}\n\nexport interface TrackSourceTxParams extends TrackBaseParams {\n currentTransfer: SourcePendingTransfer;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n}\n\nexport interface TrackTargetTxParams extends TrackBaseParams {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n currentTransfer: SourceCompletedTransfer | TargetPendingTransfer;\n}\n\nexport async function _trackSourceTx({\n currentTransfer,\n signal,\n sourceClient,\n}: TrackSourceTxParams): Promise<SourcePendingTransfer | SourceCompletedTransfer | FailedTransfer> {\n const { confirmationCount, requiredConfirmationCount, txHash } = currentTransfer.source;\n\n if (!isHash(txHash)) {\n const failedTransfer: FailedTransfer = {\n ...currentTransfer,\n errorCode: ErrorCode.INVALID_PARAMS,\n errorReason: 'Invalid source transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n };\n\n return failedTransfer;\n }\n\n if (confirmationCount >= requiredConfirmationCount) {\n // Move to next stage\n\n const updatedTransfer: SourceCompletedTransfer = {\n ...currentTransfer,\n status: 'source-completed',\n };\n\n return updatedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n sourceClient.waitForTransactionReceipt({\n confirmations: Math.min(confirmationCount + 1, requiredConfirmationCount),\n hash: txHash,\n }),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transactionReceipt = receiptResult.value;\n\n if (transactionReceipt.status === 'reverted') {\n return {\n ...currentTransfer,\n errorCode: ErrorCode.TRANSACTION_REVERTED,\n errorReason: 'Source transaction was reverted',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n const newConfirmationCount: number = await sourceClient\n .getTransactionConfirmations({\n transactionReceipt,\n })\n .then((count) => Number(count))\n .catch(() => confirmationCount + 1);\n\n // If still not enough confirmations, update the count\n if (newConfirmationCount < requiredConfirmationCount) {\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n } satisfies SourcePendingTransfer;\n }\n\n // Move to next stage\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n status: 'source-completed',\n } satisfies SourceCompletedTransfer;\n } catch {\n return currentTransfer;\n }\n}\n\nexport async function _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal,\n}: TrackTargetTxParams): Promise<SourceCompletedTransfer | TargetPendingTransfer | FailedTransfer | CompletedTransfer> {\n switch (currentTransfer.status) {\n case 'source-completed': {\n try {\n const unstakesResult = await awaitOrAbort(\n getUnstakesByAddress({ address: currentTransfer.fromAddress, env: config.env }),\n signal,\n );\n\n if (unstakesResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transferUnstake = unstakesResult.value.find(\n (d) => d.txHash.toLowerCase() === currentTransfer.source.txHash.toLowerCase(),\n );\n\n if (transferUnstake?.payoutTxHash) {\n const updatedTransfer: TargetPendingTransfer = {\n ...currentTransfer,\n status: 'target-pending',\n target: {\n confirmationCount: 0,\n requiredConfirmationCount: 1,\n startedAtMs: Date.now(),\n txHash: transferUnstake.payoutTxHash,\n },\n };\n\n return updatedTransfer;\n }\n\n return currentTransfer;\n } catch {\n return currentTransfer;\n }\n }\n default: {\n // target-pending\n if (!currentTransfer.target?.txHash) {\n // Fail if no target tx hash, should not reach here\n return {\n ...currentTransfer,\n errorCode: ErrorCode.UNKNOWN,\n errorReason: 'Missing target transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n bitcoinFunctions.getTransaction(currentTransfer.target.txHash),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const targetConfirmationCount = receiptResult.value.confirmations;\n\n if (targetConfirmationCount < currentTransfer.target.requiredConfirmationCount) {\n return {\n ...currentTransfer,\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n } satisfies TargetPendingTransfer;\n }\n\n // Completed\n return {\n ...currentTransfer,\n completedAtMs: Date.now(),\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n status: 'completed',\n } satisfies CompletedTransfer;\n } catch {\n return currentTransfer;\n }\n }\n }\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"track-transfer.js","names":[],"sources":["../../../../../src/transfer-service/lombard/btcb-to-btc/_handlers/track-transfer.ts"],"sourcesContent":["import { isHash } from 'viem';\nimport { ErrorCode } from '../../../../errors';\nimport type { BitcoinFunctions } from '../../../../types/bitcoin';\nimport type { TransferService } from '../../../../types/service';\nimport type {\n CompletedTransfer,\n FailedTransfer,\n SourceCompletedTransfer,\n SourcePendingTransfer,\n TargetPendingTransfer,\n Transfer,\n} from '../../../../types/transfer';\nimport type { BtcbToBtcConfig } from '../../types';\nimport { awaitOrAbort, getEvmClientForChain, waitForTimeoutOrAbort } from '../../../_utils';\nimport { getUnstakesByAddress } from '@lombard.finance/sdk';\n\n// Polling interval for BTC redemptions (60 seconds - BTC confirmations are slow)\nconst REDEMPTION_POLLING_INTERVAL_MS = 1000 * 60;\n\nexport interface TrackTransferFactoryOptions {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n}\n\nexport function trackTransferFactory({\n bitcoinFunctions,\n config,\n}: TrackTransferFactoryOptions): TransferService['trackTransfer'] {\n return ({ transfer, updateListener }) => {\n const ac = new AbortController();\n const cancel = () => {\n ac.abort();\n };\n\n if (transfer.status === 'completed' || transfer.status === 'failed') {\n return {\n cancel,\n result: Promise.resolve(transfer),\n };\n }\n\n const executeTracking = async (): Promise<Transfer> => {\n let currentTransfer: Transfer = structuredClone(transfer);\n\n const sourceClient = getEvmClientForChain({ chain: transfer.sourceChain });\n\n while (!ac.signal.aborted) {\n const statusBeforeTracking = currentTransfer.status;\n\n switch (currentTransfer.status) {\n case 'source-pending': {\n currentTransfer = await _trackSourceTx({\n currentTransfer,\n signal: ac.signal,\n sourceClient,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n case 'source-completed':\n case 'target-pending': {\n currentTransfer = await _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal: ac.signal,\n });\n\n if (!ac.signal.aborted) {\n updateListener(currentTransfer);\n }\n\n break;\n }\n default: {\n return currentTransfer;\n }\n }\n\n const hasStatusChanged = currentTransfer.status !== statusBeforeTracking;\n\n if (!hasStatusChanged) {\n await waitForTimeoutOrAbort({ timeoutMs: REDEMPTION_POLLING_INTERVAL_MS, signal: ac.signal });\n }\n }\n\n return currentTransfer;\n };\n\n return {\n cancel,\n result: executeTracking(),\n };\n };\n}\n\nexport interface TrackBaseParams {\n currentTransfer: Transfer;\n signal: AbortSignal;\n}\n\nexport interface TrackSourceTxParams extends TrackBaseParams {\n currentTransfer: SourcePendingTransfer;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n}\n\nexport interface TrackTargetTxParams extends TrackBaseParams {\n bitcoinFunctions: BitcoinFunctions;\n config: BtcbToBtcConfig;\n currentTransfer: SourceCompletedTransfer | TargetPendingTransfer;\n}\n\nexport async function _trackSourceTx({\n currentTransfer,\n signal,\n sourceClient,\n}: TrackSourceTxParams): Promise<SourcePendingTransfer | SourceCompletedTransfer | FailedTransfer> {\n const { confirmationCount, requiredConfirmationCount, txHash } = currentTransfer.source;\n\n if (!isHash(txHash)) {\n const failedTransfer: FailedTransfer = {\n ...currentTransfer,\n errorCode: ErrorCode.INVALID_PARAMS,\n errorReason: 'Invalid source transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n };\n\n return failedTransfer;\n }\n\n if (confirmationCount >= requiredConfirmationCount) {\n // Move to next stage\n\n const updatedTransfer: SourceCompletedTransfer = {\n ...currentTransfer,\n status: 'source-completed',\n };\n\n return updatedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n sourceClient.waitForTransactionReceipt({\n hash: txHash,\n }),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transactionReceipt = receiptResult.value;\n\n if (transactionReceipt.status === 'reverted') {\n return {\n ...currentTransfer,\n errorCode: ErrorCode.TRANSACTION_REVERTED,\n errorReason: 'Source transaction was reverted',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n const newConfirmationCountResult = await awaitOrAbort(\n sourceClient\n .getTransactionConfirmations({\n transactionReceipt,\n })\n .then((count) => Math.max(Number(count), 0))\n .catch(() => Math.max(confirmationCount, 0)),\n signal,\n );\n\n if (newConfirmationCountResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const newConfirmationCount = newConfirmationCountResult.value;\n\n // If still not enough confirmations, update the count\n if (newConfirmationCount < requiredConfirmationCount) {\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n } satisfies SourcePendingTransfer;\n }\n\n // Move to next stage\n return {\n ...currentTransfer,\n source: {\n ...currentTransfer.source,\n confirmationCount: newConfirmationCount,\n },\n status: 'source-completed',\n } satisfies SourceCompletedTransfer;\n } catch {\n return currentTransfer;\n }\n}\n\nexport async function _trackTargetTx({\n bitcoinFunctions,\n config,\n currentTransfer,\n signal,\n}: TrackTargetTxParams): Promise<SourceCompletedTransfer | TargetPendingTransfer | FailedTransfer | CompletedTransfer> {\n switch (currentTransfer.status) {\n case 'source-completed': {\n try {\n const unstakesResult = await awaitOrAbort(\n getUnstakesByAddress({ address: currentTransfer.fromAddress, env: config.env }),\n signal,\n );\n\n if (unstakesResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const transferUnstake = unstakesResult.value.find(\n (d) => d.txHash.toLowerCase() === currentTransfer.source.txHash.toLowerCase(),\n );\n\n if (transferUnstake?.payoutTxHash) {\n const updatedTransfer: TargetPendingTransfer = {\n ...currentTransfer,\n status: 'target-pending',\n target: {\n confirmationCount: 0,\n requiredConfirmationCount: 1,\n startedAtMs: Date.now(),\n txHash: transferUnstake.payoutTxHash,\n },\n };\n\n return updatedTransfer;\n }\n\n return currentTransfer;\n } catch {\n return currentTransfer;\n }\n }\n default: {\n // target-pending\n if (!currentTransfer.target?.txHash) {\n // Fail if no target tx hash, should not reach here\n return {\n ...currentTransfer,\n errorCode: ErrorCode.UNKNOWN,\n errorReason: 'Missing target transaction hash',\n failedAtMs: Date.now(),\n status: 'failed',\n } satisfies FailedTransfer;\n }\n\n try {\n const receiptResult = await awaitOrAbort(\n bitcoinFunctions.getTransaction(currentTransfer.target.txHash),\n signal,\n );\n\n if (receiptResult.status === 'aborted') {\n return currentTransfer;\n }\n\n const targetConfirmationCount = receiptResult.value.confirmations;\n\n if (targetConfirmationCount < currentTransfer.target.requiredConfirmationCount) {\n return {\n ...currentTransfer,\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n } satisfies TargetPendingTransfer;\n }\n\n // Completed\n return {\n ...currentTransfer,\n completedAtMs: Date.now(),\n target: {\n ...currentTransfer.target,\n confirmationCount: targetConfirmationCount,\n },\n status: 'completed',\n } satisfies CompletedTransfer;\n } catch {\n return currentTransfer;\n }\n }\n }\n}\n"],"mappings":"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"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
require(`../../../_virtual/_rolldown/runtime.cjs`);const e=require(`../../../errors.cjs`),t=require(`../../../utils/caip.cjs`),n=require(`../../../_utils/chain.cjs`),r=require(`../../_utils.cjs`),i=require(`../../_evm-gas.cjs`),a=require(`../_api.cjs`),o=require(`../_utils.cjs`),s=require(`../_type-guards.cjs`),c=require(`../_solana-utils.cjs`);let l=require(`viem`),u=require(`@solana/kit`);const d=1000000n;function f(t){return async(r,i)=>{if(n.isEvmNamespace(r.sourceChain.chainId))return await p(r,i,t);if(n.isSolanaNamespace(r.sourceChain.chainId))return await m(r,i,t);throw new e.InvalidParamsError(e.ErrorReason.INVALID_PARAMS,`Unsupported source chain namespace for estimateNativeFee: ${r.sourceChain.chainId}`)}}async function p(n,c,{apiOptions:u,appId:d}){if(!(0,l.isAddress)(n.fromAddress))throw new e.InvalidParamsError(e.ErrorReason.INVALID_PARAMS,`Invalid fromAddress: ${n.fromAddress}`);let f=r.getEvmClientForChain({chain:n.sourceChain}),p=o.assetToAddressString(n.assetIn,n.sourceChain.chainId);if(!(0,l.isAddress)(p))throw new e.SdkError(e.ErrorReason.INVALID_PARAMS,e.ErrorCode.INVALID_PARAMS,{details:`assetIn address is not a valid EVM address. Can not call estimateGas.`});let m=o.isTokenAddressNative(p),g=n.sourceChain.chainId.toLowerCase()!==n.targetChain.chainId.toLowerCase(),_=0n;if(!m){let r,{address:i}=await a.markrGetSpenderAddress(u,{chainId:t.caip2ToEip155ChainId(n.sourceChain.chainId),crossChainSwap:g,quoteId:n.id});try{r=await f.readContract({address:p,abi:l.erc20Abi,functionName:`allowance`,args:[n.fromAddress,i]})}catch(t){throw new e.SdkError(`Error during allowance check`,e.ErrorCode.VIEM_ERROR,{cause:t,details:`Failed to read ERC20 allowance for Markr spender.`})}if(r<
|
|
1
|
+
require(`../../../_virtual/_rolldown/runtime.cjs`);const e=require(`../../../errors.cjs`),t=require(`../../../utils/caip.cjs`),n=require(`../../../_utils/chain.cjs`),r=require(`../../_utils.cjs`),i=require(`../../_evm-gas.cjs`),a=require(`../_api.cjs`),o=require(`../_utils.cjs`),s=require(`../_type-guards.cjs`),c=require(`../_solana-utils.cjs`);let l=require(`viem`),u=require(`@solana/kit`);const d=1000000n;function f(t){return async(r,i)=>{if(n.isEvmNamespace(r.sourceChain.chainId))return await p(r,i,t);if(n.isSolanaNamespace(r.sourceChain.chainId))return await m(r,i,t);throw new e.InvalidParamsError(e.ErrorReason.INVALID_PARAMS,`Unsupported source chain namespace for estimateNativeFee: ${r.sourceChain.chainId}`)}}async function p(n,c,{apiOptions:u,appId:d}){if(!(0,l.isAddress)(n.fromAddress))throw new e.InvalidParamsError(e.ErrorReason.INVALID_PARAMS,`Invalid fromAddress: ${n.fromAddress}`);let f=r.getEvmClientForChain({chain:n.sourceChain}),p=o.assetToAddressString(n.assetIn,n.sourceChain.chainId);if(!(0,l.isAddress)(p))throw new e.SdkError(e.ErrorReason.INVALID_PARAMS,e.ErrorCode.INVALID_PARAMS,{details:`assetIn address is not a valid EVM address. Can not call estimateGas.`});let m=o.isTokenAddressNative(p),g=n.sourceChain.chainId.toLowerCase()!==n.targetChain.chainId.toLowerCase(),_=n.amountIn+o.getAdditiveSourceAssetFeeAmount(n),v=0n;if(!m){let r,{address:i}=await a.markrGetSpenderAddress(u,{chainId:t.caip2ToEip155ChainId(n.sourceChain.chainId),crossChainSwap:g,quoteId:n.id});try{r=await f.readContract({address:p,abi:l.erc20Abi,functionName:`allowance`,args:[n.fromAddress,i]})}catch(t){throw new e.SdkError(`Error during allowance check`,e.ErrorCode.VIEM_ERROR,{cause:t,details:`Failed to read ERC20 allowance for Markr spender.`})}if(r<_)try{v=await f.estimateGas({account:n.fromAddress,to:p,data:(0,l.encodeFunctionData)({abi:l.erc20Abi,functionName:`approve`,args:[i,_]}),value:0n})}catch(t){throw new e.SdkError(`Error during gas estimation`,e.ErrorCode.VIEM_ERROR,{cause:t,details:`Failed to estimate gas for ERC20 approval transaction.`})}}let y=v>0n,b=o.calculateMarkrMinimumAmountOut({amountOut:n.amountOut,assetOut:n.assetOut,slippageBps:n.slippageBps}),x=await a.markrSwap(u,{amountIn:n.amountIn.toString(),appId:d,minAmountOut:b.toString(),tokenIn:p,tokenOut:o.assetToAddressString(n.assetOut,n.targetChain.chainId),uuid:n.id});if(!s.isEvmSwapResponse(x))throw new e.SdkError(e.ErrorReason.CHAIN_NOT_SUPPORTED,e.ErrorCode.INVALID_PARAMS,{details:`Received non-EVM swap response from Markr. Expected EVM transaction data for gas estimation, but got response with chainType ${x.chainType}.`});let S=0n;S=y?n.gasEstimate?r.applyFeeUnitsBpsMargin(n.gasEstimate,c?.feeUnitsMarginBps):r.applyFeeUnitsBpsMargin(n.fees.filter(e=>e.type===`gas`&&e.chainId===n.sourceChain.chainId).reduce((e,t)=>e+t.amount,0n)||700000n,c?.feeUnitsMarginBps):await h({crossChain:g,feeUnitsMarginBps:c?.feeUnitsMarginBps,fromAddress:n.fromAddress,sourceClient:f,swap:x});let C=await i.estimateEvmFeesPerGas(f,n.sourceChain,c?.overrides?.feeRateTier),w=c?.overrides?.maxFeePerGas??C.maxFeePerGas,T=c?.overrides?.maxPriorityFeePerGas??C.maxPriorityFeePerGas,E=(S+r.applyFeeUnitsBpsMargin(v,c?.feeUnitsMarginBps))*w;return{asset:n.sourceChain.networkToken,totalFee:E,totalUpfrontFee:E,meta:{approvalFee:y?r.applyFeeUnitsBpsMargin(v,c?.feeUnitsMarginBps)*w:void 0,maxFeePerGas:w,maxPriorityFeePerGas:T}}}async function m(t,n,{apiOptions:i,appId:l}){let d=o.assetToAddressString(t.assetIn,t.sourceChain.chainId);if(!(0,u.isAddress)(d))throw new e.SdkError(e.ErrorReason.INVALID_PARAMS,e.ErrorCode.INVALID_PARAMS,{details:`assetIn address is not a valid Solana address. Can not call estimateGas.`});let f=o.calculateMarkrMinimumAmountOut({amountOut:t.amountOut,assetOut:t.assetOut,slippageBps:t.slippageBps}),p=await a.markrSwap(i,{amountIn:t.amountIn.toString(),appId:l,minAmountOut:f.toString(),tokenIn:d,tokenOut:o.assetToAddressString(t.assetOut,t.targetChain.chainId),userPublicKey:t.fromAddress,uuid:t.id});if(!s.isSolanaSwapResponse(p))throw new e.SdkError(e.ErrorReason.CHAIN_NOT_SUPPORTED,e.ErrorCode.INVALID_PARAMS,{details:`Received non-Solana swap response from Markr. Expected Solana transaction data for fee estimation.`});let m=r.getSolanaRpcForChain({chain:t.sourceChain}),h=await c.refreshSolanaSwapTransactionBlockhash(p.swapTransaction,m),b=c.decodeSolanaTransactionBase64(h),x=b.messageBytes,S=v(b)*5000n,{computeUnitLimit:C,computeUnitPriceMicroLamports:w}=g(x),T=_(C,w),E=S+T,D=r.applyFeeUnitsBpsMargin(E,n?.feeUnitsMarginBps),O=(await m.simulateTransaction(h,{encoding:`base64`,sigVerify:!1,replaceRecentBlockhash:!0}).send()).value.unitsConsumed??0n,k=await y({fromAddress:t.fromAddress,rpc:m,transaction:b}),A=D+k.rentFee;return{asset:t.sourceChain.networkToken,totalFee:A,totalUpfrontFee:A,refundable:k.refundable,meta:{baseFee:S,priorityFee:T,transactionFee:E,transactionFeeWithMargin:D,computeUnitLimit:C,computeUnitPriceMicroLamports:w,unitsConsumed:O,createdAccounts:k.createdAccounts,rentFee:k.rentFee,rentFeeNet:k.rentFeeNet}}}async function h({crossChain:t,feeUnitsMarginBps:n,fromAddress:i,sourceClient:a,swap:s}){try{return r.applyFeeUnitsBpsMargin(await a.estimateGas({account:i,to:s.to,data:s.data,value:s.value}),n)}catch(n){let r=`Failed to estimate gas for Markr swap transaction.`;try{let e=o.decodeMarkrRevertError(await o.getMarkrSwapWrapperAbi(t),n);e&&(r=`${r} Markr revert: ${e}.`)}catch{}throw new e.SdkError(`Error during gas estimation`,e.ErrorCode.VIEM_ERROR,{cause:n,details:r})}}function g(e){let t=(0,u.getCompiledTransactionMessageDecoder)().decode(e),n=0n,r=0n;for(let e of t.instructions){if(t.staticAccounts[e.programAddressIndex]!==`ComputeBudget111111111111111111111111111111`)continue;let i=e.data;if(!i||i.length<1)continue;let a=i[0],o=new DataView(i.buffer,i.byteOffset,i.byteLength);if(a===2&&i.length>=5){n=BigInt(o.getUint32(1,!0));continue}a===3&&i.length>=9&&(r=o.getBigUint64(1,!0))}return{computeUnitLimit:n,computeUnitPriceMicroLamports:r}}function _(e,t){return e<=0n||t<=0n?0n:(e*t+(d-1n))/d}function v(e){let t=e.signatures;return Array.isArray(t)?BigInt(t.length):t&&typeof t==`object`?BigInt(Object.keys(t).length):0n}async function y({fromAddress:e,rpc:t,transaction:n}){let r=(0,u.getCompiledTransactionMessageDecoder)().decode(n.messageBytes),i=await D(r,t),a=new Map,o=new Map,s;for(let e of r.instructions){let n=i[e.programAddressIndex];if(n===`11111111111111111111111111111111`){let t=O(i,e,1),n=k(e.data);t&&n>0n&&a.set(t,{address:t,kind:`system`,lamports:n});continue}if(n===`ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL`){let n=e.data?.[0];if(!(n===void 0||n===0||n===1))continue;let r=O(i,e,1);if(!r||!(0,u.isAddress)(r)||(await t.getAccountInfo(r).send()).value)continue;s===void 0&&(s=await t.getMinimumBalanceForRentExemption(165n).send()),a.set(r,{address:r,kind:`ata`,lamports:s});continue}if(!(n===`TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA`||n===`TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb`)||e.data?.[0]!==9)continue;let r=O(i,e,0),c=O(i,e,1);r&&c&&o.set(r,c)}let c=(0,u.isAddress)(e)?e:void 0,l=Array.from(a.values()).map(e=>{let t=o.get(e.address),n=c!==void 0&&t===c;return{...e,refunded:n}}),d=l.reduce((e,t)=>e+t.lamports,0n),f=l.reduce((e,t)=>e+(t.refunded?t.lamports:0n),0n);return{createdAccounts:l,rentFee:d,rentFeeNet:d-f,refundable:f}}function b(e){return!!e&&typeof e==`object`}function x(e,t){if(b(e))return Reflect.get(e,t)}function S(e){if(e instanceof Uint8Array)return Array.from(e);if(Array.isArray(e)&&e.every(e=>Number.isInteger(e)&&e>=0))return e}function C(e){let t=x(e,`addressTableLookups`);if(!Array.isArray(t))return[];let n=[];for(let e of t){let t=x(e,`accountKey`),r=S(x(e,`writableIndexes`)),i=S(x(e,`readonlyIndexes`));typeof t!=`string`||!r||!i||n.push({accountKey:t,writableIndexes:r,readonlyIndexes:i})}return n}function w(e){let t=x(e,`value`);if(!b(t))return[];let n=x(t,`addresses`);return Array.isArray(n)?n.filter(e=>typeof e==`string`):[]}async function T(e,t){let n=[],r=[],i=C(e);for(let e of i){if(!(0,u.isAddress)(e.accountKey))continue;let i=await E(t,e.accountKey);for(let t of e.writableIndexes){let e=i[t];e&&n.push(e)}for(let t of e.readonlyIndexes){let e=i[t];e&&r.push(e)}}return{writableAddresses:n,readonlyAddresses:r}}async function E(e,t){let n=x(e,`getAddressLookupTable`);if(typeof n!=`function`)return[];let r=Reflect.apply(n,e,[t]),i=x(r,`send`);return typeof i==`function`?w(await Reflect.apply(i,r,[])):[]}async function D(e,t){let{writableAddresses:n,readonlyAddresses:r}=await T(e,t);return[...e.staticAccounts,...n,...r]}function O(e,t,n){let r=t.accountIndices?.[n];if(r!==void 0)return e[r]}function k(e){if(!e||e.length<12)return 0n;let t=new DataView(e.buffer,e.byteOffset,e.byteLength),n=t.getUint32(0,!0);if(n===0&&e.length>=12)return t.getBigUint64(4,!0);if(n===3&&e.length>=44){let n=Number(t.getBigUint64(36,!0));if(!Number.isSafeInteger(n)||n<0)return 0n;let r=44+n;return r+8>e.length?0n:new DataView(e.buffer,e.byteOffset+r,8).getBigUint64(0,!0)}return 0n}exports._estimateGasFromSwapResponse=h,exports.estimateNativeFeeFactory=f;
|
|
2
2
|
//# sourceMappingURL=estimate-native-fee.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"estimate-native-fee.cjs","names":["isEvmNamespace","isSolanaNamespace","InvalidParamsError","ErrorReason","getEvmClientForChain","assetToAddressString","SdkError","ErrorCode","isTokenAddressNative","markrGetSpenderAddress","caip2ToEip155ChainId","erc20Abi","calculateMarkrMinimumAmountOut","markrSwap","isEvmSwapResponse","applyFeeUnitsBpsMargin","estimateEvmFeesPerGas","isSolanaSwapResponse","getSolanaRpcForChain","refreshSolanaSwapTransactionBlockhash","decodeSolanaTransactionBase64","decodeMarkrRevertError","getMarkrSwapWrapperAbi"],"sources":["../../../../src/transfer-service/markr/_handlers/estimate-native-fee.ts"],"sourcesContent":["import {\n type Address as SolanaAddress,\n type CompiledTransactionMessage,\n getCompiledTransactionMessageDecoder,\n isAddress as isSolanaAddress,\n type Rpc,\n type SolanaRpcApi,\n type Transaction,\n type TransactionMessageBytes,\n} from '@solana/kit';\nimport { encodeFunctionData, erc20Abi, type Address as EvmAddress, isAddress as isEvmAddress } from 'viem';\nimport { ErrorCode, ErrorReason, InvalidParamsError, SdkError } from '../../../errors';\nimport type { EstimateNativeFeeOptions, NativeFeeEstimate, TransferService } from '../../../types/service';\nimport { applyFeeUnitsBpsMargin, getEvmClientForChain, getSolanaRpcForChain } from '../../_utils';\nimport { estimateEvmFeesPerGas } from '../../_evm-gas';\nimport { markrGetSpenderAddress, markrSwap, type ApiOptions } from '../_api';\nimport {\n assetToAddressString,\n calculateMarkrMinimumAmountOut,\n decodeMarkrRevertError,\n getMarkrSwapWrapperAbi,\n isTokenAddressNative,\n} from '../_utils';\nimport type { WrappedSwapTransactionResponse } from '../_schema';\nimport { isEvmSwapResponse, isSolanaSwapResponse } from '../_type-guards';\nimport { caip2ToEip155ChainId } from '../../../utils/caip';\nimport type { Quote } from '../../../types/quote';\nimport { isEvmNamespace, isSolanaNamespace } from '../../../_utils/chain';\nimport { decodeSolanaTransactionBase64, refreshSolanaSwapTransactionBlockhash } from '../_solana-utils';\n\n/**\n * This is just a fallback value used in the case that an\n * allowance approval is needed, and Markr didn't return us any gas estimate data.\n *\n * Just based on my review of some quotes, I think this is a reasonable estimate\n * left on the higher end.\n *\n * Assume this could need to change.\n */\nconst EVM_SWAP_FALLBACK_GAS_ESTIMATE = 700_000n;\n\n/**\n * Compute Budget program address.\n *\n * Used to detect `SetComputeUnitPrice` instructions when deriving\n * the Solana priority fee component from transaction message bytes.\n */\nconst SOLANA_COMPUTE_BUDGET_PROGRAM = 'ComputeBudget111111111111111111111111111111';\n\n/**\n * System Program address.\n *\n * Used to detect account-creation instructions and extract lamports\n * funded upfront for newly created system accounts.\n */\nconst SOLANA_SYSTEM_PROGRAM = '11111111111111111111111111111111';\n\n/**\n * Associated Token Account (ATA) program address.\n *\n * Used to detect ATA create instructions so the estimator can include\n * rent-exempt funding when the ATA does not already exist on-chain.\n */\nconst SOLANA_ASSOCIATED_TOKEN_PROGRAM = 'ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL';\n\n/**\n * Legacy SPL Token program address.\n *\n * Used to detect close-account instructions that may refund lamports\n * from temporary token accounts back to the sender.\n */\nconst SOLANA_SPL_TOKEN_PROGRAM = 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA';\n\n/**\n * SPL Token-2022 program address.\n *\n * Used alongside the legacy token program for close-account detection\n * when computing expected refundable lamports.\n */\nconst SOLANA_SPL_TOKEN_2022_PROGRAM = 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb';\n\n/**\n * SPL token account size for rent-exemption calculations.\n *\n * Used with `getMinimumBalanceForRentExemption` to estimate ATA\n * creation funding requirements.\n */\nconst SPL_TOKEN_ACCOUNT_SIZE_BYTES = 165n;\nconst SOLANA_BASE_FEE_PER_SIGNATURE_LAMPORTS = 5_000n;\nconst MICRO_LAMPORTS_PER_LAMPORT = 1_000_000n;\n\nexport interface EstimateNativeFeeFactoryConfig {\n apiOptions: ApiOptions;\n appId: string;\n}\n\ninterface CreatedRentAccount {\n address: string;\n kind: 'ata' | 'system';\n lamports: bigint;\n refunded: boolean;\n}\n\ninterface SolanaRentEstimate {\n createdAccounts: CreatedRentAccount[];\n rentFee: bigint;\n rentFeeNet: bigint;\n refundable: bigint;\n}\n\ninterface CompiledMessageAddressTableLookup {\n accountKey: string;\n writableIndexes: readonly number[];\n readonlyIndexes: readonly number[];\n}\n\ninterface LookupTableAddresses {\n writableAddresses: string[];\n readonlyAddresses: string[];\n}\n\nexport function estimateNativeFeeFactory(config: EstimateNativeFeeFactoryConfig): TransferService['estimateNativeFee'] {\n return async (quote, options) => {\n // Either the source chain is EVM or Solana, we need to call different functions that\n // calculate the native fee depending on the chain kind.\n\n if (isEvmNamespace(quote.sourceChain.chainId)) {\n return await _estimateNativeFeeEvm(quote, options, config);\n }\n\n if (isSolanaNamespace(quote.sourceChain.chainId)) {\n return await _estimateNativeFeeSolana(quote, options, config);\n }\n\n throw new InvalidParamsError(\n ErrorReason.INVALID_PARAMS,\n `Unsupported source chain namespace for estimateNativeFee: ${quote.sourceChain.chainId}`,\n );\n };\n}\n\n/** @internal */\nexport async function _estimateNativeFeeEvm(\n quote: Quote,\n options: EstimateNativeFeeOptions | undefined,\n { apiOptions, appId }: EstimateNativeFeeFactoryConfig,\n): Promise<NativeFeeEstimate> {\n if (!isEvmAddress(quote.fromAddress)) {\n throw new InvalidParamsError(ErrorReason.INVALID_PARAMS, `Invalid fromAddress: ${quote.fromAddress}`);\n }\n\n const sourceClient = getEvmClientForChain({ chain: quote.sourceChain });\n\n const assetInAddressString = assetToAddressString(quote.assetIn, quote.sourceChain.chainId);\n\n if (!isEvmAddress(assetInAddressString)) {\n throw new SdkError(ErrorReason.INVALID_PARAMS, ErrorCode.INVALID_PARAMS, {\n details: `assetIn address is not a valid EVM address. Can not call estimateGas.`,\n });\n }\n\n const isAssetInNative = isTokenAddressNative(assetInAddressString);\n const isCrossChainSwap = quote.sourceChain.chainId.toLowerCase() !== quote.targetChain.chainId.toLowerCase();\n\n let allowanceApprovalGas = 0n;\n if (!isAssetInNative) {\n // Check if approval is needed, and if so, calculate the gas cost for the approval.\n let allowance: bigint;\n\n const { address: spenderAddress } = await markrGetSpenderAddress(apiOptions, {\n chainId: caip2ToEip155ChainId(quote.sourceChain.chainId),\n crossChainSwap: isCrossChainSwap,\n quoteId: quote.id,\n });\n\n try {\n allowance = await sourceClient.readContract({\n address: assetInAddressString,\n abi: erc20Abi,\n functionName: 'allowance',\n args: [quote.fromAddress, spenderAddress],\n });\n } catch (error) {\n throw new SdkError('Error during allowance check', ErrorCode.VIEM_ERROR, {\n cause: error,\n details: 'Failed to read ERC20 allowance for Markr spender.',\n });\n }\n\n const approvalNeeded = allowance < quote.amountIn;\n\n if (approvalNeeded) {\n try {\n allowanceApprovalGas = await sourceClient.estimateGas({\n account: quote.fromAddress,\n to: assetInAddressString,\n data: encodeFunctionData({\n abi: erc20Abi,\n functionName: 'approve',\n args: [spenderAddress, quote.amountIn],\n }),\n value: 0n,\n });\n } catch (error) {\n throw new SdkError('Error during gas estimation', ErrorCode.VIEM_ERROR, {\n cause: error,\n details: 'Failed to estimate gas for ERC20 approval transaction.',\n });\n }\n }\n }\n\n const allowanceApprovalIsNeeded = allowanceApprovalGas > 0n;\n\n const minAmountOut = calculateMarkrMinimumAmountOut({\n amountOut: quote.amountOut,\n assetOut: quote.assetOut,\n slippageBps: quote.slippageBps,\n });\n\n const swap = await markrSwap(apiOptions, {\n amountIn: quote.amountIn.toString(),\n appId,\n minAmountOut: minAmountOut.toString(),\n tokenIn: assetInAddressString,\n tokenOut: assetToAddressString(quote.assetOut, quote.targetChain.chainId),\n uuid: quote.id,\n });\n\n if (!isEvmSwapResponse(swap)) {\n // Should hopefully never happen.\n throw new SdkError(ErrorReason.CHAIN_NOT_SUPPORTED, ErrorCode.INVALID_PARAMS, {\n details: `Received non-EVM swap response from Markr. Expected EVM transaction data for gas estimation, but got response with chainType ${swap.chainType}.`,\n });\n }\n\n let gasWithMargin = 0n;\n\n // If an approval is needed, we can't call `eth_estimateGas` on the\n // swap transaction because no actual allowance approval was performed yet,\n // so a gas estimation would fail due to needing allowance.\n //\n // So if an allowance approval is needed, we fall back to some other logic for\n // estimating the swap tx gas.\n if (allowanceApprovalIsNeeded) {\n // Attempt to use the Markr provided `gasEstimate` first if available.\n // Otherwise we attempt to get the gas from the `fees` component that are applicable\n // to the source chain. If neither of those are available, we fall back to a hardcoded value.\n if (quote.gasEstimate) {\n gasWithMargin = applyFeeUnitsBpsMargin(quote.gasEstimate, options?.feeUnitsMarginBps);\n } else {\n const sourceGasFee = quote.fees\n .filter((fee) => fee.type === 'gas' && fee.chainId === quote.sourceChain.chainId)\n .reduce((acc, fee) => acc + fee.amount, 0n);\n\n gasWithMargin = applyFeeUnitsBpsMargin(\n sourceGasFee || EVM_SWAP_FALLBACK_GAS_ESTIMATE,\n options?.feeUnitsMarginBps,\n );\n }\n } else {\n gasWithMargin = await _estimateGasFromSwapResponse({\n crossChain: isCrossChainSwap,\n feeUnitsMarginBps: options?.feeUnitsMarginBps,\n fromAddress: quote.fromAddress,\n sourceClient,\n swap,\n });\n }\n\n const fees = await estimateEvmFeesPerGas(sourceClient, quote.sourceChain, options?.overrides?.feeRateTier);\n\n const maxFeePerGas = options?.overrides?.maxFeePerGas ?? fees.maxFeePerGas;\n const maxPriorityFeePerGas = options?.overrides?.maxPriorityFeePerGas ?? fees.maxPriorityFeePerGas;\n\n const totalFee =\n (gasWithMargin + applyFeeUnitsBpsMargin(allowanceApprovalGas, options?.feeUnitsMarginBps)) * maxFeePerGas;\n\n return {\n asset: quote.sourceChain.networkToken,\n totalFee,\n totalUpfrontFee: totalFee,\n meta: {\n approvalFee: allowanceApprovalIsNeeded\n ? applyFeeUnitsBpsMargin(allowanceApprovalGas, options?.feeUnitsMarginBps) * maxFeePerGas\n : undefined,\n maxFeePerGas,\n maxPriorityFeePerGas,\n },\n };\n}\n\n/** @internal */\nexport async function _estimateNativeFeeSolana(\n quote: Quote,\n options: EstimateNativeFeeOptions | undefined,\n { apiOptions, appId }: EstimateNativeFeeFactoryConfig,\n): Promise<NativeFeeEstimate> {\n const assetInAddressString = assetToAddressString(quote.assetIn, quote.sourceChain.chainId);\n\n if (!isSolanaAddress(assetInAddressString)) {\n throw new SdkError(ErrorReason.INVALID_PARAMS, ErrorCode.INVALID_PARAMS, {\n details: `assetIn address is not a valid Solana address. Can not call estimateGas.`,\n });\n }\n\n const minAmountOut = calculateMarkrMinimumAmountOut({\n amountOut: quote.amountOut,\n assetOut: quote.assetOut,\n slippageBps: quote.slippageBps,\n });\n\n const swap = await markrSwap(apiOptions, {\n amountIn: quote.amountIn.toString(),\n appId,\n minAmountOut: minAmountOut.toString(),\n tokenIn: assetInAddressString,\n tokenOut: assetToAddressString(quote.assetOut, quote.targetChain.chainId),\n userPublicKey: quote.fromAddress,\n uuid: quote.id,\n });\n\n if (!isSolanaSwapResponse(swap)) {\n // Should hopefully never happen.\n throw new SdkError(ErrorReason.CHAIN_NOT_SUPPORTED, ErrorCode.INVALID_PARAMS, {\n details: 'Received non-Solana swap response from Markr. Expected Solana transaction data for fee estimation.',\n });\n }\n\n const rpc = getSolanaRpcForChain({ chain: quote.sourceChain });\n const refreshedSwapTransactionBase64 = await refreshSolanaSwapTransactionBlockhash(swap.swapTransaction, rpc);\n const decodedTransaction = decodeSolanaTransactionBase64(refreshedSwapTransactionBase64);\n const transactionMessageBytes = decodedTransaction.messageBytes;\n const signatureCount = _getSignatureCount(decodedTransaction);\n const baseFeeLamports = signatureCount * SOLANA_BASE_FEE_PER_SIGNATURE_LAMPORTS;\n const { computeUnitLimit, computeUnitPriceMicroLamports } = _getComputeBudgetPriceAndLimit(transactionMessageBytes);\n const priorityFee = _calculatePriorityFeeLamports(computeUnitLimit, computeUnitPriceMicroLamports);\n const transactionFee = baseFeeLamports + priorityFee;\n const transactionFeeWithMargin = applyFeeUnitsBpsMargin(transactionFee, options?.feeUnitsMarginBps);\n\n const sim = await rpc\n .simulateTransaction(refreshedSwapTransactionBase64, {\n encoding: 'base64',\n sigVerify: false,\n replaceRecentBlockhash: true, // Helps prevent block hash expiration issues during simulation.\n })\n .send();\n\n const unitsConsumed = sim.value.unitsConsumed ?? 0n;\n const rentEstimate = await _estimateRentFeesFromTransaction({\n fromAddress: quote.fromAddress,\n rpc,\n transaction: decodedTransaction,\n });\n const totalLamports = transactionFeeWithMargin + rentEstimate.rentFee;\n\n return {\n asset: quote.sourceChain.networkToken,\n totalFee: totalLamports,\n totalUpfrontFee: totalLamports,\n refundable: rentEstimate.refundable,\n meta: {\n baseFee: baseFeeLamports,\n priorityFee,\n transactionFee,\n transactionFeeWithMargin,\n computeUnitLimit,\n computeUnitPriceMicroLamports,\n unitsConsumed,\n createdAccounts: rentEstimate.createdAccounts,\n rentFee: rentEstimate.rentFee,\n rentFeeNet: rentEstimate.rentFeeNet,\n },\n };\n}\n\n/** @internal */\nexport async function _estimateGasFromSwapResponse({\n crossChain,\n feeUnitsMarginBps,\n fromAddress,\n sourceClient,\n swap,\n}: {\n crossChain: boolean;\n feeUnitsMarginBps?: number;\n fromAddress: EvmAddress;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n swap: WrappedSwapTransactionResponse;\n}): Promise<bigint> {\n try {\n const gasEstimate = await sourceClient.estimateGas({\n account: fromAddress,\n to: swap.to,\n data: swap.data,\n value: swap.value,\n });\n\n return applyFeeUnitsBpsMargin(gasEstimate, feeUnitsMarginBps);\n } catch (err) {\n let details = 'Failed to estimate gas for Markr swap transaction.';\n\n try {\n const markrSwapWrapperAbi = await getMarkrSwapWrapperAbi(crossChain);\n const decodedRevert = decodeMarkrRevertError(markrSwapWrapperAbi, err);\n\n if (decodedRevert) {\n details = `${details} Markr revert: ${decodedRevert}.`;\n }\n } catch {\n // Keep the base details message if revert decoding fails.\n }\n\n throw new SdkError('Error during gas estimation', ErrorCode.VIEM_ERROR, {\n cause: err,\n details,\n });\n }\n}\n\nfunction _getComputeBudgetPriceAndLimit(transactionMessageBytes: TransactionMessageBytes): {\n computeUnitLimit: bigint;\n computeUnitPriceMicroLamports: bigint;\n} {\n const compiledTxMessage = getCompiledTransactionMessageDecoder().decode(transactionMessageBytes);\n let computeUnitLimit = 0n;\n let computeUnitPriceMicroLamports = 0n;\n\n for (const instruction of compiledTxMessage.instructions) {\n const programAddress = compiledTxMessage.staticAccounts[instruction.programAddressIndex];\n if (programAddress !== SOLANA_COMPUTE_BUDGET_PROGRAM) {\n continue;\n }\n\n const data = instruction.data;\n\n if (!data || data.length < 1) {\n continue;\n }\n\n const discriminator = data[0];\n const view = new DataView(data.buffer, data.byteOffset, data.byteLength);\n\n // ComputeBudget::SetComputeUnitLimit(u32)\n if (discriminator === 2 && data.length >= 1 + 4) {\n computeUnitLimit = BigInt(view.getUint32(1, true));\n continue;\n }\n\n // ComputeBudget::SetComputeUnitPrice(u64)\n if (discriminator === 3 && data.length >= 1 + 8) {\n computeUnitPriceMicroLamports = view.getBigUint64(1, true);\n }\n }\n\n return {\n computeUnitLimit,\n computeUnitPriceMicroLamports,\n };\n}\n\nfunction _calculatePriorityFeeLamports(computeUnitLimit: bigint, computeUnitPriceMicroLamports: bigint): bigint {\n if (computeUnitLimit <= 0n || computeUnitPriceMicroLamports <= 0n) {\n return 0n;\n }\n\n return (\n (computeUnitLimit * computeUnitPriceMicroLamports + (MICRO_LAMPORTS_PER_LAMPORT - 1n)) / MICRO_LAMPORTS_PER_LAMPORT\n );\n}\n\nfunction _getSignatureCount(transaction: Transaction): bigint {\n const signatures = (transaction as { signatures?: unknown }).signatures;\n\n if (Array.isArray(signatures)) {\n return BigInt(signatures.length);\n }\n\n if (signatures && typeof signatures === 'object') {\n return BigInt(Object.keys(signatures).length);\n }\n\n return 0n;\n}\n\nasync function _estimateRentFeesFromTransaction({\n fromAddress,\n rpc,\n transaction,\n}: {\n fromAddress: string;\n rpc: Rpc<SolanaRpcApi>;\n transaction: Transaction;\n}): Promise<SolanaRentEstimate> {\n const compiledMessage: CompiledTransactionMessage = getCompiledTransactionMessageDecoder().decode(\n transaction.messageBytes,\n );\n const allAccounts = await _resolveAllTransactionAccounts(compiledMessage, rpc);\n\n const createdAccounts = new Map<string, Omit<CreatedRentAccount, 'refunded'>>();\n const closedAccountDestinationByAddress = new Map<string, string>();\n\n let ataRentLamports: bigint | undefined;\n\n for (const instruction of compiledMessage.instructions) {\n const programAddress = allAccounts[instruction.programAddressIndex];\n\n if (programAddress === SOLANA_SYSTEM_PROGRAM) {\n const createdAccountAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n const lamports = _getSystemCreateAccountLamports(instruction.data);\n if (createdAccountAddress && lamports > 0n) {\n createdAccounts.set(createdAccountAddress, {\n address: createdAccountAddress,\n kind: 'system',\n lamports,\n });\n }\n continue;\n }\n\n if (programAddress === SOLANA_ASSOCIATED_TOKEN_PROGRAM) {\n const discriminator = instruction.data?.[0];\n const isAssociatedTokenCreateInstruction =\n discriminator === undefined || discriminator === 0 || discriminator === 1;\n\n if (!isAssociatedTokenCreateInstruction) {\n continue;\n }\n\n const associatedTokenAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n if (!associatedTokenAddress || !isSolanaAddress(associatedTokenAddress)) {\n continue;\n }\n\n const accountInfo = await rpc.getAccountInfo(associatedTokenAddress).send();\n if (accountInfo.value) {\n continue;\n }\n\n if (ataRentLamports === undefined) {\n ataRentLamports = await rpc.getMinimumBalanceForRentExemption(SPL_TOKEN_ACCOUNT_SIZE_BYTES).send();\n }\n\n createdAccounts.set(associatedTokenAddress, {\n address: associatedTokenAddress,\n kind: 'ata',\n lamports: ataRentLamports,\n });\n\n continue;\n }\n\n const isSplTokenProgram =\n programAddress === SOLANA_SPL_TOKEN_PROGRAM || programAddress === SOLANA_SPL_TOKEN_2022_PROGRAM;\n if (!isSplTokenProgram) {\n continue;\n }\n\n const discriminator = instruction.data?.[0];\n if (discriminator !== 9) {\n continue;\n }\n\n const closedAccountAddress = _getInstructionAccountAddress(allAccounts, instruction, 0);\n const refundDestinationAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n\n if (closedAccountAddress && refundDestinationAddress) {\n closedAccountDestinationByAddress.set(closedAccountAddress, refundDestinationAddress);\n }\n }\n\n const normalizedFromAddress = isSolanaAddress(fromAddress) ? fromAddress : undefined;\n const createdAccountEntries = Array.from(createdAccounts.values()).map((createdAccount) => {\n const refundDestinationAddress = closedAccountDestinationByAddress.get(createdAccount.address);\n const refunded = normalizedFromAddress !== undefined && refundDestinationAddress === normalizedFromAddress;\n\n return {\n ...createdAccount,\n refunded,\n } satisfies CreatedRentAccount;\n });\n\n const rentLamportsUpfront = createdAccountEntries.reduce((acc, account) => acc + account.lamports, 0n);\n const refundedLamports = createdAccountEntries.reduce(\n (acc, account) => acc + (account.refunded ? account.lamports : 0n),\n 0n,\n );\n const rentFeeNet = rentLamportsUpfront - refundedLamports;\n\n return {\n createdAccounts: createdAccountEntries,\n rentFee: rentLamportsUpfront,\n rentFeeNet,\n refundable: refundedLamports,\n };\n}\n\nfunction _isObject(value: unknown): value is object {\n return !!value && typeof value === 'object';\n}\n\nfunction _getObjectProperty(value: unknown, key: string): unknown {\n if (!_isObject(value)) {\n return undefined;\n }\n\n return Reflect.get(value, key);\n}\n\nfunction _toNumberIndexArray(value: unknown): number[] | undefined {\n if (value instanceof Uint8Array) {\n return Array.from(value);\n }\n\n if (Array.isArray(value) && value.every((entry) => Number.isInteger(entry) && entry >= 0)) {\n return value;\n }\n\n return undefined;\n}\n\nfunction _getCompiledMessageAddressTableLookups(\n compiledMessage: CompiledTransactionMessage,\n): CompiledMessageAddressTableLookup[] {\n const addressTableLookupsRaw = _getObjectProperty(compiledMessage, 'addressTableLookups');\n if (!Array.isArray(addressTableLookupsRaw)) {\n return [];\n }\n\n const lookups: CompiledMessageAddressTableLookup[] = [];\n\n for (const lookupRaw of addressTableLookupsRaw) {\n const accountKey = _getObjectProperty(lookupRaw, 'accountKey');\n const writableIndexes = _toNumberIndexArray(_getObjectProperty(lookupRaw, 'writableIndexes'));\n const readonlyIndexes = _toNumberIndexArray(_getObjectProperty(lookupRaw, 'readonlyIndexes'));\n\n if (typeof accountKey !== 'string' || !writableIndexes || !readonlyIndexes) {\n continue;\n }\n\n lookups.push({\n accountKey,\n writableIndexes,\n readonlyIndexes,\n });\n }\n\n return lookups;\n}\n\nfunction _extractLookupTableAddresses(response: unknown): string[] {\n const lookupTableValue = _getObjectProperty(response, 'value');\n if (!_isObject(lookupTableValue)) {\n return [];\n }\n\n const addresses = _getObjectProperty(lookupTableValue, 'addresses');\n if (!Array.isArray(addresses)) {\n return [];\n }\n\n return addresses.filter((entry): entry is string => typeof entry === 'string');\n}\n\nasync function _loadLookupTableAddresses(\n compiledMessage: CompiledTransactionMessage,\n rpc: Rpc<SolanaRpcApi>,\n): Promise<LookupTableAddresses> {\n const writableAddresses: string[] = [];\n const readonlyAddresses: string[] = [];\n const addressTableLookups = _getCompiledMessageAddressTableLookups(compiledMessage);\n\n for (const lookup of addressTableLookups) {\n if (!isSolanaAddress(lookup.accountKey)) {\n continue;\n }\n\n const tableAddresses = await _fetchAddressLookupTableAddresses(rpc, lookup.accountKey);\n\n for (const index of lookup.writableIndexes) {\n const address = tableAddresses[index];\n if (address) {\n writableAddresses.push(address);\n }\n }\n\n for (const index of lookup.readonlyIndexes) {\n const address = tableAddresses[index];\n if (address) {\n readonlyAddresses.push(address);\n }\n }\n }\n\n return {\n writableAddresses,\n readonlyAddresses,\n };\n}\n\nasync function _fetchAddressLookupTableAddresses(rpc: Rpc<SolanaRpcApi>, accountKey: SolanaAddress): Promise<string[]> {\n const getAddressLookupTable = _getObjectProperty(rpc, 'getAddressLookupTable');\n if (typeof getAddressLookupTable !== 'function') {\n return [];\n }\n\n const request = Reflect.apply(getAddressLookupTable, rpc, [accountKey]);\n const send = _getObjectProperty(request, 'send');\n\n if (typeof send !== 'function') {\n return [];\n }\n\n const response = await Reflect.apply(send, request, []);\n return _extractLookupTableAddresses(response);\n}\n\nasync function _resolveAllTransactionAccounts(\n compiledMessage: CompiledTransactionMessage,\n rpc: Rpc<SolanaRpcApi>,\n): Promise<string[]> {\n const { writableAddresses, readonlyAddresses } = await _loadLookupTableAddresses(compiledMessage, rpc);\n\n return [...compiledMessage.staticAccounts, ...writableAddresses, ...readonlyAddresses];\n}\n\n/**\n * Resolves an account address used by a compiled instruction.\n *\n * Compiled instructions reference accounts by index into the transaction\n * message account list. For versioned transactions this includes static\n * accounts followed by writable and readonly addresses loaded from lookup\n * tables. This helper converts an instruction-local `accountIndex` into the\n * corresponding account address.\n *\n * Returns `undefined` when the instruction does not provide an account at the\n * requested index.\n */\nfunction _getInstructionAccountAddress(\n allAccounts: readonly string[],\n instruction: CompiledTransactionMessage['instructions'][number],\n accountIndex: number,\n): string | undefined {\n const messageAccountIndex = instruction.accountIndices?.[accountIndex];\n\n if (messageAccountIndex === undefined) {\n return undefined;\n }\n\n return allAccounts[messageAccountIndex];\n}\n\n/**\n * Extracts lamports from System Program account-creation instruction data.\n *\n * Supported instruction layouts:\n * - `CreateAccount` (`u32 discriminator = 0`): lamports at byte offset `4`.\n * - `CreateAccountWithSeed` (`u32 discriminator = 3`): lamports are located\n * after the variable-length seed field.\n *\n * Returns `0n` when data is missing, malformed, or not one of the supported\n * System Program create-account variants.\n */\nfunction _getSystemCreateAccountLamports(data: CompiledTransactionMessage['instructions'][number]['data']): bigint {\n if (!data || data.length < 12) {\n return 0n;\n }\n\n const view = new DataView(data.buffer, data.byteOffset, data.byteLength);\n const instructionDiscriminator = view.getUint32(0, true);\n\n if (instructionDiscriminator === 0 && data.length >= 12) {\n return view.getBigUint64(4, true);\n }\n\n if (instructionDiscriminator === 3 && data.length >= 44) {\n const seedLenOffset = 36;\n const seedLength = Number(view.getBigUint64(seedLenOffset, true));\n if (!Number.isSafeInteger(seedLength) || seedLength < 0) {\n return 0n;\n }\n\n const lamportsOffset = seedLenOffset + 8 + seedLength;\n if (lamportsOffset + 8 > data.length) {\n return 0n;\n }\n\n return new DataView(data.buffer, data.byteOffset + lamportsOffset, 8).getBigUint64(0, true);\n }\n\n return 0n;\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"estimate-native-fee.cjs","names":["isEvmNamespace","isSolanaNamespace","InvalidParamsError","ErrorReason","getEvmClientForChain","assetToAddressString","SdkError","ErrorCode","isTokenAddressNative","getAdditiveSourceAssetFeeAmount","markrGetSpenderAddress","caip2ToEip155ChainId","erc20Abi","calculateMarkrMinimumAmountOut","markrSwap","isEvmSwapResponse","applyFeeUnitsBpsMargin","estimateEvmFeesPerGas","isSolanaSwapResponse","getSolanaRpcForChain","refreshSolanaSwapTransactionBlockhash","decodeSolanaTransactionBase64","decodeMarkrRevertError","getMarkrSwapWrapperAbi"],"sources":["../../../../src/transfer-service/markr/_handlers/estimate-native-fee.ts"],"sourcesContent":["import {\n type Address as SolanaAddress,\n type CompiledTransactionMessage,\n getCompiledTransactionMessageDecoder,\n isAddress as isSolanaAddress,\n type Rpc,\n type SolanaRpcApi,\n type Transaction,\n type TransactionMessageBytes,\n} from '@solana/kit';\nimport { encodeFunctionData, erc20Abi, type Address as EvmAddress, isAddress as isEvmAddress } from 'viem';\nimport { ErrorCode, ErrorReason, InvalidParamsError, SdkError } from '../../../errors';\nimport type { EstimateNativeFeeOptions, NativeFeeEstimate, TransferService } from '../../../types/service';\nimport { applyFeeUnitsBpsMargin, getEvmClientForChain, getSolanaRpcForChain } from '../../_utils';\nimport { estimateEvmFeesPerGas } from '../../_evm-gas';\nimport { markrGetSpenderAddress, markrSwap, type ApiOptions } from '../_api';\nimport {\n assetToAddressString,\n calculateMarkrMinimumAmountOut,\n decodeMarkrRevertError,\n getAdditiveSourceAssetFeeAmount,\n getMarkrSwapWrapperAbi,\n isTokenAddressNative,\n} from '../_utils';\nimport type { WrappedSwapTransactionResponse } from '../_schema';\nimport { isEvmSwapResponse, isSolanaSwapResponse } from '../_type-guards';\nimport { caip2ToEip155ChainId } from '../../../utils/caip';\nimport type { Quote } from '../../../types/quote';\nimport { isEvmNamespace, isSolanaNamespace } from '../../../_utils/chain';\nimport { decodeSolanaTransactionBase64, refreshSolanaSwapTransactionBlockhash } from '../_solana-utils';\n\n/**\n * This is just a fallback value used in the case that an\n * allowance approval is needed, and Markr didn't return us any gas estimate data.\n *\n * Just based on my review of some quotes, I think this is a reasonable estimate\n * left on the higher end.\n *\n * Assume this could need to change.\n */\nconst EVM_SWAP_FALLBACK_GAS_ESTIMATE = 700_000n;\n\n/**\n * Compute Budget program address.\n *\n * Used to detect `SetComputeUnitPrice` instructions when deriving\n * the Solana priority fee component from transaction message bytes.\n */\nconst SOLANA_COMPUTE_BUDGET_PROGRAM = 'ComputeBudget111111111111111111111111111111';\n\n/**\n * System Program address.\n *\n * Used to detect account-creation instructions and extract lamports\n * funded upfront for newly created system accounts.\n */\nconst SOLANA_SYSTEM_PROGRAM = '11111111111111111111111111111111';\n\n/**\n * Associated Token Account (ATA) program address.\n *\n * Used to detect ATA create instructions so the estimator can include\n * rent-exempt funding when the ATA does not already exist on-chain.\n */\nconst SOLANA_ASSOCIATED_TOKEN_PROGRAM = 'ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL';\n\n/**\n * Legacy SPL Token program address.\n *\n * Used to detect close-account instructions that may refund lamports\n * from temporary token accounts back to the sender.\n */\nconst SOLANA_SPL_TOKEN_PROGRAM = 'TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA';\n\n/**\n * SPL Token-2022 program address.\n *\n * Used alongside the legacy token program for close-account detection\n * when computing expected refundable lamports.\n */\nconst SOLANA_SPL_TOKEN_2022_PROGRAM = 'TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb';\n\n/**\n * SPL token account size for rent-exemption calculations.\n *\n * Used with `getMinimumBalanceForRentExemption` to estimate ATA\n * creation funding requirements.\n */\nconst SPL_TOKEN_ACCOUNT_SIZE_BYTES = 165n;\nconst SOLANA_BASE_FEE_PER_SIGNATURE_LAMPORTS = 5_000n;\nconst MICRO_LAMPORTS_PER_LAMPORT = 1_000_000n;\n\nexport interface EstimateNativeFeeFactoryConfig {\n apiOptions: ApiOptions;\n appId: string;\n}\n\ninterface CreatedRentAccount {\n address: string;\n kind: 'ata' | 'system';\n lamports: bigint;\n refunded: boolean;\n}\n\ninterface SolanaRentEstimate {\n createdAccounts: CreatedRentAccount[];\n rentFee: bigint;\n rentFeeNet: bigint;\n refundable: bigint;\n}\n\ninterface CompiledMessageAddressTableLookup {\n accountKey: string;\n writableIndexes: readonly number[];\n readonlyIndexes: readonly number[];\n}\n\ninterface LookupTableAddresses {\n writableAddresses: string[];\n readonlyAddresses: string[];\n}\n\nexport function estimateNativeFeeFactory(config: EstimateNativeFeeFactoryConfig): TransferService['estimateNativeFee'] {\n return async (quote, options) => {\n // Either the source chain is EVM or Solana, we need to call different functions that\n // calculate the native fee depending on the chain kind.\n\n if (isEvmNamespace(quote.sourceChain.chainId)) {\n return await _estimateNativeFeeEvm(quote, options, config);\n }\n\n if (isSolanaNamespace(quote.sourceChain.chainId)) {\n return await _estimateNativeFeeSolana(quote, options, config);\n }\n\n throw new InvalidParamsError(\n ErrorReason.INVALID_PARAMS,\n `Unsupported source chain namespace for estimateNativeFee: ${quote.sourceChain.chainId}`,\n );\n };\n}\n\n/** @internal */\nexport async function _estimateNativeFeeEvm(\n quote: Quote,\n options: EstimateNativeFeeOptions | undefined,\n { apiOptions, appId }: EstimateNativeFeeFactoryConfig,\n): Promise<NativeFeeEstimate> {\n if (!isEvmAddress(quote.fromAddress)) {\n throw new InvalidParamsError(ErrorReason.INVALID_PARAMS, `Invalid fromAddress: ${quote.fromAddress}`);\n }\n\n const sourceClient = getEvmClientForChain({ chain: quote.sourceChain });\n\n const assetInAddressString = assetToAddressString(quote.assetIn, quote.sourceChain.chainId);\n\n if (!isEvmAddress(assetInAddressString)) {\n throw new SdkError(ErrorReason.INVALID_PARAMS, ErrorCode.INVALID_PARAMS, {\n details: `assetIn address is not a valid EVM address. Can not call estimateGas.`,\n });\n }\n\n const isAssetInNative = isTokenAddressNative(assetInAddressString);\n const isCrossChainSwap = quote.sourceChain.chainId.toLowerCase() !== quote.targetChain.chainId.toLowerCase();\n const allowanceAmount = quote.amountIn + getAdditiveSourceAssetFeeAmount(quote);\n\n let allowanceApprovalGas = 0n;\n if (!isAssetInNative) {\n // Check if approval is needed, and if so, calculate the gas cost for the approval.\n let allowance: bigint;\n\n const { address: spenderAddress } = await markrGetSpenderAddress(apiOptions, {\n chainId: caip2ToEip155ChainId(quote.sourceChain.chainId),\n crossChainSwap: isCrossChainSwap,\n quoteId: quote.id,\n });\n\n try {\n allowance = await sourceClient.readContract({\n address: assetInAddressString,\n abi: erc20Abi,\n functionName: 'allowance',\n args: [quote.fromAddress, spenderAddress],\n });\n } catch (error) {\n throw new SdkError('Error during allowance check', ErrorCode.VIEM_ERROR, {\n cause: error,\n details: 'Failed to read ERC20 allowance for Markr spender.',\n });\n }\n\n const approvalNeeded = allowance < allowanceAmount;\n\n if (approvalNeeded) {\n try {\n allowanceApprovalGas = await sourceClient.estimateGas({\n account: quote.fromAddress,\n to: assetInAddressString,\n data: encodeFunctionData({\n abi: erc20Abi,\n functionName: 'approve',\n args: [spenderAddress, allowanceAmount],\n }),\n value: 0n,\n });\n } catch (error) {\n throw new SdkError('Error during gas estimation', ErrorCode.VIEM_ERROR, {\n cause: error,\n details: 'Failed to estimate gas for ERC20 approval transaction.',\n });\n }\n }\n }\n\n const allowanceApprovalIsNeeded = allowanceApprovalGas > 0n;\n\n const minAmountOut = calculateMarkrMinimumAmountOut({\n amountOut: quote.amountOut,\n assetOut: quote.assetOut,\n slippageBps: quote.slippageBps,\n });\n\n const swap = await markrSwap(apiOptions, {\n amountIn: quote.amountIn.toString(),\n appId,\n minAmountOut: minAmountOut.toString(),\n tokenIn: assetInAddressString,\n tokenOut: assetToAddressString(quote.assetOut, quote.targetChain.chainId),\n uuid: quote.id,\n });\n\n if (!isEvmSwapResponse(swap)) {\n // Should hopefully never happen.\n throw new SdkError(ErrorReason.CHAIN_NOT_SUPPORTED, ErrorCode.INVALID_PARAMS, {\n details: `Received non-EVM swap response from Markr. Expected EVM transaction data for gas estimation, but got response with chainType ${swap.chainType}.`,\n });\n }\n\n let gasWithMargin = 0n;\n\n // If an approval is needed, we can't call `eth_estimateGas` on the\n // swap transaction because no actual allowance approval was performed yet,\n // so a gas estimation would fail due to needing allowance.\n //\n // So if an allowance approval is needed, we fall back to some other logic for\n // estimating the swap tx gas.\n if (allowanceApprovalIsNeeded) {\n // Attempt to use the Markr provided `gasEstimate` first if available.\n // Otherwise we attempt to get the gas from the `fees` component that are applicable\n // to the source chain. If neither of those are available, we fall back to a hardcoded value.\n if (quote.gasEstimate) {\n gasWithMargin = applyFeeUnitsBpsMargin(quote.gasEstimate, options?.feeUnitsMarginBps);\n } else {\n const sourceGasFee = quote.fees\n .filter((fee) => fee.type === 'gas' && fee.chainId === quote.sourceChain.chainId)\n .reduce((acc, fee) => acc + fee.amount, 0n);\n\n gasWithMargin = applyFeeUnitsBpsMargin(\n sourceGasFee || EVM_SWAP_FALLBACK_GAS_ESTIMATE,\n options?.feeUnitsMarginBps,\n );\n }\n } else {\n gasWithMargin = await _estimateGasFromSwapResponse({\n crossChain: isCrossChainSwap,\n feeUnitsMarginBps: options?.feeUnitsMarginBps,\n fromAddress: quote.fromAddress,\n sourceClient,\n swap,\n });\n }\n\n const fees = await estimateEvmFeesPerGas(sourceClient, quote.sourceChain, options?.overrides?.feeRateTier);\n\n const maxFeePerGas = options?.overrides?.maxFeePerGas ?? fees.maxFeePerGas;\n const maxPriorityFeePerGas = options?.overrides?.maxPriorityFeePerGas ?? fees.maxPriorityFeePerGas;\n\n const totalFee =\n (gasWithMargin + applyFeeUnitsBpsMargin(allowanceApprovalGas, options?.feeUnitsMarginBps)) * maxFeePerGas;\n\n return {\n asset: quote.sourceChain.networkToken,\n totalFee,\n totalUpfrontFee: totalFee,\n meta: {\n approvalFee: allowanceApprovalIsNeeded\n ? applyFeeUnitsBpsMargin(allowanceApprovalGas, options?.feeUnitsMarginBps) * maxFeePerGas\n : undefined,\n maxFeePerGas,\n maxPriorityFeePerGas,\n },\n };\n}\n\n/** @internal */\nexport async function _estimateNativeFeeSolana(\n quote: Quote,\n options: EstimateNativeFeeOptions | undefined,\n { apiOptions, appId }: EstimateNativeFeeFactoryConfig,\n): Promise<NativeFeeEstimate> {\n const assetInAddressString = assetToAddressString(quote.assetIn, quote.sourceChain.chainId);\n\n if (!isSolanaAddress(assetInAddressString)) {\n throw new SdkError(ErrorReason.INVALID_PARAMS, ErrorCode.INVALID_PARAMS, {\n details: `assetIn address is not a valid Solana address. Can not call estimateGas.`,\n });\n }\n\n const minAmountOut = calculateMarkrMinimumAmountOut({\n amountOut: quote.amountOut,\n assetOut: quote.assetOut,\n slippageBps: quote.slippageBps,\n });\n\n const swap = await markrSwap(apiOptions, {\n amountIn: quote.amountIn.toString(),\n appId,\n minAmountOut: minAmountOut.toString(),\n tokenIn: assetInAddressString,\n tokenOut: assetToAddressString(quote.assetOut, quote.targetChain.chainId),\n userPublicKey: quote.fromAddress,\n uuid: quote.id,\n });\n\n if (!isSolanaSwapResponse(swap)) {\n // Should hopefully never happen.\n throw new SdkError(ErrorReason.CHAIN_NOT_SUPPORTED, ErrorCode.INVALID_PARAMS, {\n details: 'Received non-Solana swap response from Markr. Expected Solana transaction data for fee estimation.',\n });\n }\n\n const rpc = getSolanaRpcForChain({ chain: quote.sourceChain });\n const refreshedSwapTransactionBase64 = await refreshSolanaSwapTransactionBlockhash(swap.swapTransaction, rpc);\n const decodedTransaction = decodeSolanaTransactionBase64(refreshedSwapTransactionBase64);\n const transactionMessageBytes = decodedTransaction.messageBytes;\n const signatureCount = _getSignatureCount(decodedTransaction);\n const baseFeeLamports = signatureCount * SOLANA_BASE_FEE_PER_SIGNATURE_LAMPORTS;\n const { computeUnitLimit, computeUnitPriceMicroLamports } = _getComputeBudgetPriceAndLimit(transactionMessageBytes);\n const priorityFee = _calculatePriorityFeeLamports(computeUnitLimit, computeUnitPriceMicroLamports);\n const transactionFee = baseFeeLamports + priorityFee;\n const transactionFeeWithMargin = applyFeeUnitsBpsMargin(transactionFee, options?.feeUnitsMarginBps);\n\n const sim = await rpc\n .simulateTransaction(refreshedSwapTransactionBase64, {\n encoding: 'base64',\n sigVerify: false,\n replaceRecentBlockhash: true, // Helps prevent block hash expiration issues during simulation.\n })\n .send();\n\n const unitsConsumed = sim.value.unitsConsumed ?? 0n;\n const rentEstimate = await _estimateRentFeesFromTransaction({\n fromAddress: quote.fromAddress,\n rpc,\n transaction: decodedTransaction,\n });\n const totalLamports = transactionFeeWithMargin + rentEstimate.rentFee;\n\n return {\n asset: quote.sourceChain.networkToken,\n totalFee: totalLamports,\n totalUpfrontFee: totalLamports,\n refundable: rentEstimate.refundable,\n meta: {\n baseFee: baseFeeLamports,\n priorityFee,\n transactionFee,\n transactionFeeWithMargin,\n computeUnitLimit,\n computeUnitPriceMicroLamports,\n unitsConsumed,\n createdAccounts: rentEstimate.createdAccounts,\n rentFee: rentEstimate.rentFee,\n rentFeeNet: rentEstimate.rentFeeNet,\n },\n };\n}\n\n/** @internal */\nexport async function _estimateGasFromSwapResponse({\n crossChain,\n feeUnitsMarginBps,\n fromAddress,\n sourceClient,\n swap,\n}: {\n crossChain: boolean;\n feeUnitsMarginBps?: number;\n fromAddress: EvmAddress;\n sourceClient: ReturnType<typeof getEvmClientForChain>;\n swap: WrappedSwapTransactionResponse;\n}): Promise<bigint> {\n try {\n const gasEstimate = await sourceClient.estimateGas({\n account: fromAddress,\n to: swap.to,\n data: swap.data,\n value: swap.value,\n });\n\n return applyFeeUnitsBpsMargin(gasEstimate, feeUnitsMarginBps);\n } catch (err) {\n let details = 'Failed to estimate gas for Markr swap transaction.';\n\n try {\n const markrSwapWrapperAbi = await getMarkrSwapWrapperAbi(crossChain);\n const decodedRevert = decodeMarkrRevertError(markrSwapWrapperAbi, err);\n\n if (decodedRevert) {\n details = `${details} Markr revert: ${decodedRevert}.`;\n }\n } catch {\n // Keep the base details message if revert decoding fails.\n }\n\n throw new SdkError('Error during gas estimation', ErrorCode.VIEM_ERROR, {\n cause: err,\n details,\n });\n }\n}\n\nfunction _getComputeBudgetPriceAndLimit(transactionMessageBytes: TransactionMessageBytes): {\n computeUnitLimit: bigint;\n computeUnitPriceMicroLamports: bigint;\n} {\n const compiledTxMessage = getCompiledTransactionMessageDecoder().decode(transactionMessageBytes);\n let computeUnitLimit = 0n;\n let computeUnitPriceMicroLamports = 0n;\n\n for (const instruction of compiledTxMessage.instructions) {\n const programAddress = compiledTxMessage.staticAccounts[instruction.programAddressIndex];\n if (programAddress !== SOLANA_COMPUTE_BUDGET_PROGRAM) {\n continue;\n }\n\n const data = instruction.data;\n\n if (!data || data.length < 1) {\n continue;\n }\n\n const discriminator = data[0];\n const view = new DataView(data.buffer, data.byteOffset, data.byteLength);\n\n // ComputeBudget::SetComputeUnitLimit(u32)\n if (discriminator === 2 && data.length >= 1 + 4) {\n computeUnitLimit = BigInt(view.getUint32(1, true));\n continue;\n }\n\n // ComputeBudget::SetComputeUnitPrice(u64)\n if (discriminator === 3 && data.length >= 1 + 8) {\n computeUnitPriceMicroLamports = view.getBigUint64(1, true);\n }\n }\n\n return {\n computeUnitLimit,\n computeUnitPriceMicroLamports,\n };\n}\n\nfunction _calculatePriorityFeeLamports(computeUnitLimit: bigint, computeUnitPriceMicroLamports: bigint): bigint {\n if (computeUnitLimit <= 0n || computeUnitPriceMicroLamports <= 0n) {\n return 0n;\n }\n\n return (\n (computeUnitLimit * computeUnitPriceMicroLamports + (MICRO_LAMPORTS_PER_LAMPORT - 1n)) / MICRO_LAMPORTS_PER_LAMPORT\n );\n}\n\nfunction _getSignatureCount(transaction: Transaction): bigint {\n const signatures = (transaction as { signatures?: unknown }).signatures;\n\n if (Array.isArray(signatures)) {\n return BigInt(signatures.length);\n }\n\n if (signatures && typeof signatures === 'object') {\n return BigInt(Object.keys(signatures).length);\n }\n\n return 0n;\n}\n\nasync function _estimateRentFeesFromTransaction({\n fromAddress,\n rpc,\n transaction,\n}: {\n fromAddress: string;\n rpc: Rpc<SolanaRpcApi>;\n transaction: Transaction;\n}): Promise<SolanaRentEstimate> {\n const compiledMessage: CompiledTransactionMessage = getCompiledTransactionMessageDecoder().decode(\n transaction.messageBytes,\n );\n const allAccounts = await _resolveAllTransactionAccounts(compiledMessage, rpc);\n\n const createdAccounts = new Map<string, Omit<CreatedRentAccount, 'refunded'>>();\n const closedAccountDestinationByAddress = new Map<string, string>();\n\n let ataRentLamports: bigint | undefined;\n\n for (const instruction of compiledMessage.instructions) {\n const programAddress = allAccounts[instruction.programAddressIndex];\n\n if (programAddress === SOLANA_SYSTEM_PROGRAM) {\n const createdAccountAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n const lamports = _getSystemCreateAccountLamports(instruction.data);\n if (createdAccountAddress && lamports > 0n) {\n createdAccounts.set(createdAccountAddress, {\n address: createdAccountAddress,\n kind: 'system',\n lamports,\n });\n }\n continue;\n }\n\n if (programAddress === SOLANA_ASSOCIATED_TOKEN_PROGRAM) {\n const discriminator = instruction.data?.[0];\n const isAssociatedTokenCreateInstruction =\n discriminator === undefined || discriminator === 0 || discriminator === 1;\n\n if (!isAssociatedTokenCreateInstruction) {\n continue;\n }\n\n const associatedTokenAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n if (!associatedTokenAddress || !isSolanaAddress(associatedTokenAddress)) {\n continue;\n }\n\n const accountInfo = await rpc.getAccountInfo(associatedTokenAddress).send();\n if (accountInfo.value) {\n continue;\n }\n\n if (ataRentLamports === undefined) {\n ataRentLamports = await rpc.getMinimumBalanceForRentExemption(SPL_TOKEN_ACCOUNT_SIZE_BYTES).send();\n }\n\n createdAccounts.set(associatedTokenAddress, {\n address: associatedTokenAddress,\n kind: 'ata',\n lamports: ataRentLamports,\n });\n\n continue;\n }\n\n const isSplTokenProgram =\n programAddress === SOLANA_SPL_TOKEN_PROGRAM || programAddress === SOLANA_SPL_TOKEN_2022_PROGRAM;\n if (!isSplTokenProgram) {\n continue;\n }\n\n const discriminator = instruction.data?.[0];\n if (discriminator !== 9) {\n continue;\n }\n\n const closedAccountAddress = _getInstructionAccountAddress(allAccounts, instruction, 0);\n const refundDestinationAddress = _getInstructionAccountAddress(allAccounts, instruction, 1);\n\n if (closedAccountAddress && refundDestinationAddress) {\n closedAccountDestinationByAddress.set(closedAccountAddress, refundDestinationAddress);\n }\n }\n\n const normalizedFromAddress = isSolanaAddress(fromAddress) ? fromAddress : undefined;\n const createdAccountEntries = Array.from(createdAccounts.values()).map((createdAccount) => {\n const refundDestinationAddress = closedAccountDestinationByAddress.get(createdAccount.address);\n const refunded = normalizedFromAddress !== undefined && refundDestinationAddress === normalizedFromAddress;\n\n return {\n ...createdAccount,\n refunded,\n } satisfies CreatedRentAccount;\n });\n\n const rentLamportsUpfront = createdAccountEntries.reduce((acc, account) => acc + account.lamports, 0n);\n const refundedLamports = createdAccountEntries.reduce(\n (acc, account) => acc + (account.refunded ? account.lamports : 0n),\n 0n,\n );\n const rentFeeNet = rentLamportsUpfront - refundedLamports;\n\n return {\n createdAccounts: createdAccountEntries,\n rentFee: rentLamportsUpfront,\n rentFeeNet,\n refundable: refundedLamports,\n };\n}\n\nfunction _isObject(value: unknown): value is object {\n return !!value && typeof value === 'object';\n}\n\nfunction _getObjectProperty(value: unknown, key: string): unknown {\n if (!_isObject(value)) {\n return undefined;\n }\n\n return Reflect.get(value, key);\n}\n\nfunction _toNumberIndexArray(value: unknown): number[] | undefined {\n if (value instanceof Uint8Array) {\n return Array.from(value);\n }\n\n if (Array.isArray(value) && value.every((entry) => Number.isInteger(entry) && entry >= 0)) {\n return value;\n }\n\n return undefined;\n}\n\nfunction _getCompiledMessageAddressTableLookups(\n compiledMessage: CompiledTransactionMessage,\n): CompiledMessageAddressTableLookup[] {\n const addressTableLookupsRaw = _getObjectProperty(compiledMessage, 'addressTableLookups');\n if (!Array.isArray(addressTableLookupsRaw)) {\n return [];\n }\n\n const lookups: CompiledMessageAddressTableLookup[] = [];\n\n for (const lookupRaw of addressTableLookupsRaw) {\n const accountKey = _getObjectProperty(lookupRaw, 'accountKey');\n const writableIndexes = _toNumberIndexArray(_getObjectProperty(lookupRaw, 'writableIndexes'));\n const readonlyIndexes = _toNumberIndexArray(_getObjectProperty(lookupRaw, 'readonlyIndexes'));\n\n if (typeof accountKey !== 'string' || !writableIndexes || !readonlyIndexes) {\n continue;\n }\n\n lookups.push({\n accountKey,\n writableIndexes,\n readonlyIndexes,\n });\n }\n\n return lookups;\n}\n\nfunction _extractLookupTableAddresses(response: unknown): string[] {\n const lookupTableValue = _getObjectProperty(response, 'value');\n if (!_isObject(lookupTableValue)) {\n return [];\n }\n\n const addresses = _getObjectProperty(lookupTableValue, 'addresses');\n if (!Array.isArray(addresses)) {\n return [];\n }\n\n return addresses.filter((entry): entry is string => typeof entry === 'string');\n}\n\nasync function _loadLookupTableAddresses(\n compiledMessage: CompiledTransactionMessage,\n rpc: Rpc<SolanaRpcApi>,\n): Promise<LookupTableAddresses> {\n const writableAddresses: string[] = [];\n const readonlyAddresses: string[] = [];\n const addressTableLookups = _getCompiledMessageAddressTableLookups(compiledMessage);\n\n for (const lookup of addressTableLookups) {\n if (!isSolanaAddress(lookup.accountKey)) {\n continue;\n }\n\n const tableAddresses = await _fetchAddressLookupTableAddresses(rpc, lookup.accountKey);\n\n for (const index of lookup.writableIndexes) {\n const address = tableAddresses[index];\n if (address) {\n writableAddresses.push(address);\n }\n }\n\n for (const index of lookup.readonlyIndexes) {\n const address = tableAddresses[index];\n if (address) {\n readonlyAddresses.push(address);\n }\n }\n }\n\n return {\n writableAddresses,\n readonlyAddresses,\n };\n}\n\nasync function _fetchAddressLookupTableAddresses(rpc: Rpc<SolanaRpcApi>, accountKey: SolanaAddress): Promise<string[]> {\n const getAddressLookupTable = _getObjectProperty(rpc, 'getAddressLookupTable');\n if (typeof getAddressLookupTable !== 'function') {\n return [];\n }\n\n const request = Reflect.apply(getAddressLookupTable, rpc, [accountKey]);\n const send = _getObjectProperty(request, 'send');\n\n if (typeof send !== 'function') {\n return [];\n }\n\n const response = await Reflect.apply(send, request, []);\n return _extractLookupTableAddresses(response);\n}\n\nasync function _resolveAllTransactionAccounts(\n compiledMessage: CompiledTransactionMessage,\n rpc: Rpc<SolanaRpcApi>,\n): Promise<string[]> {\n const { writableAddresses, readonlyAddresses } = await _loadLookupTableAddresses(compiledMessage, rpc);\n\n return [...compiledMessage.staticAccounts, ...writableAddresses, ...readonlyAddresses];\n}\n\n/**\n * Resolves an account address used by a compiled instruction.\n *\n * Compiled instructions reference accounts by index into the transaction\n * message account list. For versioned transactions this includes static\n * accounts followed by writable and readonly addresses loaded from lookup\n * tables. This helper converts an instruction-local `accountIndex` into the\n * corresponding account address.\n *\n * Returns `undefined` when the instruction does not provide an account at the\n * requested index.\n */\nfunction _getInstructionAccountAddress(\n allAccounts: readonly string[],\n instruction: CompiledTransactionMessage['instructions'][number],\n accountIndex: number,\n): string | undefined {\n const messageAccountIndex = instruction.accountIndices?.[accountIndex];\n\n if (messageAccountIndex === undefined) {\n return undefined;\n }\n\n return allAccounts[messageAccountIndex];\n}\n\n/**\n * Extracts lamports from System Program account-creation instruction data.\n *\n * Supported instruction layouts:\n * - `CreateAccount` (`u32 discriminator = 0`): lamports at byte offset `4`.\n * - `CreateAccountWithSeed` (`u32 discriminator = 3`): lamports are located\n * after the variable-length seed field.\n *\n * Returns `0n` when data is missing, malformed, or not one of the supported\n * System Program create-account variants.\n */\nfunction _getSystemCreateAccountLamports(data: CompiledTransactionMessage['instructions'][number]['data']): bigint {\n if (!data || data.length < 12) {\n return 0n;\n }\n\n const view = new DataView(data.buffer, data.byteOffset, data.byteLength);\n const instructionDiscriminator = view.getUint32(0, true);\n\n if (instructionDiscriminator === 0 && data.length >= 12) {\n return view.getBigUint64(4, true);\n }\n\n if (instructionDiscriminator === 3 && data.length >= 44) {\n const seedLenOffset = 36;\n const seedLength = Number(view.getBigUint64(seedLenOffset, true));\n if (!Number.isSafeInteger(seedLength) || seedLength < 0) {\n return 0n;\n }\n\n const lamportsOffset = seedLenOffset + 8 + seedLength;\n if (lamportsOffset + 8 > data.length) {\n return 0n;\n }\n\n return new DataView(data.buffer, data.byteOffset + lamportsOffset, 8).getBigUint64(0, true);\n }\n\n return 0n;\n}\n"],"mappings":"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"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{ErrorCode as e,ErrorReason as t,InvalidParamsError as n,SdkError as r}from"../../../errors.js";import{caip2ToEip155ChainId as i}from"../../../utils/caip.js";import{isEvmNamespace as a,isSolanaNamespace as o}from"../../../_utils/chain.js";import{applyFeeUnitsBpsMargin as s,getEvmClientForChain as c,getSolanaRpcForChain as l}from"../../_utils.js";import{estimateEvmFeesPerGas as u}from"../../_evm-gas.js";import{markrGetSpenderAddress as d,markrSwap as f}from"../_api.js";import{assetToAddressString as p,calculateMarkrMinimumAmountOut as m,decodeMarkrRevertError as h,
|
|
1
|
+
import{ErrorCode as e,ErrorReason as t,InvalidParamsError as n,SdkError as r}from"../../../errors.js";import{caip2ToEip155ChainId as i}from"../../../utils/caip.js";import{isEvmNamespace as a,isSolanaNamespace as o}from"../../../_utils/chain.js";import{applyFeeUnitsBpsMargin as s,getEvmClientForChain as c,getSolanaRpcForChain as l}from"../../_utils.js";import{estimateEvmFeesPerGas as u}from"../../_evm-gas.js";import{markrGetSpenderAddress as d,markrSwap as f}from"../_api.js";import{assetToAddressString as p,calculateMarkrMinimumAmountOut as m,decodeMarkrRevertError as h,getAdditiveSourceAssetFeeAmount as g,getMarkrSwapWrapperAbi as _,isTokenAddressNative as v}from"../_utils.js";import{isEvmSwapResponse as y,isSolanaSwapResponse as b}from"../_type-guards.js";import{decodeSolanaTransactionBase64 as x,refreshSolanaSwapTransactionBlockhash as S}from"../_solana-utils.js";import{encodeFunctionData as C,erc20Abi as w,isAddress as T}from"viem";import{getCompiledTransactionMessageDecoder as E,isAddress as D}from"@solana/kit";const O=1000000n;function k(e){return async(r,i)=>{if(a(r.sourceChain.chainId))return await A(r,i,e);if(o(r.sourceChain.chainId))return await j(r,i,e);throw new n(t.INVALID_PARAMS,`Unsupported source chain namespace for estimateNativeFee: ${r.sourceChain.chainId}`)}}async function A(a,o,{apiOptions:l,appId:h}){if(!T(a.fromAddress))throw new n(t.INVALID_PARAMS,`Invalid fromAddress: ${a.fromAddress}`);let _=c({chain:a.sourceChain}),b=p(a.assetIn,a.sourceChain.chainId);if(!T(b))throw new r(t.INVALID_PARAMS,e.INVALID_PARAMS,{details:`assetIn address is not a valid EVM address. Can not call estimateGas.`});let x=v(b),S=a.sourceChain.chainId.toLowerCase()!==a.targetChain.chainId.toLowerCase(),E=a.amountIn+g(a),D=0n;if(!x){let t,{address:n}=await d(l,{chainId:i(a.sourceChain.chainId),crossChainSwap:S,quoteId:a.id});try{t=await _.readContract({address:b,abi:w,functionName:`allowance`,args:[a.fromAddress,n]})}catch(t){throw new r(`Error during allowance check`,e.VIEM_ERROR,{cause:t,details:`Failed to read ERC20 allowance for Markr spender.`})}if(t<E)try{D=await _.estimateGas({account:a.fromAddress,to:b,data:C({abi:w,functionName:`approve`,args:[n,E]}),value:0n})}catch(t){throw new r(`Error during gas estimation`,e.VIEM_ERROR,{cause:t,details:`Failed to estimate gas for ERC20 approval transaction.`})}}let O=D>0n,k=m({amountOut:a.amountOut,assetOut:a.assetOut,slippageBps:a.slippageBps}),A=await f(l,{amountIn:a.amountIn.toString(),appId:h,minAmountOut:k.toString(),tokenIn:b,tokenOut:p(a.assetOut,a.targetChain.chainId),uuid:a.id});if(!y(A))throw new r(t.CHAIN_NOT_SUPPORTED,e.INVALID_PARAMS,{details:`Received non-EVM swap response from Markr. Expected EVM transaction data for gas estimation, but got response with chainType ${A.chainType}.`});let j=0n;j=O?a.gasEstimate?s(a.gasEstimate,o?.feeUnitsMarginBps):s(a.fees.filter(e=>e.type===`gas`&&e.chainId===a.sourceChain.chainId).reduce((e,t)=>e+t.amount,0n)||700000n,o?.feeUnitsMarginBps):await M({crossChain:S,feeUnitsMarginBps:o?.feeUnitsMarginBps,fromAddress:a.fromAddress,sourceClient:_,swap:A});let N=await u(_,a.sourceChain,o?.overrides?.feeRateTier),P=o?.overrides?.maxFeePerGas??N.maxFeePerGas,F=o?.overrides?.maxPriorityFeePerGas??N.maxPriorityFeePerGas,I=(j+s(D,o?.feeUnitsMarginBps))*P;return{asset:a.sourceChain.networkToken,totalFee:I,totalUpfrontFee:I,meta:{approvalFee:O?s(D,o?.feeUnitsMarginBps)*P:void 0,maxFeePerGas:P,maxPriorityFeePerGas:F}}}async function j(n,i,{apiOptions:a,appId:o}){let c=p(n.assetIn,n.sourceChain.chainId);if(!D(c))throw new r(t.INVALID_PARAMS,e.INVALID_PARAMS,{details:`assetIn address is not a valid Solana address. Can not call estimateGas.`});let u=m({amountOut:n.amountOut,assetOut:n.assetOut,slippageBps:n.slippageBps}),d=await f(a,{amountIn:n.amountIn.toString(),appId:o,minAmountOut:u.toString(),tokenIn:c,tokenOut:p(n.assetOut,n.targetChain.chainId),userPublicKey:n.fromAddress,uuid:n.id});if(!b(d))throw new r(t.CHAIN_NOT_SUPPORTED,e.INVALID_PARAMS,{details:`Received non-Solana swap response from Markr. Expected Solana transaction data for fee estimation.`});let h=l({chain:n.sourceChain}),g=await S(d.swapTransaction,h),_=x(g),v=_.messageBytes,y=F(_)*5000n,{computeUnitLimit:C,computeUnitPriceMicroLamports:w}=N(v),T=P(C,w),E=y+T,O=s(E,i?.feeUnitsMarginBps),k=(await h.simulateTransaction(g,{encoding:`base64`,sigVerify:!1,replaceRecentBlockhash:!0}).send()).value.unitsConsumed??0n,A=await I({fromAddress:n.fromAddress,rpc:h,transaction:_}),j=O+A.rentFee;return{asset:n.sourceChain.networkToken,totalFee:j,totalUpfrontFee:j,refundable:A.refundable,meta:{baseFee:y,priorityFee:T,transactionFee:E,transactionFeeWithMargin:O,computeUnitLimit:C,computeUnitPriceMicroLamports:w,unitsConsumed:k,createdAccounts:A.createdAccounts,rentFee:A.rentFee,rentFeeNet:A.rentFeeNet}}}async function M({crossChain:t,feeUnitsMarginBps:n,fromAddress:i,sourceClient:a,swap:o}){try{return s(await a.estimateGas({account:i,to:o.to,data:o.data,value:o.value}),n)}catch(n){let i=`Failed to estimate gas for Markr swap transaction.`;try{let e=h(await _(t),n);e&&(i=`${i} Markr revert: ${e}.`)}catch{}throw new r(`Error during gas estimation`,e.VIEM_ERROR,{cause:n,details:i})}}function N(e){let t=E().decode(e),n=0n,r=0n;for(let e of t.instructions){if(t.staticAccounts[e.programAddressIndex]!==`ComputeBudget111111111111111111111111111111`)continue;let i=e.data;if(!i||i.length<1)continue;let a=i[0],o=new DataView(i.buffer,i.byteOffset,i.byteLength);if(a===2&&i.length>=5){n=BigInt(o.getUint32(1,!0));continue}a===3&&i.length>=9&&(r=o.getBigUint64(1,!0))}return{computeUnitLimit:n,computeUnitPriceMicroLamports:r}}function P(e,t){return e<=0n||t<=0n?0n:(e*t+(O-1n))/O}function F(e){let t=e.signatures;return Array.isArray(t)?BigInt(t.length):t&&typeof t==`object`?BigInt(Object.keys(t).length):0n}async function I({fromAddress:e,rpc:t,transaction:n}){let r=E().decode(n.messageBytes),i=await W(r,t),a=new Map,o=new Map,s;for(let e of r.instructions){let n=i[e.programAddressIndex];if(n===`11111111111111111111111111111111`){let t=G(i,e,1),n=K(e.data);t&&n>0n&&a.set(t,{address:t,kind:`system`,lamports:n});continue}if(n===`ATokenGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL`){let n=e.data?.[0];if(!(n===void 0||n===0||n===1))continue;let r=G(i,e,1);if(!r||!D(r)||(await t.getAccountInfo(r).send()).value)continue;s===void 0&&(s=await t.getMinimumBalanceForRentExemption(165n).send()),a.set(r,{address:r,kind:`ata`,lamports:s});continue}if(!(n===`TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA`||n===`TokenzQdBNbLqP5VEhdkAS6EPFLC1PHnBqCXEpPxuEb`)||e.data?.[0]!==9)continue;let r=G(i,e,0),c=G(i,e,1);r&&c&&o.set(r,c)}let c=D(e)?e:void 0,l=Array.from(a.values()).map(e=>{let t=o.get(e.address),n=c!==void 0&&t===c;return{...e,refunded:n}}),u=l.reduce((e,t)=>e+t.lamports,0n),d=l.reduce((e,t)=>e+(t.refunded?t.lamports:0n),0n);return{createdAccounts:l,rentFee:u,rentFeeNet:u-d,refundable:d}}function L(e){return!!e&&typeof e==`object`}function R(e,t){if(L(e))return Reflect.get(e,t)}function z(e){if(e instanceof Uint8Array)return Array.from(e);if(Array.isArray(e)&&e.every(e=>Number.isInteger(e)&&e>=0))return e}function B(e){let t=R(e,`addressTableLookups`);if(!Array.isArray(t))return[];let n=[];for(let e of t){let t=R(e,`accountKey`),r=z(R(e,`writableIndexes`)),i=z(R(e,`readonlyIndexes`));typeof t!=`string`||!r||!i||n.push({accountKey:t,writableIndexes:r,readonlyIndexes:i})}return n}function V(e){let t=R(e,`value`);if(!L(t))return[];let n=R(t,`addresses`);return Array.isArray(n)?n.filter(e=>typeof e==`string`):[]}async function H(e,t){let n=[],r=[],i=B(e);for(let e of i){if(!D(e.accountKey))continue;let i=await U(t,e.accountKey);for(let t of e.writableIndexes){let e=i[t];e&&n.push(e)}for(let t of e.readonlyIndexes){let e=i[t];e&&r.push(e)}}return{writableAddresses:n,readonlyAddresses:r}}async function U(e,t){let n=R(e,`getAddressLookupTable`);if(typeof n!=`function`)return[];let r=Reflect.apply(n,e,[t]),i=R(r,`send`);return typeof i==`function`?V(await Reflect.apply(i,r,[])):[]}async function W(e,t){let{writableAddresses:n,readonlyAddresses:r}=await H(e,t);return[...e.staticAccounts,...n,...r]}function G(e,t,n){let r=t.accountIndices?.[n];if(r!==void 0)return e[r]}function K(e){if(!e||e.length<12)return 0n;let t=new DataView(e.buffer,e.byteOffset,e.byteLength),n=t.getUint32(0,!0);if(n===0&&e.length>=12)return t.getBigUint64(4,!0);if(n===3&&e.length>=44){let n=Number(t.getBigUint64(36,!0));if(!Number.isSafeInteger(n)||n<0)return 0n;let r=44+n;return r+8>e.length?0n:new DataView(e.buffer,e.byteOffset+r,8).getBigUint64(0,!0)}return 0n}export{M as _estimateGasFromSwapResponse,k as estimateNativeFeeFactory};
|
|
2
2
|
//# sourceMappingURL=estimate-native-fee.js.map
|