agentwallet-sdk 2.5.0 → 3.1.0
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/README.md +432 -1
- package/dist/bridge/client.d.ts +17 -49
- package/dist/bridge/client.d.ts.map +1 -1
- package/dist/bridge/client.js +94 -124
- package/dist/bridge/client.js.map +1 -1
- package/dist/bridge/index.d.ts +4 -2
- package/dist/bridge/index.d.ts.map +1 -1
- package/dist/bridge/index.js +8 -2
- package/dist/bridge/index.js.map +1 -1
- package/dist/bridge/types.d.ts +100 -19
- package/dist/bridge/types.d.ts.map +1 -1
- package/dist/bridge/types.js +167 -18
- package/dist/bridge/types.js.map +1 -1
- package/dist/bridge/unified.d.ts +101 -0
- package/dist/bridge/unified.d.ts.map +1 -0
- package/dist/bridge/unified.js +284 -0
- package/dist/bridge/unified.js.map +1 -0
- package/dist/chains.d.ts +25 -0
- package/dist/chains.d.ts.map +1 -1
- package/dist/chains.js +33 -0
- package/dist/chains.js.map +1 -1
- package/dist/fiat/index.d.ts +10 -0
- package/dist/fiat/index.d.ts.map +1 -0
- package/dist/fiat/index.js +9 -0
- package/dist/fiat/index.js.map +1 -0
- package/dist/fiat/onramp.d.ts +101 -0
- package/dist/fiat/onramp.d.ts.map +1 -0
- package/dist/fiat/onramp.js +155 -0
- package/dist/fiat/onramp.js.map +1 -0
- package/dist/fiat/providers/index.d.ts +16 -0
- package/dist/fiat/providers/index.d.ts.map +1 -0
- package/dist/fiat/providers/index.js +30 -0
- package/dist/fiat/providers/index.js.map +1 -0
- package/dist/fiat/providers/moonpay.d.ts +22 -0
- package/dist/fiat/providers/moonpay.d.ts.map +1 -0
- package/dist/fiat/providers/moonpay.js +107 -0
- package/dist/fiat/providers/moonpay.js.map +1 -0
- package/dist/fiat/providers/stripe.d.ts +26 -0
- package/dist/fiat/providers/stripe.d.ts.map +1 -0
- package/dist/fiat/providers/stripe.js +135 -0
- package/dist/fiat/providers/stripe.js.map +1 -0
- package/dist/fiat/providers/transak.d.ts +26 -0
- package/dist/fiat/providers/transak.d.ts.map +1 -0
- package/dist/fiat/providers/transak.js +119 -0
- package/dist/fiat/providers/transak.js.map +1 -0
- package/dist/fiat/types.d.ts +106 -0
- package/dist/fiat/types.d.ts.map +1 -0
- package/dist/fiat/types.js +13 -0
- package/dist/fiat/types.js.map +1 -0
- package/dist/gas/index.d.ts +4 -0
- package/dist/gas/index.d.ts.map +1 -0
- package/dist/gas/index.js +3 -0
- package/dist/gas/index.js.map +1 -0
- package/dist/gas/sponsor.d.ts +70 -0
- package/dist/gas/sponsor.d.ts.map +1 -0
- package/dist/gas/sponsor.js +193 -0
- package/dist/gas/sponsor.js.map +1 -0
- package/dist/gas/types.d.ts +76 -0
- package/dist/gas/types.d.ts.map +1 -0
- package/dist/gas/types.js +21 -0
- package/dist/gas/types.js.map +1 -0
- package/dist/index.d.ts +85 -71
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +21 -3
- package/dist/index.js.map +1 -1
- package/dist/settlement/index.d.ts +4 -0
- package/dist/settlement/index.d.ts.map +1 -0
- package/dist/settlement/index.js +3 -0
- package/dist/settlement/index.js.map +1 -0
- package/dist/settlement/types.d.ts +66 -0
- package/dist/settlement/types.d.ts.map +1 -0
- package/dist/settlement/types.js +37 -0
- package/dist/settlement/types.js.map +1 -0
- package/dist/settlement/verifier.d.ts +75 -0
- package/dist/settlement/verifier.d.ts.map +1 -0
- package/dist/settlement/verifier.js +354 -0
- package/dist/settlement/verifier.js.map +1 -0
- package/dist/solana/bridge.d.ts +144 -0
- package/dist/solana/bridge.d.ts.map +1 -0
- package/dist/solana/bridge.js +352 -0
- package/dist/solana/bridge.js.map +1 -0
- package/dist/solana/index.d.ts +8 -0
- package/dist/solana/index.d.ts.map +1 -0
- package/dist/solana/index.js +6 -0
- package/dist/solana/index.js.map +1 -0
- package/dist/solana/swap.d.ts +85 -0
- package/dist/solana/swap.d.ts.map +1 -0
- package/dist/solana/swap.js +173 -0
- package/dist/solana/swap.js.map +1 -0
- package/dist/solana/types.d.ts +126 -0
- package/dist/solana/types.d.ts.map +1 -0
- package/dist/solana/types.js +10 -0
- package/dist/solana/types.js.map +1 -0
- package/dist/solana/wallet.d.ts +83 -0
- package/dist/solana/wallet.d.ts.map +1 -0
- package/dist/solana/wallet.js +164 -0
- package/dist/solana/wallet.js.map +1 -0
- package/dist/solana/x402.d.ts +69 -0
- package/dist/solana/x402.d.ts.map +1 -0
- package/dist/solana/x402.js +154 -0
- package/dist/solana/x402.js.map +1 -0
- package/dist/swap/index.d.ts +1 -0
- package/dist/swap/index.d.ts.map +1 -1
- package/dist/swap/index.js +2 -0
- package/dist/swap/index.js.map +1 -1
- package/dist/swap/router/cache.d.ts +13 -0
- package/dist/swap/router/cache.d.ts.map +1 -0
- package/dist/swap/router/cache.js +30 -0
- package/dist/swap/router/cache.js.map +1 -0
- package/dist/swap/router/flashbots.d.ts +10 -0
- package/dist/swap/router/flashbots.d.ts.map +1 -0
- package/dist/swap/router/flashbots.js +43 -0
- package/dist/swap/router/flashbots.js.map +1 -0
- package/dist/swap/router/health.d.ts +17 -0
- package/dist/swap/router/health.d.ts.map +1 -0
- package/dist/swap/router/health.js +38 -0
- package/dist/swap/router/health.js.map +1 -0
- package/dist/swap/router/index.d.ts +10 -0
- package/dist/swap/router/index.d.ts.map +1 -0
- package/dist/swap/router/index.js +10 -0
- package/dist/swap/router/index.js.map +1 -0
- package/dist/swap/router/providers/cowswap.d.ts +11 -0
- package/dist/swap/router/providers/cowswap.d.ts.map +1 -0
- package/dist/swap/router/providers/cowswap.js +79 -0
- package/dist/swap/router/providers/cowswap.js.map +1 -0
- package/dist/swap/router/providers/index.d.ts +20 -0
- package/dist/swap/router/providers/index.d.ts.map +1 -0
- package/dist/swap/router/providers/index.js +32 -0
- package/dist/swap/router/providers/index.js.map +1 -0
- package/dist/swap/router/providers/jupiter.d.ts +12 -0
- package/dist/swap/router/providers/jupiter.d.ts.map +1 -0
- package/dist/swap/router/providers/jupiter.js +73 -0
- package/dist/swap/router/providers/jupiter.js.map +1 -0
- package/dist/swap/router/providers/lifi.d.ts +11 -0
- package/dist/swap/router/providers/lifi.d.ts.map +1 -0
- package/dist/swap/router/providers/lifi.js +123 -0
- package/dist/swap/router/providers/lifi.js.map +1 -0
- package/dist/swap/router/providers/oneinch.d.ts +13 -0
- package/dist/swap/router/providers/oneinch.d.ts.map +1 -0
- package/dist/swap/router/providers/oneinch.js +71 -0
- package/dist/swap/router/providers/oneinch.js.map +1 -0
- package/dist/swap/router/providers/paraswap.d.ts +11 -0
- package/dist/swap/router/providers/paraswap.d.ts.map +1 -0
- package/dist/swap/router/providers/paraswap.js +73 -0
- package/dist/swap/router/providers/paraswap.js.map +1 -0
- package/dist/swap/router/providers/uniswap.d.ts +31 -0
- package/dist/swap/router/providers/uniswap.d.ts.map +1 -0
- package/dist/swap/router/providers/uniswap.js +116 -0
- package/dist/swap/router/providers/uniswap.js.map +1 -0
- package/dist/swap/router/providers/zerox.d.ts +13 -0
- package/dist/swap/router/providers/zerox.d.ts.map +1 -0
- package/dist/swap/router/providers/zerox.js +94 -0
- package/dist/swap/router/providers/zerox.js.map +1 -0
- package/dist/swap/router/router.d.ts +86 -0
- package/dist/swap/router/router.d.ts.map +1 -0
- package/dist/swap/router/router.js +224 -0
- package/dist/swap/router/router.js.map +1 -0
- package/dist/swap/router/rsi/engine.d.ts +60 -0
- package/dist/swap/router/rsi/engine.d.ts.map +1 -0
- package/dist/swap/router/rsi/engine.js +483 -0
- package/dist/swap/router/rsi/engine.js.map +1 -0
- package/dist/swap/router/rsi/index.d.ts +3 -0
- package/dist/swap/router/rsi/index.d.ts.map +1 -0
- package/dist/swap/router/rsi/index.js +3 -0
- package/dist/swap/router/rsi/index.js.map +1 -0
- package/dist/swap/router/rsi/types.d.ts +106 -0
- package/dist/swap/router/rsi/types.d.ts.map +1 -0
- package/dist/swap/router/rsi/types.js +3 -0
- package/dist/swap/router/rsi/types.js.map +1 -0
- package/dist/swap/router/types.d.ts +120 -0
- package/dist/swap/router/types.d.ts.map +1 -0
- package/dist/swap/router/types.js +16 -0
- package/dist/swap/router/types.js.map +1 -0
- package/dist/x402/index.d.ts +1 -1
- package/dist/x402/index.d.ts.map +1 -1
- package/dist/x402/index.js +1 -1
- package/dist/x402/index.js.map +1 -1
- package/dist/x402/types.d.ts +4 -2
- package/dist/x402/types.d.ts.map +1 -1
- package/dist/x402/types.js +12 -1
- package/dist/x402/types.js.map +1 -1
- package/package.json +8 -3
|
@@ -0,0 +1,354 @@
|
|
|
1
|
+
// SettlementVerifier — On-chain settlement verification for agent transactions
|
|
2
|
+
//
|
|
3
|
+
// Verifies that bridge transfers, x402 payments, and regular transactions actually
|
|
4
|
+
// settled on-chain. Uses Alchemy RPC for supported chains, public RPCs as fallback.
|
|
5
|
+
//
|
|
6
|
+
// Supported verification types:
|
|
7
|
+
// - EVM transaction receipts (any EVM chain)
|
|
8
|
+
// - Solana transaction confirmation
|
|
9
|
+
// - CCTP bridge transfers end-to-end (source burn + attestation + dest mint)
|
|
10
|
+
// - x402 ERC20 payment verification (payee + amount from event logs)
|
|
11
|
+
import { SETTLEMENT_ALCHEMY_NETWORKS, SETTLEMENT_PUBLIC_RPCS, CIRCLE_ATTESTATION_API, ERC20_TRANSFER_TOPIC, SOLANA_RPC_URL, } from './types.js';
|
|
12
|
+
export class SettlementVerifier {
|
|
13
|
+
constructor(config = {}) {
|
|
14
|
+
this.alchemyApiKey = config.alchemyApiKey;
|
|
15
|
+
}
|
|
16
|
+
// ─── EVM Transaction Verification ───
|
|
17
|
+
/**
|
|
18
|
+
* Verify that an EVM transaction settled on-chain.
|
|
19
|
+
*
|
|
20
|
+
* Uses Alchemy RPC if apiKey is configured and chain is supported,
|
|
21
|
+
* otherwise falls back to public RPCs.
|
|
22
|
+
*
|
|
23
|
+
* @param params.chain - SDK chain name (e.g. 'base', 'arbitrum')
|
|
24
|
+
* @param params.txHash - Transaction hash to verify
|
|
25
|
+
* @param params.confirmations - Minimum confirmations required (default: 1)
|
|
26
|
+
*/
|
|
27
|
+
async verifyEvmTransaction(params) {
|
|
28
|
+
const requiredConfirmations = params.confirmations ?? 1;
|
|
29
|
+
const rpcUrl = this._getRpcUrl(params.chain);
|
|
30
|
+
try {
|
|
31
|
+
// Get receipt
|
|
32
|
+
const receiptResp = await this._evmRpc(rpcUrl, 'eth_getTransactionReceipt', [params.txHash]);
|
|
33
|
+
const receipt = receiptResp.result;
|
|
34
|
+
if (!receipt) {
|
|
35
|
+
// Not mined yet — check if it's in the mempool
|
|
36
|
+
return {
|
|
37
|
+
status: 'pending',
|
|
38
|
+
txHash: params.txHash,
|
|
39
|
+
blockNumber: 0n,
|
|
40
|
+
confirmations: 0,
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
// Get current block number to compute confirmations
|
|
44
|
+
const blockResp = await this._evmRpc(rpcUrl, 'eth_blockNumber', []);
|
|
45
|
+
const currentBlock = BigInt(blockResp.result);
|
|
46
|
+
const txBlock = BigInt(receipt.blockNumber);
|
|
47
|
+
const confirmationCount = Number(currentBlock - txBlock) + 1;
|
|
48
|
+
const success = receipt.status === '0x1';
|
|
49
|
+
if (!success) {
|
|
50
|
+
return {
|
|
51
|
+
status: 'failed',
|
|
52
|
+
txHash: params.txHash,
|
|
53
|
+
blockNumber: txBlock,
|
|
54
|
+
confirmations: confirmationCount,
|
|
55
|
+
gasUsed: BigInt(receipt.gasUsed),
|
|
56
|
+
success: false,
|
|
57
|
+
error: 'Transaction reverted (status=0)',
|
|
58
|
+
};
|
|
59
|
+
}
|
|
60
|
+
if (confirmationCount < requiredConfirmations) {
|
|
61
|
+
return {
|
|
62
|
+
status: 'pending',
|
|
63
|
+
txHash: params.txHash,
|
|
64
|
+
blockNumber: txBlock,
|
|
65
|
+
confirmations: confirmationCount,
|
|
66
|
+
gasUsed: BigInt(receipt.gasUsed),
|
|
67
|
+
success: true,
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
return {
|
|
71
|
+
status: 'confirmed',
|
|
72
|
+
txHash: params.txHash,
|
|
73
|
+
blockNumber: txBlock,
|
|
74
|
+
confirmations: confirmationCount,
|
|
75
|
+
gasUsed: BigInt(receipt.gasUsed),
|
|
76
|
+
success: true,
|
|
77
|
+
};
|
|
78
|
+
}
|
|
79
|
+
catch (err) {
|
|
80
|
+
return {
|
|
81
|
+
status: 'not_found',
|
|
82
|
+
txHash: params.txHash,
|
|
83
|
+
blockNumber: 0n,
|
|
84
|
+
confirmations: 0,
|
|
85
|
+
error: err instanceof Error ? err.message : String(err),
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
// ─── Solana Transaction Verification ───
|
|
90
|
+
/**
|
|
91
|
+
* Verify that a Solana transaction settled on-chain.
|
|
92
|
+
*
|
|
93
|
+
* @param params.signature - Solana transaction signature (base58)
|
|
94
|
+
* @param params.commitment - Commitment level: 'confirmed' or 'finalized' (default: 'confirmed')
|
|
95
|
+
*/
|
|
96
|
+
async verifySolanaTransaction(params) {
|
|
97
|
+
const commitment = params.commitment ?? 'confirmed';
|
|
98
|
+
try {
|
|
99
|
+
const resp = await this._solanaRpc('getTransaction', [
|
|
100
|
+
params.signature,
|
|
101
|
+
{ encoding: 'json', commitment, maxSupportedTransactionVersion: 0 },
|
|
102
|
+
]);
|
|
103
|
+
const tx = resp.result;
|
|
104
|
+
if (!tx) {
|
|
105
|
+
return {
|
|
106
|
+
status: 'not_found',
|
|
107
|
+
txHash: params.signature,
|
|
108
|
+
blockNumber: 0n,
|
|
109
|
+
confirmations: 0,
|
|
110
|
+
error: 'Transaction not found or not yet confirmed',
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
const success = tx.meta.err === null;
|
|
114
|
+
// Get current slot for confirmation count approximation
|
|
115
|
+
const slotResp = await this._solanaRpc('getSlot', [{ commitment }]);
|
|
116
|
+
const currentSlot = slotResp.result;
|
|
117
|
+
const confirmations = currentSlot - tx.slot;
|
|
118
|
+
return {
|
|
119
|
+
status: success ? 'confirmed' : 'failed',
|
|
120
|
+
txHash: params.signature,
|
|
121
|
+
blockNumber: BigInt(tx.slot),
|
|
122
|
+
confirmations,
|
|
123
|
+
success,
|
|
124
|
+
error: !success ? `Solana transaction error: ${JSON.stringify(tx.meta.err)}` : undefined,
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
catch (err) {
|
|
128
|
+
return {
|
|
129
|
+
status: 'not_found',
|
|
130
|
+
txHash: params.signature,
|
|
131
|
+
blockNumber: 0n,
|
|
132
|
+
confirmations: 0,
|
|
133
|
+
error: err instanceof Error ? err.message : String(err),
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
// ─── CCTP Bridge Settlement ───
|
|
138
|
+
/**
|
|
139
|
+
* Verify a CCTP bridge transfer settled end-to-end:
|
|
140
|
+
* 1. Verifies source chain burn tx succeeded
|
|
141
|
+
* 2. Checks Circle attestation API for the message hash
|
|
142
|
+
* 3. Attempts to find destination mint tx (if attestation is complete)
|
|
143
|
+
*
|
|
144
|
+
* @param params.sourceTxHash - Burn tx hash on source chain
|
|
145
|
+
* @param params.sourceChain - Source chain name (e.g. 'solana', 'base')
|
|
146
|
+
* @param params.destinationChain - Destination chain name
|
|
147
|
+
* @param params.expectedAmount - Expected USDC amount in base units (6 decimals)
|
|
148
|
+
*/
|
|
149
|
+
async verifyBridgeSettlement(params) {
|
|
150
|
+
// Step 1: verify source tx
|
|
151
|
+
let sourceTx;
|
|
152
|
+
if (params.sourceChain === 'solana') {
|
|
153
|
+
sourceTx = await this.verifySolanaTransaction({ signature: params.sourceTxHash });
|
|
154
|
+
}
|
|
155
|
+
else {
|
|
156
|
+
sourceTx = await this.verifyEvmTransaction({
|
|
157
|
+
chain: params.sourceChain,
|
|
158
|
+
txHash: params.sourceTxHash,
|
|
159
|
+
confirmations: params.sourceChain === 'ethereum' ? 65 : 1, // Ethereum needs more for CCTP
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
if (sourceTx.status === 'failed') {
|
|
163
|
+
return {
|
|
164
|
+
status: 'source_failed',
|
|
165
|
+
sourceTx,
|
|
166
|
+
error: 'Source chain burn transaction failed or reverted',
|
|
167
|
+
};
|
|
168
|
+
}
|
|
169
|
+
if (sourceTx.status === 'pending' || sourceTx.status === 'not_found') {
|
|
170
|
+
return {
|
|
171
|
+
status: 'source_pending',
|
|
172
|
+
sourceTx,
|
|
173
|
+
};
|
|
174
|
+
}
|
|
175
|
+
// Step 2: check Circle attestation API
|
|
176
|
+
let attestationStatus;
|
|
177
|
+
try {
|
|
178
|
+
attestationStatus = await this._checkCircleAttestation(params.sourceTxHash);
|
|
179
|
+
}
|
|
180
|
+
catch {
|
|
181
|
+
attestationStatus = 'error';
|
|
182
|
+
}
|
|
183
|
+
if (attestationStatus === 'pending_confirmations') {
|
|
184
|
+
return {
|
|
185
|
+
status: 'source_confirmed_awaiting_attestation',
|
|
186
|
+
sourceTx,
|
|
187
|
+
attestationStatus,
|
|
188
|
+
};
|
|
189
|
+
}
|
|
190
|
+
if (attestationStatus === 'complete') {
|
|
191
|
+
// Step 3: Try to find the destination mint tx
|
|
192
|
+
// For now, we report attestation complete and let the caller poll for mint
|
|
193
|
+
// A full implementation would index destination chain events
|
|
194
|
+
return {
|
|
195
|
+
status: 'attestation_complete_awaiting_mint',
|
|
196
|
+
sourceTx,
|
|
197
|
+
attestationStatus,
|
|
198
|
+
};
|
|
199
|
+
}
|
|
200
|
+
// If we can verify source + attestation but not destination, return partial
|
|
201
|
+
return {
|
|
202
|
+
status: 'error',
|
|
203
|
+
sourceTx,
|
|
204
|
+
attestationStatus,
|
|
205
|
+
error: 'Could not determine attestation status',
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
// ─── x402 Payment Verification ───
|
|
209
|
+
/**
|
|
210
|
+
* Verify an x402 payment settled on-chain by inspecting ERC20 Transfer event logs.
|
|
211
|
+
*
|
|
212
|
+
* Checks that the tx:
|
|
213
|
+
* - Succeeded (status=1)
|
|
214
|
+
* - Emitted an ERC20 Transfer event to expectedPayee
|
|
215
|
+
* - The transferred amount matches expectedAmount
|
|
216
|
+
*
|
|
217
|
+
* @param params.chain - EVM chain name
|
|
218
|
+
* @param params.txHash - Transaction hash
|
|
219
|
+
* @param params.expectedPayee - Address that should have received the payment
|
|
220
|
+
* @param params.expectedAmount - Amount in USDC base units (6 decimals)
|
|
221
|
+
*/
|
|
222
|
+
async verifyX402Payment(params) {
|
|
223
|
+
const rpcUrl = this._getRpcUrl(params.chain);
|
|
224
|
+
try {
|
|
225
|
+
const receiptResp = await this._evmRpc(rpcUrl, 'eth_getTransactionReceipt', [params.txHash]);
|
|
226
|
+
const receipt = receiptResp.result;
|
|
227
|
+
if (!receipt) {
|
|
228
|
+
return {
|
|
229
|
+
status: 'pending',
|
|
230
|
+
txHash: params.txHash,
|
|
231
|
+
payeeVerified: false,
|
|
232
|
+
amountVerified: false,
|
|
233
|
+
};
|
|
234
|
+
}
|
|
235
|
+
if (receipt.status !== '0x1') {
|
|
236
|
+
return {
|
|
237
|
+
status: 'failed',
|
|
238
|
+
txHash: params.txHash,
|
|
239
|
+
payeeVerified: false,
|
|
240
|
+
amountVerified: false,
|
|
241
|
+
error: 'Transaction reverted',
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
// Find ERC20 Transfer events — topic[0] = Transfer sig, topic[2] = to address
|
|
245
|
+
const expectedPayeeHex = params.expectedPayee.toLowerCase().replace('0x', '').padStart(64, '0');
|
|
246
|
+
const transferLog = receipt.logs.find((log) => {
|
|
247
|
+
return (log.topics[0]?.toLowerCase() === ERC20_TRANSFER_TOPIC &&
|
|
248
|
+
log.topics[2]?.toLowerCase() === `0x${expectedPayeeHex}`);
|
|
249
|
+
});
|
|
250
|
+
if (!transferLog) {
|
|
251
|
+
return {
|
|
252
|
+
status: 'confirmed',
|
|
253
|
+
txHash: params.txHash,
|
|
254
|
+
payeeVerified: false,
|
|
255
|
+
amountVerified: false,
|
|
256
|
+
error: `No ERC20 Transfer found to payee ${params.expectedPayee}`,
|
|
257
|
+
};
|
|
258
|
+
}
|
|
259
|
+
// Decode amount from log.data (ABI-encoded uint256)
|
|
260
|
+
const actualAmount = BigInt(transferLog.data);
|
|
261
|
+
const amountVerified = actualAmount === params.expectedAmount;
|
|
262
|
+
const actualPayee = `0x${transferLog.topics[2]?.slice(-40)}`;
|
|
263
|
+
const tokenAddress = transferLog.address;
|
|
264
|
+
return {
|
|
265
|
+
status: 'confirmed',
|
|
266
|
+
txHash: params.txHash,
|
|
267
|
+
payeeVerified: true,
|
|
268
|
+
amountVerified,
|
|
269
|
+
actualAmount,
|
|
270
|
+
actualPayee,
|
|
271
|
+
tokenAddress,
|
|
272
|
+
error: amountVerified
|
|
273
|
+
? undefined
|
|
274
|
+
: `Amount mismatch: expected ${params.expectedAmount}, got ${actualAmount}`,
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
catch (err) {
|
|
278
|
+
return {
|
|
279
|
+
status: 'not_found',
|
|
280
|
+
txHash: params.txHash,
|
|
281
|
+
payeeVerified: false,
|
|
282
|
+
amountVerified: false,
|
|
283
|
+
error: err instanceof Error ? err.message : String(err),
|
|
284
|
+
};
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
// ─── Internal Helpers ───
|
|
288
|
+
/** Get the RPC URL for a chain, preferring Alchemy over public RPCs */
|
|
289
|
+
_getRpcUrl(chain) {
|
|
290
|
+
if (this.alchemyApiKey && chain in SETTLEMENT_ALCHEMY_NETWORKS) {
|
|
291
|
+
const slug = SETTLEMENT_ALCHEMY_NETWORKS[chain];
|
|
292
|
+
return `https://${slug}.g.alchemy.com/v2/${this.alchemyApiKey}`;
|
|
293
|
+
}
|
|
294
|
+
const publicRpc = SETTLEMENT_PUBLIC_RPCS[chain];
|
|
295
|
+
if (!publicRpc) {
|
|
296
|
+
throw new Error(`No RPC URL available for chain "${chain}"`);
|
|
297
|
+
}
|
|
298
|
+
return publicRpc;
|
|
299
|
+
}
|
|
300
|
+
/** Make an EVM JSON-RPC call */
|
|
301
|
+
async _evmRpc(url, method, params) {
|
|
302
|
+
const resp = await fetch(url, {
|
|
303
|
+
method: 'POST',
|
|
304
|
+
headers: { 'Content-Type': 'application/json' },
|
|
305
|
+
body: JSON.stringify({ jsonrpc: '2.0', id: 1, method, params }),
|
|
306
|
+
});
|
|
307
|
+
if (!resp.ok) {
|
|
308
|
+
throw new Error(`EVM RPC error: HTTP ${resp.status}`);
|
|
309
|
+
}
|
|
310
|
+
const json = (await resp.json());
|
|
311
|
+
if (json.error)
|
|
312
|
+
throw new Error(`EVM RPC error: ${json.error.message}`);
|
|
313
|
+
return json;
|
|
314
|
+
}
|
|
315
|
+
/** Make a Solana JSON-RPC call */
|
|
316
|
+
async _solanaRpc(method, params) {
|
|
317
|
+
const resp = await fetch(SOLANA_RPC_URL, {
|
|
318
|
+
method: 'POST',
|
|
319
|
+
headers: { 'Content-Type': 'application/json' },
|
|
320
|
+
body: JSON.stringify({ jsonrpc: '2.0', id: 1, method, params }),
|
|
321
|
+
});
|
|
322
|
+
if (!resp.ok) {
|
|
323
|
+
throw new Error(`Solana RPC error: HTTP ${resp.status}`);
|
|
324
|
+
}
|
|
325
|
+
const json = (await resp.json());
|
|
326
|
+
if (json.error)
|
|
327
|
+
throw new Error(`Solana RPC error: ${json.error.message}`);
|
|
328
|
+
return json;
|
|
329
|
+
}
|
|
330
|
+
/** Check Circle attestation API for a CCTP burn message */
|
|
331
|
+
async _checkCircleAttestation(burnTxHash) {
|
|
332
|
+
// Circle attestation API: GET /attestations/{messageHash}
|
|
333
|
+
// For CCTP, the messageHash is derived from the MessageSent event in the burn tx.
|
|
334
|
+
// Since we don't have the full event parsing here, we query by tx hash pattern.
|
|
335
|
+
// In production, callers should provide the message hash from the burn tx receipt.
|
|
336
|
+
const url = `${CIRCLE_ATTESTATION_API}/${burnTxHash}`;
|
|
337
|
+
const resp = await fetch(url, {
|
|
338
|
+
headers: { Accept: 'application/json' },
|
|
339
|
+
});
|
|
340
|
+
if (resp.status === 404) {
|
|
341
|
+
return 'pending_confirmations';
|
|
342
|
+
}
|
|
343
|
+
if (!resp.ok) {
|
|
344
|
+
return 'error';
|
|
345
|
+
}
|
|
346
|
+
const json = (await resp.json());
|
|
347
|
+
if (json.status === 'complete')
|
|
348
|
+
return 'complete';
|
|
349
|
+
if (json.status === 'pending_confirmations')
|
|
350
|
+
return 'pending_confirmations';
|
|
351
|
+
return 'error';
|
|
352
|
+
}
|
|
353
|
+
}
|
|
354
|
+
//# sourceMappingURL=verifier.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"verifier.js","sourceRoot":"","sources":["../../src/settlement/verifier.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
import { Connection, PublicKey, Keypair, type TransactionSignature } from '@solana/web3.js';
|
|
2
|
+
import type { BridgeChain } from '../bridge/types.js';
|
|
3
|
+
import { SOLANA_CCTP_PROGRAMS, SOLANA_USDC_MINT, SOLANA_CCTP_DOMAIN } from '../bridge/types.js';
|
|
4
|
+
export type { BridgeChain };
|
|
5
|
+
export interface SolanaBurnResult {
|
|
6
|
+
/** Solana transaction signature of the burn tx */
|
|
7
|
+
signature: TransactionSignature;
|
|
8
|
+
/** The CCTP message hash (keccak256 of message bytes) */
|
|
9
|
+
messageHash: string;
|
|
10
|
+
/** Raw CCTP message bytes (hex-encoded) */
|
|
11
|
+
messageBytes: string;
|
|
12
|
+
/** Source domain (Solana = 5) */
|
|
13
|
+
sourceDomain: number;
|
|
14
|
+
/** Destination domain ID */
|
|
15
|
+
destinationDomain: number;
|
|
16
|
+
/** Nonce (derived from message bytes) */
|
|
17
|
+
nonce: bigint;
|
|
18
|
+
}
|
|
19
|
+
export interface SolanaBridgeParams {
|
|
20
|
+
/** Amount in USDC base units (6 decimals) */
|
|
21
|
+
amount: bigint;
|
|
22
|
+
/** EVM destination chain */
|
|
23
|
+
destinationChain: BridgeChain;
|
|
24
|
+
/** EVM destination address (hex) */
|
|
25
|
+
destinationAddress: string;
|
|
26
|
+
/** Apply 0.1% platform fee */
|
|
27
|
+
applyPlatformFee?: boolean;
|
|
28
|
+
/** Finality threshold (0 = fast, 1000 = finalized) */
|
|
29
|
+
minFinalityThreshold?: number;
|
|
30
|
+
/** Max fee (in USDC base units) for Circle's fee mechanism */
|
|
31
|
+
maxFee?: bigint;
|
|
32
|
+
/** Circle attestation API override */
|
|
33
|
+
attestationApiUrl?: string;
|
|
34
|
+
}
|
|
35
|
+
export interface SolanaReceiveParams {
|
|
36
|
+
/** CCTP message bytes (from EVM burn) */
|
|
37
|
+
message: Uint8Array | string;
|
|
38
|
+
/** Circle attestation bytes */
|
|
39
|
+
attestation: string;
|
|
40
|
+
/** Solana wallet to receive USDC (must have USDC ATA) */
|
|
41
|
+
recipientUsdcAccount?: string;
|
|
42
|
+
}
|
|
43
|
+
export interface SolanaBridgeResult {
|
|
44
|
+
/** Solana burn signature */
|
|
45
|
+
burnSignature: TransactionSignature;
|
|
46
|
+
/** EVM mint transaction hash */
|
|
47
|
+
evmMintTxHash: string;
|
|
48
|
+
/** Amount bridged */
|
|
49
|
+
amount: bigint;
|
|
50
|
+
/** Platform fee collected */
|
|
51
|
+
platformFee: bigint;
|
|
52
|
+
/** Source: Solana */
|
|
53
|
+
fromChain: 'solana';
|
|
54
|
+
/** Destination EVM chain */
|
|
55
|
+
toChain: BridgeChain;
|
|
56
|
+
/** Recipient address */
|
|
57
|
+
recipient: string;
|
|
58
|
+
/** Elapsed time (ms) */
|
|
59
|
+
elapsedMs: number;
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* SolanaCCTPBridge — Handles CCTP V2 burns and receives on Solana.
|
|
63
|
+
*
|
|
64
|
+
* For the full Solana→EVM or EVM→Solana flow, use UnifiedBridge which
|
|
65
|
+
* orchestrates both sides together.
|
|
66
|
+
*
|
|
67
|
+
* ```ts
|
|
68
|
+
* const bridge = new SolanaCCTPBridge({
|
|
69
|
+
* connection: new Connection('https://api.mainnet-beta.solana.com'),
|
|
70
|
+
* payer: keypair,
|
|
71
|
+
* });
|
|
72
|
+
*
|
|
73
|
+
* // Burn USDC on Solana → receive on Base
|
|
74
|
+
* const result = await bridge.bridgeToEvm({
|
|
75
|
+
* amount: 1_000_000n,
|
|
76
|
+
* destinationChain: 'base',
|
|
77
|
+
* destinationAddress: '0x...',
|
|
78
|
+
* });
|
|
79
|
+
* ```
|
|
80
|
+
*/
|
|
81
|
+
export declare class SolanaCCTPBridge {
|
|
82
|
+
private readonly connection;
|
|
83
|
+
private readonly payer;
|
|
84
|
+
private readonly tokenMessengerMinterProgram;
|
|
85
|
+
private readonly messageTransmitterProgram;
|
|
86
|
+
private readonly usdcMint;
|
|
87
|
+
constructor(config: {
|
|
88
|
+
connection: Connection;
|
|
89
|
+
payer: Keypair;
|
|
90
|
+
/** Override USDC mint (default: mainnet) */
|
|
91
|
+
usdcMint?: string;
|
|
92
|
+
});
|
|
93
|
+
/**
|
|
94
|
+
* Burn USDC on Solana and return burn result for attestation polling.
|
|
95
|
+
* Part 1 of the Solana→EVM bridge flow.
|
|
96
|
+
*
|
|
97
|
+
* @note The Solana burn instruction calls `depositForBurn` on the
|
|
98
|
+
* TokenMessengerMinterV2 program. The message is stored in a PDA
|
|
99
|
+
* (MessageSent account) rather than emitted as an EVM log event.
|
|
100
|
+
* Circle's IRIS API indexes these events by transaction signature.
|
|
101
|
+
*/
|
|
102
|
+
depositForBurn(params: {
|
|
103
|
+
amount: bigint;
|
|
104
|
+
destinationChain: BridgeChain;
|
|
105
|
+
destinationAddress: string;
|
|
106
|
+
minFinalityThreshold?: number;
|
|
107
|
+
maxFee?: bigint;
|
|
108
|
+
}): Promise<SolanaBurnResult>;
|
|
109
|
+
/**
|
|
110
|
+
* Poll Circle IRIS API for attestation after a Solana burn.
|
|
111
|
+
* Same API as EVM — IRIS indexes Solana transactions by message hash.
|
|
112
|
+
*/
|
|
113
|
+
waitForAttestation(messageHash: string, apiUrl?: string): Promise<string>;
|
|
114
|
+
/**
|
|
115
|
+
* Receive a CCTP message on Solana (EVM→Solana destination).
|
|
116
|
+
* Calls receiveMessage on the Solana MessageTransmitterV2 program.
|
|
117
|
+
*
|
|
118
|
+
* @param message - CCTP message bytes from EVM burn tx
|
|
119
|
+
* @param attestation - Circle attestation bytes
|
|
120
|
+
* @param recipientUsdcAccount - Solana USDC ATA for recipient (must exist)
|
|
121
|
+
*/
|
|
122
|
+
receiveMessage(params: SolanaReceiveParams): Promise<TransactionSignature>;
|
|
123
|
+
/**
|
|
124
|
+
* Ensure the payer's USDC ATA exists (create it if not).
|
|
125
|
+
*/
|
|
126
|
+
ensureUsdcAta(owner?: PublicKey): Promise<PublicKey>;
|
|
127
|
+
/**
|
|
128
|
+
* Fetch the MessageSent account data and extract message bytes.
|
|
129
|
+
* On Solana, CCTP messages are stored in PDA accounts (not events).
|
|
130
|
+
*/
|
|
131
|
+
private fetchMessageSent;
|
|
132
|
+
/**
|
|
133
|
+
* Convert an EVM hex address to a 32-byte Buffer for Solana mintRecipient.
|
|
134
|
+
* Per Circle spec: pad the EVM address on the left with 12 zero bytes.
|
|
135
|
+
*/
|
|
136
|
+
private evmAddressToBytes32;
|
|
137
|
+
private sleep;
|
|
138
|
+
}
|
|
139
|
+
export declare class SolanaBridgeError extends Error {
|
|
140
|
+
readonly code: string;
|
|
141
|
+
constructor(code: string, message: string);
|
|
142
|
+
}
|
|
143
|
+
export { SOLANA_CCTP_PROGRAMS, SOLANA_USDC_MINT, SOLANA_CCTP_DOMAIN, };
|
|
144
|
+
//# sourceMappingURL=bridge.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"bridge.d.ts","sourceRoot":"","sources":["../../src/solana/bridge.ts"],"names":[],"mappings":"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"}
|