@dexterai/x402 3.9.0 → 3.10.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 +6 -2
- package/dist/adapters/index.cjs +1 -1
- package/dist/adapters/index.d.cts +1 -1
- package/dist/adapters/index.d.ts +1 -1
- package/dist/adapters/index.js +1 -1
- package/dist/batch-settlement/index.cjs +1 -1
- package/dist/batch-settlement/index.js +1 -1
- package/dist/client/index.cjs +1 -1
- package/dist/client/index.d.cts +1 -1
- package/dist/client/index.d.ts +1 -1
- package/dist/client/index.js +1 -1
- package/dist/{constants-qU-4U3L-.d.cts → constants-D41hDAG6.d.cts} +13 -1
- package/dist/{constants-qU-4U3L-.d.ts → constants-D41hDAG6.d.ts} +13 -1
- package/dist/react/index.cjs +1 -1
- package/dist/react/index.js +1 -1
- package/dist/server/index.cjs +3 -3
- package/dist/server/index.d.cts +1 -1
- package/dist/server/index.d.ts +1 -1
- package/dist/server/index.js +3 -3
- package/dist/tab/adapters/solana/index.cjs +1 -0
- package/dist/tab/adapters/solana/index.d.cts +124 -0
- package/dist/tab/adapters/solana/index.d.ts +124 -0
- package/dist/tab/adapters/solana/index.js +1 -0
- package/dist/tab/index.cjs +6 -0
- package/dist/tab/index.d.cts +31 -0
- package/dist/tab/index.d.ts +31 -0
- package/dist/tab/index.js +6 -0
- package/dist/tab/seller/index.cjs +6 -0
- package/dist/tab/seller/index.d.cts +291 -0
- package/dist/tab/seller/index.d.ts +291 -0
- package/dist/tab/seller/index.js +6 -0
- package/dist/types-DIrmhiD-.d.cts +234 -0
- package/dist/types-DIrmhiD-.d.ts +234 -0
- package/dist/utils/index.cjs +1 -1
- package/dist/utils/index.d.cts +10 -1
- package/dist/utils/index.d.ts +10 -1
- package/dist/utils/index.js +1 -1
- package/package.json +18 -1
package/README.md
CHANGED
|
@@ -28,6 +28,8 @@ The audience this is built for in 2026 is **agents**: Claude, ChatGPT, Cursor, a
|
|
|
28
28
|
|
|
29
29
|
You call `payAndFetch()` on the client. You add `x402Middleware()` on the server. Payments happen.
|
|
30
30
|
|
|
31
|
+
Built against the official x402 v1 and v2 specs. Adds the multi-chain buyer and seller surface, the React hook, a discriminated `PayResult` type, and batch-settlement channels for high-frequency calls.
|
|
32
|
+
|
|
31
33
|
---
|
|
32
34
|
|
|
33
35
|
## Quick start
|
|
@@ -136,6 +138,8 @@ Batch settlement lets a buyer pre-fund an escrow channel once, make many **discr
|
|
|
136
138
|
|
|
137
139
|
It is **not** a streaming primitive; it batches discrete purchases. EVM only (Base, Arbitrum, Polygon). The buyer never needs a gas token: every step (deposit, voucher, claim, settle, refund) is signature-based; the Dexter facilitator submits the transactions and pays the gas.
|
|
138
140
|
|
|
141
|
+
> **Coming: `@dexterai/x402/tab` — OTS-backed streaming payments.** A peer module for *continuous metered consumption* (tokens, bytes, frames, seconds) where the unit of billing is smaller than a request. One passkey-authorized session signs many vouchers; the seller verifies locally and demands a fresh signature before each chunk. The contract is locked (`@dexterai/x402/tab` + `@dexterai/x402/tab/seller` subpaths emit type stubs today); implementation is phased in [`docs/DESIGN-tab-streaming.md`](./docs/DESIGN-tab-streaming.md). Architectural roadmap on [github.com/Dexter-DAO/dexter-vault/issues](https://github.com/Dexter-DAO/dexter-vault/issues?q=is%3Aissue+label%3Aroadmap).
|
|
142
|
+
|
|
139
143
|
### Buyer
|
|
140
144
|
|
|
141
145
|
```ts
|
|
@@ -485,7 +489,7 @@ Several v1-era helpers ship with `@deprecated` markers in 3.9. They keep working
|
|
|
485
489
|
| `stripePayTo` | No replacement in the SDK. Integrate Stripe at your application layer if needed. |
|
|
486
490
|
| `x402BrowserSupport` | No replacement. Build a custom paywall page if you need one. |
|
|
487
491
|
|
|
488
|
-
|
|
492
|
+
None of these will be removed in 3.x.
|
|
489
493
|
|
|
490
494
|
---
|
|
491
495
|
|
|
@@ -580,7 +584,7 @@ MIT. See [LICENSE](./LICENSE).
|
|
|
580
584
|
|
|
581
585
|
<p align="center">
|
|
582
586
|
<a href="https://x402.dexter.cash">Dexter Facilitator</a> ·
|
|
583
|
-
<a href="https://dexter.cash/opendexter">OpenDexter
|
|
587
|
+
<a href="https://dexter.cash/opendexter">OpenDexter Catalog</a> ·
|
|
584
588
|
<a href="https://dexter.cash/sdk">Live Demo</a> ·
|
|
585
589
|
<a href="https://dexter.cash/onboard">Become a Seller</a>
|
|
586
590
|
</p>
|
package/dist/adapters/index.cjs
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";var It=Object.create;var Q=Object.defineProperty;var wt=Object.getOwnPropertyDescriptor;var Ct=Object.getOwnPropertyNames;var _t=Object.getPrototypeOf,Pt=Object.prototype.hasOwnProperty;var xt=(c,t)=>{for(var e in t)Q(c,e,{get:t[e],enumerable:!0})},gt=(c,t,e,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of Ct(t))!Pt.call(c,r)&&r!==e&&Q(c,r,{get:()=>t[r],enumerable:!(o=wt(t,r))||o.enumerable});return c};var tt=(c,t,e)=>(e=c!=null?It(_t(c)):{},gt(t||!c||!c.__esModule?Q(e,"default",{value:c,enumerable:!0}):e,c)),bt=c=>gt(Q({},"__esModule",{value:!0}),c);var Lt={};xt(Lt,{ARBITRUM_ONE:()=>O,AVALANCHE:()=>U,BASE_MAINNET:()=>N,BASE_SEPOLIA:()=>X,BSC_MAINNET:()=>k,BSC_STABLECOIN_ADDRESSES:()=>Y,BSC_USDC:()=>G,BSC_USDT:()=>ut,ETHEREUM_MAINNET:()=>F,EvmAdapter:()=>Z,OPTIMISM:()=>L,PERMIT2_ADDRESS:()=>C,POLYGON:()=>M,SKALE_BASE:()=>W,SKALE_BASE_SEPOLIA:()=>K,SOLANA_DEVNET:()=>v,SOLANA_MAINNET:()=>b,SOLANA_TESTNET:()=>D,SolanaAdapter:()=>z,USDC_ADDRESSES:()=>J,X402_EXACT_PERMIT2_PROXY:()=>j,createDefaultAdapters:()=>Ot,createEvmAdapter:()=>ft,createSolanaAdapter:()=>mt,findAdapter:()=>Mt,isEvmWallet:()=>$,isKnownUSDC:()=>Dt,isSolanaWallet:()=>V});module.exports=bt(Lt);var m=require("@solana/web3.js"),f=require("@solana/spl-token");var At="solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp",Et="solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1",St="solana:4uhcVJyU9pJkvQyS88uRDiswHXSCkY3z",b=At,v=Et,D=St,et="eip155:8453",nt="eip155:84532",rt="eip155:42161",ot="eip155:137",st="eip155:10",at="eip155:43114",it="eip155:56",ct="eip155:1187947933",dt="eip155:324705682",pt="eip155:1",N=et,X=nt,O=rt,M=ot,L=st,U=at,k=it,W=ct,K=dt,F=pt,ht="EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v",Tt="4zMMC9srt5Ri5X14GAgXhaHii3GnPAEERYPJgZJDncDU",Nt="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",ut="0x55d398326f99059fF775485246999027B3197955",G="0x8AC76a51cc950d9822D68b83fE1Ad97B32Cd580d",J={[it]:G,[et]:Nt,[nt]:"0x036CbD53842c5426634e7929541eC2318f3dCF7e",[rt]:"0xaf88d065e77c8cC2239327C5EDb3A432268e5831",[ot]:"0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",[st]:"0x0b2C639c533813f4Aa9D7837CAf62653d097Ff85",[at]:"0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E",[ct]:"0x85889c8c714505E0c94b30fcfcF64fE3Ac8FCb20",[dt]:"0x2e08028E3C4c2356572E096d8EF835cD5C6030bD",[pt]:"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48"},Y={[ut]:{symbol:"USDT",decimals:18},[G]:{symbol:"USDC",decimals:18}};var C="0x000000000022D473030F116dDEE9F6B43aC78BA3",j="0x402085c248EeA27D92E8b30b2C58ed07f9E20001",lt={[it]:56,[et]:8453,[nt]:84532,[rt]:42161,[ot]:137,[st]:10,[at]:43114,[ct]:1187947933,[dt]:324705682,[pt]:1},B={[At]:"https://api.dexter.cash/api/solana/rpc",[Et]:"https://api.devnet.solana.com",[St]:"https://api.testnet.solana.com"},S={[it]:"https://bsc-dataseed1.binance.org",[et]:"https://api.dexter.cash/api/base/rpc",[nt]:"https://sepolia.base.org",[rt]:"https://arb1.arbitrum.io/rpc",[ot]:"https://polygon-rpc.com",[st]:"https://mainnet.optimism.io",[at]:"https://api.avax.network/ext/bc/C/rpc",[ct]:"https://skale-base.skalenodes.com/v1/base",[dt]:"https://base-sepolia-testnet.skalenodes.com/v1/jubilant-horrible-ancha",[pt]:"https://eth.llamarpc.com"};var Bt=12e3,Rt=1;function V(c){if(!c||typeof c!="object")return!1;let t=c;return"publicKey"in t&&"signTransaction"in t&&typeof t.signTransaction=="function"}var z=class{name="Solana";networks=[b,v,D];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:solana]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="solana"||t==="solana-devnet"||t==="solana-testnet"||t.startsWith("solana:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:B[t]?B[t]:t==="solana"?B[b]:t==="solana-devnet"?B[v]:t==="solana-testnet"?B[D]:B[b]}getAddress(t){return V(t)?t.publicKey?.toBase58()??null:null}isConnected(t){return V(t)?t.publicKey!==null:!1}async getBalance(t,e,o){if(!V(e)||!e.publicKey)return 0;let r=o||this.getDefaultRpcUrl(t.network),n=new m.Connection(r,"confirmed"),i=new m.PublicKey(e.publicKey.toBase58()),a=new m.PublicKey(t.asset);try{let p=(await n.getAccountInfo(a,"confirmed"))?.owner.toBase58()===f.TOKEN_2022_PROGRAM_ID.toBase58()?f.TOKEN_2022_PROGRAM_ID:f.TOKEN_PROGRAM_ID,d=await(0,f.getAssociatedTokenAddress)(a,i,!1,p),l=await(0,f.getAccount)(n,d,void 0,p),u=t.extra?.decimals??6;return Number(l.amount)/Math.pow(10,u)}catch(s){if(s&&typeof s=="object"&&"name"in s&&(s.name==="TokenAccountNotFoundError"||s.name==="TokenInvalidAccountOwnerError"))return 0;throw s}}async buildTransaction(t,e,o){if(!V(e))throw new Error("Invalid Solana wallet");if(!e.publicKey)throw new Error("Wallet not connected");let r=o||this.getDefaultRpcUrl(t.network),n=new m.Connection(r,"confirmed"),i=new m.PublicKey(e.publicKey.toBase58()),{payTo:a,asset:s,extra:p}=t,d=t.amount??t.maxAmountRequired;if(!d)throw new Error("Missing amount in payment requirements");if(!p?.feePayer)throw new Error("Missing feePayer in payment requirements");let l=new m.PublicKey(p.feePayer),u=new m.PublicKey(s),g=new m.PublicKey(a);this.log("Building transaction:",{from:i.toBase58(),to:a,amount:d,asset:s,feePayer:p.feePayer});let A=[];A.push(m.ComputeBudgetProgram.setComputeUnitLimit({units:Bt})),A.push(m.ComputeBudgetProgram.setComputeUnitPrice({microLamports:Rt}));let h=await n.getAccountInfo(u,"confirmed");if(!h)throw new Error(`Token mint ${s} not found`);let E=h.owner.toBase58()===f.TOKEN_2022_PROGRAM_ID.toBase58()?f.TOKEN_2022_PROGRAM_ID:f.TOKEN_PROGRAM_ID,I=await(0,f.getMint)(n,u,void 0,E);typeof p?.decimals=="number"&&I.decimals!==p.decimals&&this.log(`Decimals mismatch: requirements say ${p.decimals}, mint says ${I.decimals}`);let T=await(0,f.getAssociatedTokenAddress)(u,i,!1,E),y=await(0,f.getAssociatedTokenAddress)(u,g,!0,E);if(!await n.getAccountInfo(T,"confirmed"))throw new Error(`No token account found for ${s}. Please ensure you have USDC in your wallet.`);if(!await n.getAccountInfo(y,"confirmed"))throw new Error(`Seller token account not found. The seller (${a}) must have a USDC account.`);let q=BigInt(d);A.push((0,f.createTransferCheckedInstruction)(T,u,y,i,q,I.decimals,[],E));let{blockhash:x}=await n.getLatestBlockhash("confirmed"),H=new m.TransactionMessage({payerKey:l,recentBlockhash:x,instructions:A}).compileToV0Message(),_=new m.VersionedTransaction(H),R=await e.signTransaction(_);return this.log("Transaction signed successfully"),{serialized:Buffer.from(R.serialize()).toString("base64"),settlementProbe:{kind:"solana",sourceAta:T.toBase58(),destinationAta:y.toBase58(),asset:s,amount:d,blockhash:x}}}async confirmSettlement(t,e){if(t.kind!=="solana")throw new Error(`SolanaAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`);let o=new m.Connection(e,"confirmed"),r=new m.PublicKey(t.destinationAta),n=await o.getSignaturesForAddress(r,{limit:25});if(n.length===0)return{settled:!1};let i=BigInt(t.amount);for(let a of n){if(a.err)continue;let s=await o.getTransaction(a.signature,{commitment:"confirmed",maxSupportedTransactionVersion:0});if(!s?.meta)continue;let p=s.transaction.message.getAccountKeys().keySegments().flat(),d=s.meta.preTokenBalances??[],l=s.meta.postTokenBalances??[];for(let u of l){if(u.mint!==t.asset)continue;let g=p[u.accountIndex];if(!g||!g.equals(r))continue;let A=d.find(I=>I.accountIndex===u.accountIndex),h=BigInt(A?.uiTokenAmount.amount??"0");if(BigInt(u.uiTokenAmount.amount??"0")-h===i)return{settled:!0,txSignature:a.signature}}}return{settled:!1}}};function mt(c){return new z(c)}var vt={PermitWitnessTransferFrom:[{name:"permitted",type:"TokenPermissions"},{name:"spender",type:"address"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"witness",type:"Witness"}],TokenPermissions:[{name:"token",type:"address"},{name:"amount",type:"uint256"}],Witness:[{name:"to",type:"address"},{name:"validAfter",type:"uint256"}]},yt=BigInt("0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff");function $(c){if(!c||typeof c!="object")return!1;let t=c;return"address"in t&&typeof t.address=="string"&&t.address.startsWith("0x")}var Z=class{name="EVM";networks=[k,N,X,F,O,M,L,U,W,K];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:evm]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="base"||t==="bsc"||t==="ethereum"||t==="arbitrum"||t==="polygon"||t==="optimism"||t==="avalanche"||t==="skale-base"||t==="skale-base-sepolia"||t.startsWith("eip155:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:S[t]?S[t]:t==="base"?S[N]:t==="bsc"?S[k]:t==="ethereum"?S[F]:t==="arbitrum"?S[O]:t==="polygon"?S[M]:t==="optimism"?S[L]:t==="avalanche"?S[U]:t==="skale-base"?S[W]:t==="skale-base-sepolia"?S[K]:S[N]}getAddress(t){return $(t)?t.address:null}isConnected(t){return $(t)?!!t.address:!1}getChainId(t){if(lt[t])return lt[t];if(t.startsWith("eip155:")){let e=t.split(":")[1];return parseInt(e,10)}return t==="base"?8453:t==="bsc"?56:t==="ethereum"?1:t==="arbitrum"?42161:8453}async getBalance(t,e,o){if(!$(e)||!e.address)return 0;let r=o||this.getDefaultRpcUrl(t.network);try{let n=this.encodeBalanceOf(e.address),i=await fetch(r,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:t.asset,data:n},"latest"]})});if(!i.ok)throw new Error(`RPC request failed: ${i.status}`);let a=await i.json();if(a.error)throw new Error(`RPC error: ${JSON.stringify(a.error)}`);if(!a.result||a.result==="0x")return 0;let s=BigInt(a.result),p=t.extra?.decimals??6;return Number(s)/Math.pow(10,p)}catch(n){throw n}}encodeBalanceOf(t){let e="0x70a08231",o=t.slice(2).toLowerCase().padStart(64,"0");return e+o}async confirmSettlement(t,e){if(t.kind==="eip3009"){let o="0xe94a0102",r=t.from.slice(2).toLowerCase().padStart(64,"0"),n=t.nonce.slice(2).toLowerCase().padStart(64,"0"),i=o+r+n,a=await this.ethCall(e,t.asset,i);return{settled:BigInt(a)!==0n}}if(t.kind==="permit2"){let o="0x4fe02b44",r=BigInt(t.nonce),n=r>>8n,i=r&0xffn,a=t.from.slice(2).toLowerCase().padStart(64,"0"),s=n.toString(16).padStart(64,"0"),p=o+a+s,d=await this.ethCall(e,C,p);return{settled:(BigInt(d)>>i&1n)===1n}}throw new Error(`EvmAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`)}async ethCall(t,e,o){let r=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:o},"latest"]})});if(!r.ok)throw new Error(`RPC request failed: ${r.status}`);let n=await r.json();if(n.error)throw new Error(`RPC error: ${JSON.stringify(n.error)}`);if(!n.result||n.result==="0x")throw new Error("RPC returned an empty result");return n.result}async buildTransaction(t,e,o){if(!$(e))throw new Error("Invalid EVM wallet");if(!e.address)throw new Error("Wallet not connected");if(t.scheme==="exact-approval")return this.buildApprovalTransaction(t,e,o);if(t.extra?.assetTransferMethod==="permit2")return this.buildPermit2Transaction(t,e,o);let{payTo:r,asset:n,extra:i}=t,a=t.amount??t.maxAmountRequired;if(!a)throw new Error("Missing amount in payment requirements");this.log("Building EVM transaction:",{from:e.address,to:r,amount:a,asset:n,network:t.network});let s=this.getChainId(t.network),p={name:i?.name??"USD Coin",version:i?.version??"2",chainId:BigInt(s),verifyingContract:n},d={TransferWithAuthorization:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"value",type:"uint256"},{name:"validAfter",type:"uint256"},{name:"validBefore",type:"uint256"},{name:"nonce",type:"bytes32"}]},l=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(l);let u="0x"+[...l].map(T=>T.toString(16).padStart(2,"0")).join(""),g=Math.floor(Date.now()/1e3),A={from:e.address,to:r,value:a,validAfter:String(g-600),validBefore:String(g+(t.maxTimeoutSeconds||60)),nonce:u},h={from:e.address,to:r,value:BigInt(a),validAfter:BigInt(g-600),validBefore:BigInt(g+(t.maxTimeoutSeconds||60)),nonce:u};if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");let E=await e.signTypedData({domain:p,types:d,primaryType:"TransferWithAuthorization",message:h});return this.log("EIP-712 signature obtained"),{serialized:JSON.stringify({authorization:A,signature:E}),signature:E,settlementProbe:{kind:"eip3009",from:e.address,nonce:u,asset:n,chainId:s}}}async buildApprovalTransaction(t,e,o){let{payTo:r,asset:n,extra:i}=t,a=t.amount??t.maxAmountRequired;if(!a)throw new Error("Missing amount in payment requirements");let s=i?.facilitatorContract;if(!s)throw new Error("exact-approval scheme requires extra.facilitatorContract from the facilitator. The /supported endpoint should provide this.");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building approval-based transaction:",{from:e.address,to:r,amount:a,asset:n,network:t.network,facilitatorContract:s});let p=o||this.getDefaultRpcUrl(t.network),d=i?.fee??"0",l=BigInt(a)+BigInt(d),u=await this.readAllowance(p,n,e.address,s);if(u<l){if(!e.sendTransaction)throw new Error("BSC payments require a wallet that supports sendTransaction for the one-time token approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.");let _=this.calculateApprovalAmount(a,d,i?.approvalStrategy);this.log(`Approving ${_} for ${s} (current allowance: ${u})`);let R=await e.sendTransaction({to:n,data:this.encodeApprove(s,_),value:0n});this.log(`Approval tx sent: ${R}`),await this.waitForReceipt(p,R),this.log("Approval confirmed")}else this.log("Sufficient allowance, skipping approval");let g=new Uint8Array(16);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(g);let A=[...g].reduce((_,R)=>_*256n+BigInt(R),0n).toString(),h=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(h);let E="0x"+[...h].map(_=>_.toString(16).padStart(2,"0")).join(""),T=Math.floor(Date.now()/1e3)+(t.maxTimeoutSeconds||300),y=i?.eip712Domain,P=y?{name:y.name,version:y.version,chainId:BigInt(y.chainId),verifyingContract:y.verifyingContract}:{name:"DexterBSCFacilitator",version:"1",chainId:BigInt(this.getChainId(t.network)),verifyingContract:s},w=i?.eip712Types??{Payment:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"token",type:"address"},{name:"amount",type:"uint256"},{name:"fee",type:"uint256"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"paymentId",type:"bytes32"}]},q={from:e.address,to:r,token:n,amount:BigInt(a),fee:BigInt(d),nonce:BigInt(A),deadline:BigInt(T),paymentId:E},x=await e.signTypedData({domain:P,types:w,primaryType:"Payment",message:q});this.log("EIP-712 Payment signature obtained");let H={from:e.address,to:r,token:n,amount:a,fee:d,nonce:A,deadline:T,paymentId:E,signature:x};return{serialized:JSON.stringify(H),signature:x}}async buildPermit2Transaction(t,e,o){let{payTo:r,asset:n}=t,i=t.amount??t.maxAmountRequired;if(!i)throw new Error("Missing amount in payment requirements");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building Permit2 transaction:",{from:e.address,to:r,amount:i,asset:n,network:t.network});let a=o||this.getDefaultRpcUrl(t.network),s=await this.readAllowance(a,n,e.address,C),p;if(s<BigInt(i)){let P=this.encodeApprove(C,yt);if(e.signTransaction){this.log(`Signing Permit2 approval for relay (current allowance: ${s})`);let w=this.getChainId(t.network),q=await this.readGasPrice(a),x=await this.readNonce(a,e.address),H=await e.signTransaction({to:n,data:P,chainId:w,gas:50000n,gasPrice:q,nonce:x});p={erc20ApprovalGasSponsoring:{info:{from:e.address,asset:n,spender:C,amount:yt.toString(),signedTransaction:H,version:"1"}}},this.log("Permit2 approval signed for facilitator relay")}else if(e.sendTransaction){this.log(`Approving Permit2 directly (current allowance: ${s})`);let w=await e.sendTransaction({to:n,data:P,value:0n});this.log(`Permit2 approval tx sent: ${w}`),await this.waitForReceipt(a,w),this.log("Permit2 approval confirmed")}else throw new Error("Permit2 payments require a wallet that supports signTransaction or sendTransaction for the one-time Permit2 approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.")}else this.log("Sufficient Permit2 allowance, skipping approval");let d=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(d);let l=[...d].reduce((P,w)=>P*256n+BigInt(w),0n),u=Math.floor(Date.now()/1e3),g=u-600,A=u+(t.maxTimeoutSeconds||300),h=this.getChainId(t.network),E={name:"Permit2",chainId:BigInt(h),verifyingContract:C},I={permitted:{token:n,amount:BigInt(i)},spender:j,nonce:l,deadline:BigInt(A),witness:{to:r,validAfter:BigInt(g)}},T=await e.signTypedData({domain:E,types:vt,primaryType:"PermitWitnessTransferFrom",message:I});this.log("Permit2 PermitWitnessTransferFrom signature obtained");let y={signature:T,permit2Authorization:{from:e.address,permitted:{token:n,amount:i},spender:j,nonce:l.toString(),deadline:String(A),witness:{to:r,validAfter:String(g)}}};return{serialized:JSON.stringify(y),signature:T,extensions:p,settlementProbe:{kind:"permit2",from:e.address,nonce:l.toString(),chainId:h}}}async readAllowance(t,e,o,r){let n="0xdd62ed3e",i=o.slice(2).toLowerCase().padStart(64,"0"),a=r.slice(2).toLowerCase().padStart(64,"0"),s=n+i+a;try{let d=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:s},"latest"]})})).json();return d.error||!d.result||d.result==="0x"?0n:BigInt(d.result)}catch{return 0n}}encodeApprove(t,e){let o="0x095ea7b3",r=t.slice(2).toLowerCase().padStart(64,"0"),n=e.toString(16).padStart(64,"0");return o+r+n}async waitForReceipt(t,e,o=3e4){let r=Date.now();for(;Date.now()-r<o;){try{let i=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionReceipt",params:[e]})})).json();if(i.result){if(i.result.status==="0x0")throw new Error(`Approval transaction reverted: ${e}`);return}}catch(n){if(n instanceof Error&&n.message.includes("reverted"))throw n}await new Promise(n=>setTimeout(n,2e3))}throw new Error(`Approval transaction receipt timeout after ${o}ms: ${e}`)}async readGasPrice(t){try{let o=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_gasPrice",params:[]})})).json();return o.result?BigInt(o.result):50000000n}catch{return 50000000n}}async readNonce(t,e){try{let r=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionCount",params:[e,"latest"]})})).json();return r.result?parseInt(r.result,16):0}catch{return 0}}calculateApprovalAmount(t,e,o){let r=BigInt(t)+BigInt(e);if(!o||o.mode==="exact")return r;let n=BigInt(o.defaultMultiple??10),i=r*n;if(o.maxCapUsd){let a=this.inferDecimals(t),s=BigInt(Math.floor(o.maxCapUsd*Math.pow(10,a)));if(i>s)return s}if(o.exactAboveUsd){let a=this.inferDecimals(t),s=BigInt(Math.floor(o.exactAboveUsd*Math.pow(10,a)));if(BigInt(t)>s)return r}return i}inferDecimals(t){return t.length>12?18:6}};function ft(c){return new Z(c)}function Dt(c){if(c===ht||c===Tt)return!0;let t=c.toLowerCase();for(let e of Object.values(J))if(e.toLowerCase()===t)return!0;for(let e of Object.keys(Y))if(e.toLowerCase()===t)return!0;return!1}function Ot(c=!1){return[mt({verbose:c}),ft({verbose:c})]}function Mt(c,t){return c.find(e=>e.canHandle(t))}0&&(module.exports={ARBITRUM_ONE,AVALANCHE,BASE_MAINNET,BASE_SEPOLIA,BSC_MAINNET,BSC_STABLECOIN_ADDRESSES,BSC_USDC,BSC_USDT,ETHEREUM_MAINNET,EvmAdapter,OPTIMISM,PERMIT2_ADDRESS,POLYGON,SKALE_BASE,SKALE_BASE_SEPOLIA,SOLANA_DEVNET,SOLANA_MAINNET,SOLANA_TESTNET,SolanaAdapter,USDC_ADDRESSES,X402_EXACT_PERMIT2_PROXY,createDefaultAdapters,createEvmAdapter,createSolanaAdapter,findAdapter,isEvmWallet,isKnownUSDC,isSolanaWallet});
|
|
1
|
+
"use strict";var It=Object.create;var Q=Object.defineProperty;var wt=Object.getOwnPropertyDescriptor;var Ct=Object.getOwnPropertyNames;var _t=Object.getPrototypeOf,Pt=Object.prototype.hasOwnProperty;var xt=(c,t)=>{for(var e in t)Q(c,e,{get:t[e],enumerable:!0})},gt=(c,t,e,o)=>{if(t&&typeof t=="object"||typeof t=="function")for(let r of Ct(t))!Pt.call(c,r)&&r!==e&&Q(c,r,{get:()=>t[r],enumerable:!(o=wt(t,r))||o.enumerable});return c};var tt=(c,t,e)=>(e=c!=null?It(_t(c)):{},gt(t||!c||!c.__esModule?Q(e,"default",{value:c,enumerable:!0}):e,c)),Nt=c=>gt(Q({},"__esModule",{value:!0}),c);var Lt={};xt(Lt,{ARBITRUM_ONE:()=>O,AVALANCHE:()=>U,BASE_MAINNET:()=>b,BASE_SEPOLIA:()=>X,BSC_MAINNET:()=>k,BSC_STABLECOIN_ADDRESSES:()=>Y,BSC_USDC:()=>G,BSC_USDT:()=>ut,ETHEREUM_MAINNET:()=>F,EvmAdapter:()=>Z,OPTIMISM:()=>L,PERMIT2_ADDRESS:()=>C,POLYGON:()=>M,SKALE_BASE:()=>W,SKALE_BASE_SEPOLIA:()=>K,SOLANA_DEVNET:()=>v,SOLANA_MAINNET:()=>N,SOLANA_TESTNET:()=>D,SolanaAdapter:()=>z,USDC_ADDRESSES:()=>J,X402_EXACT_PERMIT2_PROXY:()=>j,createDefaultAdapters:()=>Ot,createEvmAdapter:()=>ft,createSolanaAdapter:()=>mt,findAdapter:()=>Mt,isEvmWallet:()=>$,isKnownUSDC:()=>Dt,isSolanaWallet:()=>V});module.exports=Nt(Lt);var m=require("@solana/web3.js"),f=require("@solana/spl-token");var At="solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp",Et="solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1",St="solana:4uhcVJyU9pJkvQyS88uRDiswHXSCkY3z",N=At,v=Et,D=St,et="eip155:8453",nt="eip155:84532",rt="eip155:42161",ot="eip155:137",st="eip155:10",at="eip155:43114",it="eip155:56",ct="eip155:1187947933",dt="eip155:324705682",pt="eip155:1",b=et,X=nt,O=rt,M=ot,L=st,U=at,k=it,W=ct,K=dt,F=pt,ht="EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v",Tt="4zMMC9srt5Ri5X14GAgXhaHii3GnPAEERYPJgZJDncDU",bt="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",ut="0x55d398326f99059fF775485246999027B3197955",G="0x8AC76a51cc950d9822D68b83fE1Ad97B32Cd580d",J={[it]:G,[et]:bt,[nt]:"0x036CbD53842c5426634e7929541eC2318f3dCF7e",[rt]:"0xaf88d065e77c8cC2239327C5EDb3A432268e5831",[ot]:"0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",[st]:"0x0b2C639c533813f4Aa9D7837CAf62653d097Ff85",[at]:"0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E",[ct]:"0x85889c8c714505E0c94b30fcfcF64fE3Ac8FCb20",[dt]:"0x2e08028E3C4c2356572E096d8EF835cD5C6030bD",[pt]:"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48"},Y={[ut]:{symbol:"USDT",decimals:18},[G]:{symbol:"USDC",decimals:18}};var C="0x000000000022D473030F116dDEE9F6B43aC78BA3",j="0x402085c248EeA27D92E8b30b2C58ed07f9E20001",lt={[it]:56,[et]:8453,[nt]:84532,[rt]:42161,[ot]:137,[st]:10,[at]:43114,[ct]:1187947933,[dt]:324705682,[pt]:1},B={[At]:"https://api.dexter.cash/api/solana/rpc",[Et]:"https://api.devnet.solana.com",[St]:"https://api.testnet.solana.com"},S={[it]:"https://api.dexter.cash/api/evm/bsc/rpc",[et]:"https://api.dexter.cash/api/base/rpc",[nt]:"https://sepolia.base.org",[rt]:"https://api.dexter.cash/api/evm/arbitrum/rpc",[ot]:"https://api.dexter.cash/api/evm/polygon/rpc",[st]:"https://api.dexter.cash/api/evm/optimism/rpc",[at]:"https://api.dexter.cash/api/evm/avalanche/rpc",[ct]:"https://skale-base.skalenodes.com/v1/base",[dt]:"https://base-sepolia-testnet.skalenodes.com/v1/jubilant-horrible-ancha",[pt]:"https://eth.llamarpc.com"};var Bt=12e3,Rt=1;function V(c){if(!c||typeof c!="object")return!1;let t=c;return"publicKey"in t&&"signTransaction"in t&&typeof t.signTransaction=="function"}var z=class{name="Solana";networks=[N,v,D];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:solana]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="solana"||t==="solana-devnet"||t==="solana-testnet"||t.startsWith("solana:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:B[t]?B[t]:t==="solana"?B[N]:t==="solana-devnet"?B[v]:t==="solana-testnet"?B[D]:B[N]}getAddress(t){return V(t)?t.publicKey?.toBase58()??null:null}isConnected(t){return V(t)?t.publicKey!==null:!1}async getBalance(t,e,o){if(!V(e)||!e.publicKey)return 0;let r=o||this.getDefaultRpcUrl(t.network),n=new m.Connection(r,"confirmed"),i=new m.PublicKey(e.publicKey.toBase58()),a=new m.PublicKey(t.asset);try{let p=(await n.getAccountInfo(a,"confirmed"))?.owner.toBase58()===f.TOKEN_2022_PROGRAM_ID.toBase58()?f.TOKEN_2022_PROGRAM_ID:f.TOKEN_PROGRAM_ID,d=await(0,f.getAssociatedTokenAddress)(a,i,!1,p),l=await(0,f.getAccount)(n,d,void 0,p),u=t.extra?.decimals??6;return Number(l.amount)/Math.pow(10,u)}catch(s){if(s&&typeof s=="object"&&"name"in s&&(s.name==="TokenAccountNotFoundError"||s.name==="TokenInvalidAccountOwnerError"))return 0;throw s}}async buildTransaction(t,e,o){if(!V(e))throw new Error("Invalid Solana wallet");if(!e.publicKey)throw new Error("Wallet not connected");let r=o||this.getDefaultRpcUrl(t.network),n=new m.Connection(r,"confirmed"),i=new m.PublicKey(e.publicKey.toBase58()),{payTo:a,asset:s,extra:p}=t,d=t.amount??t.maxAmountRequired;if(!d)throw new Error("Missing amount in payment requirements");if(!p?.feePayer)throw new Error("Missing feePayer in payment requirements");let l=new m.PublicKey(p.feePayer),u=new m.PublicKey(s),g=new m.PublicKey(a);this.log("Building transaction:",{from:i.toBase58(),to:a,amount:d,asset:s,feePayer:p.feePayer});let A=[];A.push(m.ComputeBudgetProgram.setComputeUnitLimit({units:Bt})),A.push(m.ComputeBudgetProgram.setComputeUnitPrice({microLamports:Rt}));let h=await n.getAccountInfo(u,"confirmed");if(!h)throw new Error(`Token mint ${s} not found`);let E=h.owner.toBase58()===f.TOKEN_2022_PROGRAM_ID.toBase58()?f.TOKEN_2022_PROGRAM_ID:f.TOKEN_PROGRAM_ID,I=await(0,f.getMint)(n,u,void 0,E);typeof p?.decimals=="number"&&I.decimals!==p.decimals&&this.log(`Decimals mismatch: requirements say ${p.decimals}, mint says ${I.decimals}`);let T=await(0,f.getAssociatedTokenAddress)(u,i,!1,E),y=await(0,f.getAssociatedTokenAddress)(u,g,!0,E);if(!await n.getAccountInfo(T,"confirmed"))throw new Error(`No token account found for ${s}. Please ensure you have USDC in your wallet.`);if(!await n.getAccountInfo(y,"confirmed"))throw new Error(`Seller token account not found. The seller (${a}) must have a USDC account.`);let q=BigInt(d);A.push((0,f.createTransferCheckedInstruction)(T,u,y,i,q,I.decimals,[],E));let{blockhash:x}=await n.getLatestBlockhash("confirmed"),H=new m.TransactionMessage({payerKey:l,recentBlockhash:x,instructions:A}).compileToV0Message(),_=new m.VersionedTransaction(H),R=await e.signTransaction(_);return this.log("Transaction signed successfully"),{serialized:Buffer.from(R.serialize()).toString("base64"),settlementProbe:{kind:"solana",sourceAta:T.toBase58(),destinationAta:y.toBase58(),asset:s,amount:d,blockhash:x}}}async confirmSettlement(t,e){if(t.kind!=="solana")throw new Error(`SolanaAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`);let o=new m.Connection(e,"confirmed"),r=new m.PublicKey(t.destinationAta),n=await o.getSignaturesForAddress(r,{limit:25});if(n.length===0)return{settled:!1};let i=BigInt(t.amount);for(let a of n){if(a.err)continue;let s=await o.getTransaction(a.signature,{commitment:"confirmed",maxSupportedTransactionVersion:0});if(!s?.meta)continue;let p=s.transaction.message.getAccountKeys().keySegments().flat(),d=s.meta.preTokenBalances??[],l=s.meta.postTokenBalances??[];for(let u of l){if(u.mint!==t.asset)continue;let g=p[u.accountIndex];if(!g||!g.equals(r))continue;let A=d.find(I=>I.accountIndex===u.accountIndex),h=BigInt(A?.uiTokenAmount.amount??"0");if(BigInt(u.uiTokenAmount.amount??"0")-h===i)return{settled:!0,txSignature:a.signature}}}return{settled:!1}}};function mt(c){return new z(c)}var vt={PermitWitnessTransferFrom:[{name:"permitted",type:"TokenPermissions"},{name:"spender",type:"address"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"witness",type:"Witness"}],TokenPermissions:[{name:"token",type:"address"},{name:"amount",type:"uint256"}],Witness:[{name:"to",type:"address"},{name:"validAfter",type:"uint256"}]},yt=BigInt("0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff");function $(c){if(!c||typeof c!="object")return!1;let t=c;return"address"in t&&typeof t.address=="string"&&t.address.startsWith("0x")}var Z=class{name="EVM";networks=[k,b,X,F,O,M,L,U,W,K];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:evm]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="base"||t==="bsc"||t==="ethereum"||t==="arbitrum"||t==="polygon"||t==="optimism"||t==="avalanche"||t==="skale-base"||t==="skale-base-sepolia"||t.startsWith("eip155:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:S[t]?S[t]:t==="base"?S[b]:t==="bsc"?S[k]:t==="ethereum"?S[F]:t==="arbitrum"?S[O]:t==="polygon"?S[M]:t==="optimism"?S[L]:t==="avalanche"?S[U]:t==="skale-base"?S[W]:t==="skale-base-sepolia"?S[K]:S[b]}getAddress(t){return $(t)?t.address:null}isConnected(t){return $(t)?!!t.address:!1}getChainId(t){if(lt[t])return lt[t];if(t.startsWith("eip155:")){let e=t.split(":")[1];return parseInt(e,10)}return t==="base"?8453:t==="bsc"?56:t==="ethereum"?1:t==="arbitrum"?42161:8453}async getBalance(t,e,o){if(!$(e)||!e.address)return 0;let r=o||this.getDefaultRpcUrl(t.network);try{let n=this.encodeBalanceOf(e.address),i=await fetch(r,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:t.asset,data:n},"latest"]})});if(!i.ok)throw new Error(`RPC request failed: ${i.status}`);let a=await i.json();if(a.error)throw new Error(`RPC error: ${JSON.stringify(a.error)}`);if(!a.result||a.result==="0x")return 0;let s=BigInt(a.result),p=t.extra?.decimals??6;return Number(s)/Math.pow(10,p)}catch(n){throw n}}encodeBalanceOf(t){let e="0x70a08231",o=t.slice(2).toLowerCase().padStart(64,"0");return e+o}async confirmSettlement(t,e){if(t.kind==="eip3009"){let o="0xe94a0102",r=t.from.slice(2).toLowerCase().padStart(64,"0"),n=t.nonce.slice(2).toLowerCase().padStart(64,"0"),i=o+r+n,a=await this.ethCall(e,t.asset,i);return{settled:BigInt(a)!==0n}}if(t.kind==="permit2"){let o="0x4fe02b44",r=BigInt(t.nonce),n=r>>8n,i=r&0xffn,a=t.from.slice(2).toLowerCase().padStart(64,"0"),s=n.toString(16).padStart(64,"0"),p=o+a+s,d=await this.ethCall(e,C,p);return{settled:(BigInt(d)>>i&1n)===1n}}throw new Error(`EvmAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`)}async ethCall(t,e,o){let r=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:o},"latest"]})});if(!r.ok)throw new Error(`RPC request failed: ${r.status}`);let n=await r.json();if(n.error)throw new Error(`RPC error: ${JSON.stringify(n.error)}`);if(!n.result||n.result==="0x")throw new Error("RPC returned an empty result");return n.result}async buildTransaction(t,e,o){if(!$(e))throw new Error("Invalid EVM wallet");if(!e.address)throw new Error("Wallet not connected");if(t.scheme==="exact-approval")return this.buildApprovalTransaction(t,e,o);if(t.extra?.assetTransferMethod==="permit2")return this.buildPermit2Transaction(t,e,o);let{payTo:r,asset:n,extra:i}=t,a=t.amount??t.maxAmountRequired;if(!a)throw new Error("Missing amount in payment requirements");this.log("Building EVM transaction:",{from:e.address,to:r,amount:a,asset:n,network:t.network});let s=this.getChainId(t.network),p={name:i?.name??"USD Coin",version:i?.version??"2",chainId:BigInt(s),verifyingContract:n},d={TransferWithAuthorization:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"value",type:"uint256"},{name:"validAfter",type:"uint256"},{name:"validBefore",type:"uint256"},{name:"nonce",type:"bytes32"}]},l=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(l);let u="0x"+[...l].map(T=>T.toString(16).padStart(2,"0")).join(""),g=Math.floor(Date.now()/1e3),A={from:e.address,to:r,value:a,validAfter:String(g-600),validBefore:String(g+(t.maxTimeoutSeconds||60)),nonce:u},h={from:e.address,to:r,value:BigInt(a),validAfter:BigInt(g-600),validBefore:BigInt(g+(t.maxTimeoutSeconds||60)),nonce:u};if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");let E=await e.signTypedData({domain:p,types:d,primaryType:"TransferWithAuthorization",message:h});return this.log("EIP-712 signature obtained"),{serialized:JSON.stringify({authorization:A,signature:E}),signature:E,settlementProbe:{kind:"eip3009",from:e.address,nonce:u,asset:n,chainId:s}}}async buildApprovalTransaction(t,e,o){let{payTo:r,asset:n,extra:i}=t,a=t.amount??t.maxAmountRequired;if(!a)throw new Error("Missing amount in payment requirements");let s=i?.facilitatorContract;if(!s)throw new Error("exact-approval scheme requires extra.facilitatorContract from the facilitator. The /supported endpoint should provide this.");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building approval-based transaction:",{from:e.address,to:r,amount:a,asset:n,network:t.network,facilitatorContract:s});let p=o||this.getDefaultRpcUrl(t.network),d=i?.fee??"0",l=BigInt(a)+BigInt(d),u=await this.readAllowance(p,n,e.address,s);if(u<l){if(!e.sendTransaction)throw new Error("BSC payments require a wallet that supports sendTransaction for the one-time token approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.");let _=this.calculateApprovalAmount(a,d,i?.approvalStrategy);this.log(`Approving ${_} for ${s} (current allowance: ${u})`);let R=await e.sendTransaction({to:n,data:this.encodeApprove(s,_),value:0n});this.log(`Approval tx sent: ${R}`),await this.waitForReceipt(p,R),this.log("Approval confirmed")}else this.log("Sufficient allowance, skipping approval");let g=new Uint8Array(16);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(g);let A=[...g].reduce((_,R)=>_*256n+BigInt(R),0n).toString(),h=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(h);let E="0x"+[...h].map(_=>_.toString(16).padStart(2,"0")).join(""),T=Math.floor(Date.now()/1e3)+(t.maxTimeoutSeconds||300),y=i?.eip712Domain,P=y?{name:y.name,version:y.version,chainId:BigInt(y.chainId),verifyingContract:y.verifyingContract}:{name:"DexterBSCFacilitator",version:"1",chainId:BigInt(this.getChainId(t.network)),verifyingContract:s},w=i?.eip712Types??{Payment:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"token",type:"address"},{name:"amount",type:"uint256"},{name:"fee",type:"uint256"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"paymentId",type:"bytes32"}]},q={from:e.address,to:r,token:n,amount:BigInt(a),fee:BigInt(d),nonce:BigInt(A),deadline:BigInt(T),paymentId:E},x=await e.signTypedData({domain:P,types:w,primaryType:"Payment",message:q});this.log("EIP-712 Payment signature obtained");let H={from:e.address,to:r,token:n,amount:a,fee:d,nonce:A,deadline:T,paymentId:E,signature:x};return{serialized:JSON.stringify(H),signature:x}}async buildPermit2Transaction(t,e,o){let{payTo:r,asset:n}=t,i=t.amount??t.maxAmountRequired;if(!i)throw new Error("Missing amount in payment requirements");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building Permit2 transaction:",{from:e.address,to:r,amount:i,asset:n,network:t.network});let a=o||this.getDefaultRpcUrl(t.network),s=await this.readAllowance(a,n,e.address,C),p;if(s<BigInt(i)){let P=this.encodeApprove(C,yt);if(e.signTransaction){this.log(`Signing Permit2 approval for relay (current allowance: ${s})`);let w=this.getChainId(t.network),q=await this.readGasPrice(a),x=await this.readNonce(a,e.address),H=await e.signTransaction({to:n,data:P,chainId:w,gas:50000n,gasPrice:q,nonce:x});p={erc20ApprovalGasSponsoring:{info:{from:e.address,asset:n,spender:C,amount:yt.toString(),signedTransaction:H,version:"1"}}},this.log("Permit2 approval signed for facilitator relay")}else if(e.sendTransaction){this.log(`Approving Permit2 directly (current allowance: ${s})`);let w=await e.sendTransaction({to:n,data:P,value:0n});this.log(`Permit2 approval tx sent: ${w}`),await this.waitForReceipt(a,w),this.log("Permit2 approval confirmed")}else throw new Error("Permit2 payments require a wallet that supports signTransaction or sendTransaction for the one-time Permit2 approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.")}else this.log("Sufficient Permit2 allowance, skipping approval");let d=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(d);let l=[...d].reduce((P,w)=>P*256n+BigInt(w),0n),u=Math.floor(Date.now()/1e3),g=u-600,A=u+(t.maxTimeoutSeconds||300),h=this.getChainId(t.network),E={name:"Permit2",chainId:BigInt(h),verifyingContract:C},I={permitted:{token:n,amount:BigInt(i)},spender:j,nonce:l,deadline:BigInt(A),witness:{to:r,validAfter:BigInt(g)}},T=await e.signTypedData({domain:E,types:vt,primaryType:"PermitWitnessTransferFrom",message:I});this.log("Permit2 PermitWitnessTransferFrom signature obtained");let y={signature:T,permit2Authorization:{from:e.address,permitted:{token:n,amount:i},spender:j,nonce:l.toString(),deadline:String(A),witness:{to:r,validAfter:String(g)}}};return{serialized:JSON.stringify(y),signature:T,extensions:p,settlementProbe:{kind:"permit2",from:e.address,nonce:l.toString(),chainId:h}}}async readAllowance(t,e,o,r){let n="0xdd62ed3e",i=o.slice(2).toLowerCase().padStart(64,"0"),a=r.slice(2).toLowerCase().padStart(64,"0"),s=n+i+a;try{let d=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:s},"latest"]})})).json();return d.error||!d.result||d.result==="0x"?0n:BigInt(d.result)}catch{return 0n}}encodeApprove(t,e){let o="0x095ea7b3",r=t.slice(2).toLowerCase().padStart(64,"0"),n=e.toString(16).padStart(64,"0");return o+r+n}async waitForReceipt(t,e,o=3e4){let r=Date.now();for(;Date.now()-r<o;){try{let i=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionReceipt",params:[e]})})).json();if(i.result){if(i.result.status==="0x0")throw new Error(`Approval transaction reverted: ${e}`);return}}catch(n){if(n instanceof Error&&n.message.includes("reverted"))throw n}await new Promise(n=>setTimeout(n,2e3))}throw new Error(`Approval transaction receipt timeout after ${o}ms: ${e}`)}async readGasPrice(t){try{let o=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_gasPrice",params:[]})})).json();return o.result?BigInt(o.result):50000000n}catch{return 50000000n}}async readNonce(t,e){try{let r=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionCount",params:[e,"latest"]})})).json();return r.result?parseInt(r.result,16):0}catch{return 0}}calculateApprovalAmount(t,e,o){let r=BigInt(t)+BigInt(e);if(!o||o.mode==="exact")return r;let n=BigInt(o.defaultMultiple??10),i=r*n;if(o.maxCapUsd){let a=this.inferDecimals(t),s=BigInt(Math.floor(o.maxCapUsd*Math.pow(10,a)));if(i>s)return s}if(o.exactAboveUsd){let a=this.inferDecimals(t),s=BigInt(Math.floor(o.exactAboveUsd*Math.pow(10,a)));if(BigInt(t)>s)return r}return i}inferDecimals(t){return t.length>12?18:6}};function ft(c){return new Z(c)}function Dt(c){if(c===ht||c===Tt)return!0;let t=c.toLowerCase();for(let e of Object.values(J))if(e.toLowerCase()===t)return!0;for(let e of Object.keys(Y))if(e.toLowerCase()===t)return!0;return!1}function Ot(c=!1){return[mt({verbose:c}),ft({verbose:c})]}function Mt(c,t){return c.find(e=>e.canHandle(t))}0&&(module.exports={ARBITRUM_ONE,AVALANCHE,BASE_MAINNET,BASE_SEPOLIA,BSC_MAINNET,BSC_STABLECOIN_ADDRESSES,BSC_USDC,BSC_USDT,ETHEREUM_MAINNET,EvmAdapter,OPTIMISM,PERMIT2_ADDRESS,POLYGON,SKALE_BASE,SKALE_BASE_SEPOLIA,SOLANA_DEVNET,SOLANA_MAINNET,SOLANA_TESTNET,SolanaAdapter,USDC_ADDRESSES,X402_EXACT_PERMIT2_PROXY,createDefaultAdapters,createEvmAdapter,createSolanaAdapter,findAdapter,isEvmWallet,isKnownUSDC,isSolanaWallet});
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { l as SolanaAdapter, m as EvmAdapter, C as ChainAdapter } from '../types-RxdlGPaG.cjs';
|
|
2
2
|
export { n as AdapterConfig, B as BalanceInfo, E as EvmWallet, G as GenericWallet, o as SignedTransaction, a as SolanaWallet, W as WalletSet, b as createEvmAdapter, c as createSolanaAdapter, q as isEvmWallet, p as isSolanaWallet } from '../types-RxdlGPaG.cjs';
|
|
3
|
-
export { A as ARBITRUM_ONE, l as AVALANCHE, B as BASE_MAINNET, j as BASE_SEPOLIA, f as BSC_MAINNET, i as BSC_STABLECOIN_ADDRESSES, h as BSC_USDC, g as BSC_USDT, E as ETHEREUM_MAINNET, O as OPTIMISM, P as PERMIT2_ADDRESS, k as POLYGON, m as SKALE_BASE, n as SKALE_BASE_SEPOLIA, d as SOLANA_DEVNET, S as SOLANA_MAINNET, e as SOLANA_TESTNET, o as USDC_ADDRESSES, X as X402_EXACT_PERMIT2_PROXY } from '../constants-
|
|
3
|
+
export { A as ARBITRUM_ONE, l as AVALANCHE, B as BASE_MAINNET, j as BASE_SEPOLIA, f as BSC_MAINNET, i as BSC_STABLECOIN_ADDRESSES, h as BSC_USDC, g as BSC_USDT, E as ETHEREUM_MAINNET, O as OPTIMISM, P as PERMIT2_ADDRESS, k as POLYGON, m as SKALE_BASE, n as SKALE_BASE_SEPOLIA, d as SOLANA_DEVNET, S as SOLANA_MAINNET, e as SOLANA_TESTNET, o as USDC_ADDRESSES, X as X402_EXACT_PERMIT2_PROXY } from '../constants-D41hDAG6.cjs';
|
|
4
4
|
|
|
5
5
|
/**
|
|
6
6
|
* Check if an asset address is a known USDC contract (any chain).
|
package/dist/adapters/index.d.ts
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
import { l as SolanaAdapter, m as EvmAdapter, C as ChainAdapter } from '../types-RxdlGPaG.js';
|
|
2
2
|
export { n as AdapterConfig, B as BalanceInfo, E as EvmWallet, G as GenericWallet, o as SignedTransaction, a as SolanaWallet, W as WalletSet, b as createEvmAdapter, c as createSolanaAdapter, q as isEvmWallet, p as isSolanaWallet } from '../types-RxdlGPaG.js';
|
|
3
|
-
export { A as ARBITRUM_ONE, l as AVALANCHE, B as BASE_MAINNET, j as BASE_SEPOLIA, f as BSC_MAINNET, i as BSC_STABLECOIN_ADDRESSES, h as BSC_USDC, g as BSC_USDT, E as ETHEREUM_MAINNET, O as OPTIMISM, P as PERMIT2_ADDRESS, k as POLYGON, m as SKALE_BASE, n as SKALE_BASE_SEPOLIA, d as SOLANA_DEVNET, S as SOLANA_MAINNET, e as SOLANA_TESTNET, o as USDC_ADDRESSES, X as X402_EXACT_PERMIT2_PROXY } from '../constants-
|
|
3
|
+
export { A as ARBITRUM_ONE, l as AVALANCHE, B as BASE_MAINNET, j as BASE_SEPOLIA, f as BSC_MAINNET, i as BSC_STABLECOIN_ADDRESSES, h as BSC_USDC, g as BSC_USDT, E as ETHEREUM_MAINNET, O as OPTIMISM, P as PERMIT2_ADDRESS, k as POLYGON, m as SKALE_BASE, n as SKALE_BASE_SEPOLIA, d as SOLANA_DEVNET, S as SOLANA_MAINNET, e as SOLANA_TESTNET, o as USDC_ADDRESSES, X as X402_EXACT_PERMIT2_PROXY } from '../constants-D41hDAG6.js';
|
|
4
4
|
|
|
5
5
|
/**
|
|
6
6
|
* Check if an asset address is a known USDC contract (any chain).
|
package/dist/adapters/index.js
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
import{PublicKey as x,Connection as ut,TransactionMessage as Ct,VersionedTransaction as _t,ComputeBudgetProgram as Tt}from"@solana/web3.js";import{getAssociatedTokenAddress as mt,getAccount as Pt,createTransferCheckedInstruction as xt,getMint as bt,TOKEN_PROGRAM_ID as yt,TOKEN_2022_PROGRAM_ID as at}from"@solana/spl-token";var gt="solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp",At="solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1",Et="solana:4uhcVJyU9pJkvQyS88uRDiswHXSCkY3z",N=gt,D=At,O=Et,H="eip155:8453",X="eip155:84532",G="eip155:42161",J="eip155:137",Y="eip155:10",z="eip155:43114",Z="eip155:56",Q="eip155:1187947933",tt="eip155:324705682",et="eip155:1",B=H,nt=X,M=G,L=J,U=Y,k=z,W=Z,K=Q,F=tt,j=et,St="EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v",ht="4zMMC9srt5Ri5X14GAgXhaHii3GnPAEERYPJgZJDncDU",wt="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",dt="0x55d398326f99059fF775485246999027B3197955",rt="0x8AC76a51cc950d9822D68b83fE1Ad97B32Cd580d",ot={[Z]:rt,[H]:wt,[X]:"0x036CbD53842c5426634e7929541eC2318f3dCF7e",[G]:"0xaf88d065e77c8cC2239327C5EDb3A432268e5831",[J]:"0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",[Y]:"0x0b2C639c533813f4Aa9D7837CAf62653d097Ff85",[z]:"0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E",[Q]:"0x85889c8c714505E0c94b30fcfcF64fE3Ac8FCb20",[tt]:"0x2e08028E3C4c2356572E096d8EF835cD5C6030bD",[et]:"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48"},st={[dt]:{symbol:"USDT",decimals:18},[rt]:{symbol:"USDC",decimals:18}};var w="0x000000000022D473030F116dDEE9F6B43aC78BA3",V="0x402085c248EeA27D92E8b30b2C58ed07f9E20001",pt={[Z]:56,[H]:8453,[X]:84532,[G]:42161,[J]:137,[Y]:10,[z]:43114,[Q]:1187947933,[tt]:324705682,[et]:1},P={[gt]:"https://api.dexter.cash/api/solana/rpc",[At]:"https://api.devnet.solana.com",[Et]:"https://api.testnet.solana.com"},A={[Z]:"https://bsc-dataseed1.binance.org",[H]:"https://api.dexter.cash/api/base/rpc",[X]:"https://sepolia.base.org",[G]:"https://arb1.arbitrum.io/rpc",[J]:"https://polygon-rpc.com",[Y]:"https://mainnet.optimism.io",[z]:"https://api.avax.network/ext/bc/C/rpc",[Q]:"https://skale-base.skalenodes.com/v1/base",[tt]:"https://base-sepolia-testnet.skalenodes.com/v1/jubilant-horrible-ancha",[et]:"https://eth.llamarpc.com"};var Nt=12e3,Bt=1;function $(u){if(!u||typeof u!="object")return!1;let t=u;return"publicKey"in t&&"signTransaction"in t&&typeof t.signTransaction=="function"}var it=class{name="Solana";networks=[N,D,O];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:solana]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="solana"||t==="solana-devnet"||t==="solana-testnet"||t.startsWith("solana:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:P[t]?P[t]:t==="solana"?P[N]:t==="solana-devnet"?P[D]:t==="solana-testnet"?P[O]:P[N]}getAddress(t){return $(t)?t.publicKey?.toBase58()??null:null}isConnected(t){return $(t)?t.publicKey!==null:!1}async getBalance(t,e,s){if(!$(e)||!e.publicKey)return 0;let a=s||this.getDefaultRpcUrl(t.network),n=new ut(a,"confirmed"),i=new x(e.publicKey.toBase58()),o=new x(t.asset);try{let d=(await n.getAccountInfo(o,"confirmed"))?.owner.toBase58()===at.toBase58()?at:yt,c=await mt(o,i,!1,d),m=await Pt(n,c,void 0,d),p=t.extra?.decimals??6;return Number(m.amount)/Math.pow(10,p)}catch(r){if(r&&typeof r=="object"&&"name"in r&&(r.name==="TokenAccountNotFoundError"||r.name==="TokenInvalidAccountOwnerError"))return 0;throw r}}async buildTransaction(t,e,s){if(!$(e))throw new Error("Invalid Solana wallet");if(!e.publicKey)throw new Error("Wallet not connected");let a=s||this.getDefaultRpcUrl(t.network),n=new ut(a,"confirmed"),i=new x(e.publicKey.toBase58()),{payTo:o,asset:r,extra:d}=t,c=t.amount??t.maxAmountRequired;if(!c)throw new Error("Missing amount in payment requirements");if(!d?.feePayer)throw new Error("Missing feePayer in payment requirements");let m=new x(d.feePayer),p=new x(r),f=new x(o);this.log("Building transaction:",{from:i.toBase58(),to:o,amount:c,asset:r,feePayer:d.feePayer});let l=[];l.push(Tt.setComputeUnitLimit({units:Nt})),l.push(Tt.setComputeUnitPrice({microLamports:Bt}));let E=await n.getAccountInfo(p,"confirmed");if(!E)throw new Error(`Token mint ${r} not found`);let g=E.owner.toBase58()===at.toBase58()?at:yt,T=await bt(n,p,void 0,g);typeof d?.decimals=="number"&&T.decimals!==d.decimals&&this.log(`Decimals mismatch: requirements say ${d.decimals}, mint says ${T.decimals}`);let S=await mt(p,i,!1,g),h=await mt(p,f,!0,g);if(!await n.getAccountInfo(S,"confirmed"))throw new Error(`No token account found for ${r}. Please ensure you have USDC in your wallet.`);if(!await n.getAccountInfo(h,"confirmed"))throw new Error(`Seller token account not found. The seller (${o}) must have a USDC account.`);let R=BigInt(c);l.push(xt(S,p,h,i,R,T.decimals,[],g));let{blockhash:_}=await n.getLatestBlockhash("confirmed"),v=new Ct({payerKey:m,recentBlockhash:_,instructions:l}).compileToV0Message(),I=new _t(v),b=await e.signTransaction(I);return this.log("Transaction signed successfully"),{serialized:Buffer.from(b.serialize()).toString("base64"),settlementProbe:{kind:"solana",sourceAta:S.toBase58(),destinationAta:h.toBase58(),asset:r,amount:c,blockhash:_}}}async confirmSettlement(t,e){if(t.kind!=="solana")throw new Error(`SolanaAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`);let s=new ut(e,"confirmed"),a=new x(t.destinationAta),n=await s.getSignaturesForAddress(a,{limit:25});if(n.length===0)return{settled:!1};let i=BigInt(t.amount);for(let o of n){if(o.err)continue;let r=await s.getTransaction(o.signature,{commitment:"confirmed",maxSupportedTransactionVersion:0});if(!r?.meta)continue;let d=r.transaction.message.getAccountKeys().keySegments().flat(),c=r.meta.preTokenBalances??[],m=r.meta.postTokenBalances??[];for(let p of m){if(p.mint!==t.asset)continue;let f=d[p.accountIndex];if(!f||!f.equals(a))continue;let l=c.find(T=>T.accountIndex===p.accountIndex),E=BigInt(l?.uiTokenAmount.amount??"0");if(BigInt(p.uiTokenAmount.amount??"0")-E===i)return{settled:!0,txSignature:o.signature}}}return{settled:!1}}};function ft(u){return new it(u)}var Rt={PermitWitnessTransferFrom:[{name:"permitted",type:"TokenPermissions"},{name:"spender",type:"address"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"witness",type:"Witness"}],TokenPermissions:[{name:"token",type:"address"},{name:"amount",type:"uint256"}],Witness:[{name:"to",type:"address"},{name:"validAfter",type:"uint256"}]},It=BigInt("0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff");function q(u){if(!u||typeof u!="object")return!1;let t=u;return"address"in t&&typeof t.address=="string"&&t.address.startsWith("0x")}var ct=class{name="EVM";networks=[W,B,nt,j,M,L,U,k,K,F];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:evm]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="base"||t==="bsc"||t==="ethereum"||t==="arbitrum"||t==="polygon"||t==="optimism"||t==="avalanche"||t==="skale-base"||t==="skale-base-sepolia"||t.startsWith("eip155:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:A[t]?A[t]:t==="base"?A[B]:t==="bsc"?A[W]:t==="ethereum"?A[j]:t==="arbitrum"?A[M]:t==="polygon"?A[L]:t==="optimism"?A[U]:t==="avalanche"?A[k]:t==="skale-base"?A[K]:t==="skale-base-sepolia"?A[F]:A[B]}getAddress(t){return q(t)?t.address:null}isConnected(t){return q(t)?!!t.address:!1}getChainId(t){if(pt[t])return pt[t];if(t.startsWith("eip155:")){let e=t.split(":")[1];return parseInt(e,10)}return t==="base"?8453:t==="bsc"?56:t==="ethereum"?1:t==="arbitrum"?42161:8453}async getBalance(t,e,s){if(!q(e)||!e.address)return 0;let a=s||this.getDefaultRpcUrl(t.network);try{let n=this.encodeBalanceOf(e.address),i=await fetch(a,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:t.asset,data:n},"latest"]})});if(!i.ok)throw new Error(`RPC request failed: ${i.status}`);let o=await i.json();if(o.error)throw new Error(`RPC error: ${JSON.stringify(o.error)}`);if(!o.result||o.result==="0x")return 0;let r=BigInt(o.result),d=t.extra?.decimals??6;return Number(r)/Math.pow(10,d)}catch(n){throw n}}encodeBalanceOf(t){let e="0x70a08231",s=t.slice(2).toLowerCase().padStart(64,"0");return e+s}async confirmSettlement(t,e){if(t.kind==="eip3009"){let s="0xe94a0102",a=t.from.slice(2).toLowerCase().padStart(64,"0"),n=t.nonce.slice(2).toLowerCase().padStart(64,"0"),i=s+a+n,o=await this.ethCall(e,t.asset,i);return{settled:BigInt(o)!==0n}}if(t.kind==="permit2"){let s="0x4fe02b44",a=BigInt(t.nonce),n=a>>8n,i=a&0xffn,o=t.from.slice(2).toLowerCase().padStart(64,"0"),r=n.toString(16).padStart(64,"0"),d=s+o+r,c=await this.ethCall(e,w,d);return{settled:(BigInt(c)>>i&1n)===1n}}throw new Error(`EvmAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`)}async ethCall(t,e,s){let a=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:s},"latest"]})});if(!a.ok)throw new Error(`RPC request failed: ${a.status}`);let n=await a.json();if(n.error)throw new Error(`RPC error: ${JSON.stringify(n.error)}`);if(!n.result||n.result==="0x")throw new Error("RPC returned an empty result");return n.result}async buildTransaction(t,e,s){if(!q(e))throw new Error("Invalid EVM wallet");if(!e.address)throw new Error("Wallet not connected");if(t.scheme==="exact-approval")return this.buildApprovalTransaction(t,e,s);if(t.extra?.assetTransferMethod==="permit2")return this.buildPermit2Transaction(t,e,s);let{payTo:a,asset:n,extra:i}=t,o=t.amount??t.maxAmountRequired;if(!o)throw new Error("Missing amount in payment requirements");this.log("Building EVM transaction:",{from:e.address,to:a,amount:o,asset:n,network:t.network});let r=this.getChainId(t.network),d={name:i?.name??"USD Coin",version:i?.version??"2",chainId:BigInt(r),verifyingContract:n},c={TransferWithAuthorization:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"value",type:"uint256"},{name:"validAfter",type:"uint256"},{name:"validBefore",type:"uint256"},{name:"nonce",type:"bytes32"}]},m=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(m);let p="0x"+[...m].map(S=>S.toString(16).padStart(2,"0")).join(""),f=Math.floor(Date.now()/1e3),l={from:e.address,to:a,value:o,validAfter:String(f-600),validBefore:String(f+(t.maxTimeoutSeconds||60)),nonce:p},E={from:e.address,to:a,value:BigInt(o),validAfter:BigInt(f-600),validBefore:BigInt(f+(t.maxTimeoutSeconds||60)),nonce:p};if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");let g=await e.signTypedData({domain:d,types:c,primaryType:"TransferWithAuthorization",message:E});return this.log("EIP-712 signature obtained"),{serialized:JSON.stringify({authorization:l,signature:g}),signature:g,settlementProbe:{kind:"eip3009",from:e.address,nonce:p,asset:n,chainId:r}}}async buildApprovalTransaction(t,e,s){let{payTo:a,asset:n,extra:i}=t,o=t.amount??t.maxAmountRequired;if(!o)throw new Error("Missing amount in payment requirements");let r=i?.facilitatorContract;if(!r)throw new Error("exact-approval scheme requires extra.facilitatorContract from the facilitator. The /supported endpoint should provide this.");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building approval-based transaction:",{from:e.address,to:a,amount:o,asset:n,network:t.network,facilitatorContract:r});let d=s||this.getDefaultRpcUrl(t.network),c=i?.fee??"0",m=BigInt(o)+BigInt(c),p=await this.readAllowance(d,n,e.address,r);if(p<m){if(!e.sendTransaction)throw new Error("BSC payments require a wallet that supports sendTransaction for the one-time token approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.");let I=this.calculateApprovalAmount(o,c,i?.approvalStrategy);this.log(`Approving ${I} for ${r} (current allowance: ${p})`);let b=await e.sendTransaction({to:n,data:this.encodeApprove(r,I),value:0n});this.log(`Approval tx sent: ${b}`),await this.waitForReceipt(d,b),this.log("Approval confirmed")}else this.log("Sufficient allowance, skipping approval");let f=new Uint8Array(16);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(f);let l=[...f].reduce((I,b)=>I*256n+BigInt(b),0n).toString(),E=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(E);let g="0x"+[...E].map(I=>I.toString(16).padStart(2,"0")).join(""),S=Math.floor(Date.now()/1e3)+(t.maxTimeoutSeconds||300),h=i?.eip712Domain,C=h?{name:h.name,version:h.version,chainId:BigInt(h.chainId),verifyingContract:h.verifyingContract}:{name:"DexterBSCFacilitator",version:"1",chainId:BigInt(this.getChainId(t.network)),verifyingContract:r},y=i?.eip712Types??{Payment:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"token",type:"address"},{name:"amount",type:"uint256"},{name:"fee",type:"uint256"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"paymentId",type:"bytes32"}]},R={from:e.address,to:a,token:n,amount:BigInt(o),fee:BigInt(c),nonce:BigInt(l),deadline:BigInt(S),paymentId:g},_=await e.signTypedData({domain:C,types:y,primaryType:"Payment",message:R});this.log("EIP-712 Payment signature obtained");let v={from:e.address,to:a,token:n,amount:o,fee:c,nonce:l,deadline:S,paymentId:g,signature:_};return{serialized:JSON.stringify(v),signature:_}}async buildPermit2Transaction(t,e,s){let{payTo:a,asset:n}=t,i=t.amount??t.maxAmountRequired;if(!i)throw new Error("Missing amount in payment requirements");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building Permit2 transaction:",{from:e.address,to:a,amount:i,asset:n,network:t.network});let o=s||this.getDefaultRpcUrl(t.network),r=await this.readAllowance(o,n,e.address,w),d;if(r<BigInt(i)){let C=this.encodeApprove(w,It);if(e.signTransaction){this.log(`Signing Permit2 approval for relay (current allowance: ${r})`);let y=this.getChainId(t.network),R=await this.readGasPrice(o),_=await this.readNonce(o,e.address),v=await e.signTransaction({to:n,data:C,chainId:y,gas:50000n,gasPrice:R,nonce:_});d={erc20ApprovalGasSponsoring:{info:{from:e.address,asset:n,spender:w,amount:It.toString(),signedTransaction:v,version:"1"}}},this.log("Permit2 approval signed for facilitator relay")}else if(e.sendTransaction){this.log(`Approving Permit2 directly (current allowance: ${r})`);let y=await e.sendTransaction({to:n,data:C,value:0n});this.log(`Permit2 approval tx sent: ${y}`),await this.waitForReceipt(o,y),this.log("Permit2 approval confirmed")}else throw new Error("Permit2 payments require a wallet that supports signTransaction or sendTransaction for the one-time Permit2 approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.")}else this.log("Sufficient Permit2 allowance, skipping approval");let c=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(c);let m=[...c].reduce((C,y)=>C*256n+BigInt(y),0n),p=Math.floor(Date.now()/1e3),f=p-600,l=p+(t.maxTimeoutSeconds||300),E=this.getChainId(t.network),g={name:"Permit2",chainId:BigInt(E),verifyingContract:w},T={permitted:{token:n,amount:BigInt(i)},spender:V,nonce:m,deadline:BigInt(l),witness:{to:a,validAfter:BigInt(f)}},S=await e.signTypedData({domain:g,types:Rt,primaryType:"PermitWitnessTransferFrom",message:T});this.log("Permit2 PermitWitnessTransferFrom signature obtained");let h={signature:S,permit2Authorization:{from:e.address,permitted:{token:n,amount:i},spender:V,nonce:m.toString(),deadline:String(l),witness:{to:a,validAfter:String(f)}}};return{serialized:JSON.stringify(h),signature:S,extensions:d,settlementProbe:{kind:"permit2",from:e.address,nonce:m.toString(),chainId:E}}}async readAllowance(t,e,s,a){let n="0xdd62ed3e",i=s.slice(2).toLowerCase().padStart(64,"0"),o=a.slice(2).toLowerCase().padStart(64,"0"),r=n+i+o;try{let c=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:r},"latest"]})})).json();return c.error||!c.result||c.result==="0x"?0n:BigInt(c.result)}catch{return 0n}}encodeApprove(t,e){let s="0x095ea7b3",a=t.slice(2).toLowerCase().padStart(64,"0"),n=e.toString(16).padStart(64,"0");return s+a+n}async waitForReceipt(t,e,s=3e4){let a=Date.now();for(;Date.now()-a<s;){try{let i=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionReceipt",params:[e]})})).json();if(i.result){if(i.result.status==="0x0")throw new Error(`Approval transaction reverted: ${e}`);return}}catch(n){if(n instanceof Error&&n.message.includes("reverted"))throw n}await new Promise(n=>setTimeout(n,2e3))}throw new Error(`Approval transaction receipt timeout after ${s}ms: ${e}`)}async readGasPrice(t){try{let s=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_gasPrice",params:[]})})).json();return s.result?BigInt(s.result):50000000n}catch{return 50000000n}}async readNonce(t,e){try{let a=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionCount",params:[e,"latest"]})})).json();return a.result?parseInt(a.result,16):0}catch{return 0}}calculateApprovalAmount(t,e,s){let a=BigInt(t)+BigInt(e);if(!s||s.mode==="exact")return a;let n=BigInt(s.defaultMultiple??10),i=a*n;if(s.maxCapUsd){let o=this.inferDecimals(t),r=BigInt(Math.floor(s.maxCapUsd*Math.pow(10,o)));if(i>r)return r}if(s.exactAboveUsd){let o=this.inferDecimals(t),r=BigInt(Math.floor(s.exactAboveUsd*Math.pow(10,o)));if(BigInt(t)>r)return a}return i}inferDecimals(t){return t.length>12?18:6}};function lt(u){return new ct(u)}function $t(u){if(u===St||u===ht)return!0;let t=u.toLowerCase();for(let e of Object.values(ot))if(e.toLowerCase()===t)return!0;for(let e of Object.keys(st))if(e.toLowerCase()===t)return!0;return!1}function qt(u=!1){return[ft({verbose:u}),lt({verbose:u})]}function Ht(u,t){return u.find(e=>e.canHandle(t))}export{M as ARBITRUM_ONE,k as AVALANCHE,B as BASE_MAINNET,nt as BASE_SEPOLIA,W as BSC_MAINNET,st as BSC_STABLECOIN_ADDRESSES,rt as BSC_USDC,dt as BSC_USDT,j as ETHEREUM_MAINNET,ct as EvmAdapter,U as OPTIMISM,w as PERMIT2_ADDRESS,L as POLYGON,K as SKALE_BASE,F as SKALE_BASE_SEPOLIA,D as SOLANA_DEVNET,N as SOLANA_MAINNET,O as SOLANA_TESTNET,it as SolanaAdapter,ot as USDC_ADDRESSES,V as X402_EXACT_PERMIT2_PROXY,qt as createDefaultAdapters,lt as createEvmAdapter,ft as createSolanaAdapter,Ht as findAdapter,q as isEvmWallet,$t as isKnownUSDC,$ as isSolanaWallet};
|
|
1
|
+
import{PublicKey as x,Connection as ut,TransactionMessage as Ct,VersionedTransaction as _t,ComputeBudgetProgram as Tt}from"@solana/web3.js";import{getAssociatedTokenAddress as mt,getAccount as Pt,createTransferCheckedInstruction as xt,getMint as Nt,TOKEN_PROGRAM_ID as yt,TOKEN_2022_PROGRAM_ID as at}from"@solana/spl-token";var gt="solana:5eykt4UsFv8P8NJdTREpY1vzqKqZKvdp",At="solana:EtWTRABZaYq6iMfeYKouRu166VU2xqa1",Et="solana:4uhcVJyU9pJkvQyS88uRDiswHXSCkY3z",b=gt,D=At,O=Et,H="eip155:8453",X="eip155:84532",G="eip155:42161",J="eip155:137",Y="eip155:10",z="eip155:43114",Z="eip155:56",Q="eip155:1187947933",tt="eip155:324705682",et="eip155:1",B=H,nt=X,M=G,L=J,U=Y,k=z,W=Z,K=Q,F=tt,j=et,St="EPjFWdd5AufqSSqeM2qN1xzybapC8G4wEGGkZwyTDt1v",ht="4zMMC9srt5Ri5X14GAgXhaHii3GnPAEERYPJgZJDncDU",wt="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",dt="0x55d398326f99059fF775485246999027B3197955",rt="0x8AC76a51cc950d9822D68b83fE1Ad97B32Cd580d",ot={[Z]:rt,[H]:wt,[X]:"0x036CbD53842c5426634e7929541eC2318f3dCF7e",[G]:"0xaf88d065e77c8cC2239327C5EDb3A432268e5831",[J]:"0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",[Y]:"0x0b2C639c533813f4Aa9D7837CAf62653d097Ff85",[z]:"0xB97EF9Ef8734C71904D8002F8b6Bc66Dd9c48a6E",[Q]:"0x85889c8c714505E0c94b30fcfcF64fE3Ac8FCb20",[tt]:"0x2e08028E3C4c2356572E096d8EF835cD5C6030bD",[et]:"0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48"},st={[dt]:{symbol:"USDT",decimals:18},[rt]:{symbol:"USDC",decimals:18}};var w="0x000000000022D473030F116dDEE9F6B43aC78BA3",V="0x402085c248EeA27D92E8b30b2C58ed07f9E20001",pt={[Z]:56,[H]:8453,[X]:84532,[G]:42161,[J]:137,[Y]:10,[z]:43114,[Q]:1187947933,[tt]:324705682,[et]:1},P={[gt]:"https://api.dexter.cash/api/solana/rpc",[At]:"https://api.devnet.solana.com",[Et]:"https://api.testnet.solana.com"},A={[Z]:"https://api.dexter.cash/api/evm/bsc/rpc",[H]:"https://api.dexter.cash/api/base/rpc",[X]:"https://sepolia.base.org",[G]:"https://api.dexter.cash/api/evm/arbitrum/rpc",[J]:"https://api.dexter.cash/api/evm/polygon/rpc",[Y]:"https://api.dexter.cash/api/evm/optimism/rpc",[z]:"https://api.dexter.cash/api/evm/avalanche/rpc",[Q]:"https://skale-base.skalenodes.com/v1/base",[tt]:"https://base-sepolia-testnet.skalenodes.com/v1/jubilant-horrible-ancha",[et]:"https://eth.llamarpc.com"};var bt=12e3,Bt=1;function $(u){if(!u||typeof u!="object")return!1;let t=u;return"publicKey"in t&&"signTransaction"in t&&typeof t.signTransaction=="function"}var it=class{name="Solana";networks=[b,D,O];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:solana]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="solana"||t==="solana-devnet"||t==="solana-testnet"||t.startsWith("solana:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:P[t]?P[t]:t==="solana"?P[b]:t==="solana-devnet"?P[D]:t==="solana-testnet"?P[O]:P[b]}getAddress(t){return $(t)?t.publicKey?.toBase58()??null:null}isConnected(t){return $(t)?t.publicKey!==null:!1}async getBalance(t,e,s){if(!$(e)||!e.publicKey)return 0;let a=s||this.getDefaultRpcUrl(t.network),n=new ut(a,"confirmed"),i=new x(e.publicKey.toBase58()),o=new x(t.asset);try{let d=(await n.getAccountInfo(o,"confirmed"))?.owner.toBase58()===at.toBase58()?at:yt,c=await mt(o,i,!1,d),m=await Pt(n,c,void 0,d),p=t.extra?.decimals??6;return Number(m.amount)/Math.pow(10,p)}catch(r){if(r&&typeof r=="object"&&"name"in r&&(r.name==="TokenAccountNotFoundError"||r.name==="TokenInvalidAccountOwnerError"))return 0;throw r}}async buildTransaction(t,e,s){if(!$(e))throw new Error("Invalid Solana wallet");if(!e.publicKey)throw new Error("Wallet not connected");let a=s||this.getDefaultRpcUrl(t.network),n=new ut(a,"confirmed"),i=new x(e.publicKey.toBase58()),{payTo:o,asset:r,extra:d}=t,c=t.amount??t.maxAmountRequired;if(!c)throw new Error("Missing amount in payment requirements");if(!d?.feePayer)throw new Error("Missing feePayer in payment requirements");let m=new x(d.feePayer),p=new x(r),f=new x(o);this.log("Building transaction:",{from:i.toBase58(),to:o,amount:c,asset:r,feePayer:d.feePayer});let l=[];l.push(Tt.setComputeUnitLimit({units:bt})),l.push(Tt.setComputeUnitPrice({microLamports:Bt}));let E=await n.getAccountInfo(p,"confirmed");if(!E)throw new Error(`Token mint ${r} not found`);let g=E.owner.toBase58()===at.toBase58()?at:yt,T=await Nt(n,p,void 0,g);typeof d?.decimals=="number"&&T.decimals!==d.decimals&&this.log(`Decimals mismatch: requirements say ${d.decimals}, mint says ${T.decimals}`);let S=await mt(p,i,!1,g),h=await mt(p,f,!0,g);if(!await n.getAccountInfo(S,"confirmed"))throw new Error(`No token account found for ${r}. Please ensure you have USDC in your wallet.`);if(!await n.getAccountInfo(h,"confirmed"))throw new Error(`Seller token account not found. The seller (${o}) must have a USDC account.`);let R=BigInt(c);l.push(xt(S,p,h,i,R,T.decimals,[],g));let{blockhash:_}=await n.getLatestBlockhash("confirmed"),v=new Ct({payerKey:m,recentBlockhash:_,instructions:l}).compileToV0Message(),I=new _t(v),N=await e.signTransaction(I);return this.log("Transaction signed successfully"),{serialized:Buffer.from(N.serialize()).toString("base64"),settlementProbe:{kind:"solana",sourceAta:S.toBase58(),destinationAta:h.toBase58(),asset:r,amount:c,blockhash:_}}}async confirmSettlement(t,e){if(t.kind!=="solana")throw new Error(`SolanaAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`);let s=new ut(e,"confirmed"),a=new x(t.destinationAta),n=await s.getSignaturesForAddress(a,{limit:25});if(n.length===0)return{settled:!1};let i=BigInt(t.amount);for(let o of n){if(o.err)continue;let r=await s.getTransaction(o.signature,{commitment:"confirmed",maxSupportedTransactionVersion:0});if(!r?.meta)continue;let d=r.transaction.message.getAccountKeys().keySegments().flat(),c=r.meta.preTokenBalances??[],m=r.meta.postTokenBalances??[];for(let p of m){if(p.mint!==t.asset)continue;let f=d[p.accountIndex];if(!f||!f.equals(a))continue;let l=c.find(T=>T.accountIndex===p.accountIndex),E=BigInt(l?.uiTokenAmount.amount??"0");if(BigInt(p.uiTokenAmount.amount??"0")-E===i)return{settled:!0,txSignature:o.signature}}}return{settled:!1}}};function ft(u){return new it(u)}var Rt={PermitWitnessTransferFrom:[{name:"permitted",type:"TokenPermissions"},{name:"spender",type:"address"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"witness",type:"Witness"}],TokenPermissions:[{name:"token",type:"address"},{name:"amount",type:"uint256"}],Witness:[{name:"to",type:"address"},{name:"validAfter",type:"uint256"}]},It=BigInt("0xffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff");function q(u){if(!u||typeof u!="object")return!1;let t=u;return"address"in t&&typeof t.address=="string"&&t.address.startsWith("0x")}var ct=class{name="EVM";networks=[W,B,nt,j,M,L,U,k,K,F];config;log;constructor(t={}){this.config=t,this.log=t.verbose?console.log.bind(console,"[x402:evm]"):()=>{}}canHandle(t){return!!(this.networks.includes(t)||t==="base"||t==="bsc"||t==="ethereum"||t==="arbitrum"||t==="polygon"||t==="optimism"||t==="avalanche"||t==="skale-base"||t==="skale-base-sepolia"||t.startsWith("eip155:"))}getDefaultRpcUrl(t){return this.config.rpcUrls?.[t]?this.config.rpcUrls[t]:A[t]?A[t]:t==="base"?A[B]:t==="bsc"?A[W]:t==="ethereum"?A[j]:t==="arbitrum"?A[M]:t==="polygon"?A[L]:t==="optimism"?A[U]:t==="avalanche"?A[k]:t==="skale-base"?A[K]:t==="skale-base-sepolia"?A[F]:A[B]}getAddress(t){return q(t)?t.address:null}isConnected(t){return q(t)?!!t.address:!1}getChainId(t){if(pt[t])return pt[t];if(t.startsWith("eip155:")){let e=t.split(":")[1];return parseInt(e,10)}return t==="base"?8453:t==="bsc"?56:t==="ethereum"?1:t==="arbitrum"?42161:8453}async getBalance(t,e,s){if(!q(e)||!e.address)return 0;let a=s||this.getDefaultRpcUrl(t.network);try{let n=this.encodeBalanceOf(e.address),i=await fetch(a,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:t.asset,data:n},"latest"]})});if(!i.ok)throw new Error(`RPC request failed: ${i.status}`);let o=await i.json();if(o.error)throw new Error(`RPC error: ${JSON.stringify(o.error)}`);if(!o.result||o.result==="0x")return 0;let r=BigInt(o.result),d=t.extra?.decimals??6;return Number(r)/Math.pow(10,d)}catch(n){throw n}}encodeBalanceOf(t){let e="0x70a08231",s=t.slice(2).toLowerCase().padStart(64,"0");return e+s}async confirmSettlement(t,e){if(t.kind==="eip3009"){let s="0xe94a0102",a=t.from.slice(2).toLowerCase().padStart(64,"0"),n=t.nonce.slice(2).toLowerCase().padStart(64,"0"),i=s+a+n,o=await this.ethCall(e,t.asset,i);return{settled:BigInt(o)!==0n}}if(t.kind==="permit2"){let s="0x4fe02b44",a=BigInt(t.nonce),n=a>>8n,i=a&0xffn,o=t.from.slice(2).toLowerCase().padStart(64,"0"),r=n.toString(16).padStart(64,"0"),d=s+o+r,c=await this.ethCall(e,w,d);return{settled:(BigInt(c)>>i&1n)===1n}}throw new Error(`EvmAdapter.confirmSettlement cannot handle probe kind "${t.kind}"`)}async ethCall(t,e,s){let a=await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:s},"latest"]})});if(!a.ok)throw new Error(`RPC request failed: ${a.status}`);let n=await a.json();if(n.error)throw new Error(`RPC error: ${JSON.stringify(n.error)}`);if(!n.result||n.result==="0x")throw new Error("RPC returned an empty result");return n.result}async buildTransaction(t,e,s){if(!q(e))throw new Error("Invalid EVM wallet");if(!e.address)throw new Error("Wallet not connected");if(t.scheme==="exact-approval")return this.buildApprovalTransaction(t,e,s);if(t.extra?.assetTransferMethod==="permit2")return this.buildPermit2Transaction(t,e,s);let{payTo:a,asset:n,extra:i}=t,o=t.amount??t.maxAmountRequired;if(!o)throw new Error("Missing amount in payment requirements");this.log("Building EVM transaction:",{from:e.address,to:a,amount:o,asset:n,network:t.network});let r=this.getChainId(t.network),d={name:i?.name??"USD Coin",version:i?.version??"2",chainId:BigInt(r),verifyingContract:n},c={TransferWithAuthorization:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"value",type:"uint256"},{name:"validAfter",type:"uint256"},{name:"validBefore",type:"uint256"},{name:"nonce",type:"bytes32"}]},m=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(m);let p="0x"+[...m].map(S=>S.toString(16).padStart(2,"0")).join(""),f=Math.floor(Date.now()/1e3),l={from:e.address,to:a,value:o,validAfter:String(f-600),validBefore:String(f+(t.maxTimeoutSeconds||60)),nonce:p},E={from:e.address,to:a,value:BigInt(o),validAfter:BigInt(f-600),validBefore:BigInt(f+(t.maxTimeoutSeconds||60)),nonce:p};if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");let g=await e.signTypedData({domain:d,types:c,primaryType:"TransferWithAuthorization",message:E});return this.log("EIP-712 signature obtained"),{serialized:JSON.stringify({authorization:l,signature:g}),signature:g,settlementProbe:{kind:"eip3009",from:e.address,nonce:p,asset:n,chainId:r}}}async buildApprovalTransaction(t,e,s){let{payTo:a,asset:n,extra:i}=t,o=t.amount??t.maxAmountRequired;if(!o)throw new Error("Missing amount in payment requirements");let r=i?.facilitatorContract;if(!r)throw new Error("exact-approval scheme requires extra.facilitatorContract from the facilitator. The /supported endpoint should provide this.");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building approval-based transaction:",{from:e.address,to:a,amount:o,asset:n,network:t.network,facilitatorContract:r});let d=s||this.getDefaultRpcUrl(t.network),c=i?.fee??"0",m=BigInt(o)+BigInt(c),p=await this.readAllowance(d,n,e.address,r);if(p<m){if(!e.sendTransaction)throw new Error("BSC payments require a wallet that supports sendTransaction for the one-time token approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.");let I=this.calculateApprovalAmount(o,c,i?.approvalStrategy);this.log(`Approving ${I} for ${r} (current allowance: ${p})`);let N=await e.sendTransaction({to:n,data:this.encodeApprove(r,I),value:0n});this.log(`Approval tx sent: ${N}`),await this.waitForReceipt(d,N),this.log("Approval confirmed")}else this.log("Sufficient allowance, skipping approval");let f=new Uint8Array(16);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(f);let l=[...f].reduce((I,N)=>I*256n+BigInt(N),0n).toString(),E=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(E);let g="0x"+[...E].map(I=>I.toString(16).padStart(2,"0")).join(""),S=Math.floor(Date.now()/1e3)+(t.maxTimeoutSeconds||300),h=i?.eip712Domain,C=h?{name:h.name,version:h.version,chainId:BigInt(h.chainId),verifyingContract:h.verifyingContract}:{name:"DexterBSCFacilitator",version:"1",chainId:BigInt(this.getChainId(t.network)),verifyingContract:r},y=i?.eip712Types??{Payment:[{name:"from",type:"address"},{name:"to",type:"address"},{name:"token",type:"address"},{name:"amount",type:"uint256"},{name:"fee",type:"uint256"},{name:"nonce",type:"uint256"},{name:"deadline",type:"uint256"},{name:"paymentId",type:"bytes32"}]},R={from:e.address,to:a,token:n,amount:BigInt(o),fee:BigInt(c),nonce:BigInt(l),deadline:BigInt(S),paymentId:g},_=await e.signTypedData({domain:C,types:y,primaryType:"Payment",message:R});this.log("EIP-712 Payment signature obtained");let v={from:e.address,to:a,token:n,amount:o,fee:c,nonce:l,deadline:S,paymentId:g,signature:_};return{serialized:JSON.stringify(v),signature:_}}async buildPermit2Transaction(t,e,s){let{payTo:a,asset:n}=t,i=t.amount??t.maxAmountRequired;if(!i)throw new Error("Missing amount in payment requirements");if(!e.signTypedData)throw new Error("Wallet does not support signTypedData (EIP-712)");this.log("Building Permit2 transaction:",{from:e.address,to:a,amount:i,asset:n,network:t.network});let o=s||this.getDefaultRpcUrl(t.network),r=await this.readAllowance(o,n,e.address,w),d;if(r<BigInt(i)){let C=this.encodeApprove(w,It);if(e.signTransaction){this.log(`Signing Permit2 approval for relay (current allowance: ${r})`);let y=this.getChainId(t.network),R=await this.readGasPrice(o),_=await this.readNonce(o,e.address),v=await e.signTransaction({to:n,data:C,chainId:y,gas:50000n,gasPrice:R,nonce:_});d={erc20ApprovalGasSponsoring:{info:{from:e.address,asset:n,spender:w,amount:It.toString(),signedTransaction:v,version:"1"}}},this.log("Permit2 approval signed for facilitator relay")}else if(e.sendTransaction){this.log(`Approving Permit2 directly (current allowance: ${r})`);let y=await e.sendTransaction({to:n,data:C,value:0n});this.log(`Permit2 approval tx sent: ${y}`),await this.waitForReceipt(o,y),this.log("Permit2 approval confirmed")}else throw new Error("Permit2 payments require a wallet that supports signTransaction or sendTransaction for the one-time Permit2 approval. Use createEvmKeypairWallet() or a browser wallet with transaction support.")}else this.log("Sufficient Permit2 allowance, skipping approval");let c=new Uint8Array(32);(globalThis.crypto??(await import("crypto")).webcrypto).getRandomValues(c);let m=[...c].reduce((C,y)=>C*256n+BigInt(y),0n),p=Math.floor(Date.now()/1e3),f=p-600,l=p+(t.maxTimeoutSeconds||300),E=this.getChainId(t.network),g={name:"Permit2",chainId:BigInt(E),verifyingContract:w},T={permitted:{token:n,amount:BigInt(i)},spender:V,nonce:m,deadline:BigInt(l),witness:{to:a,validAfter:BigInt(f)}},S=await e.signTypedData({domain:g,types:Rt,primaryType:"PermitWitnessTransferFrom",message:T});this.log("Permit2 PermitWitnessTransferFrom signature obtained");let h={signature:S,permit2Authorization:{from:e.address,permitted:{token:n,amount:i},spender:V,nonce:m.toString(),deadline:String(l),witness:{to:a,validAfter:String(f)}}};return{serialized:JSON.stringify(h),signature:S,extensions:d,settlementProbe:{kind:"permit2",from:e.address,nonce:m.toString(),chainId:E}}}async readAllowance(t,e,s,a){let n="0xdd62ed3e",i=s.slice(2).toLowerCase().padStart(64,"0"),o=a.slice(2).toLowerCase().padStart(64,"0"),r=n+i+o;try{let c=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_call",params:[{to:e,data:r},"latest"]})})).json();return c.error||!c.result||c.result==="0x"?0n:BigInt(c.result)}catch{return 0n}}encodeApprove(t,e){let s="0x095ea7b3",a=t.slice(2).toLowerCase().padStart(64,"0"),n=e.toString(16).padStart(64,"0");return s+a+n}async waitForReceipt(t,e,s=3e4){let a=Date.now();for(;Date.now()-a<s;){try{let i=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionReceipt",params:[e]})})).json();if(i.result){if(i.result.status==="0x0")throw new Error(`Approval transaction reverted: ${e}`);return}}catch(n){if(n instanceof Error&&n.message.includes("reverted"))throw n}await new Promise(n=>setTimeout(n,2e3))}throw new Error(`Approval transaction receipt timeout after ${s}ms: ${e}`)}async readGasPrice(t){try{let s=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_gasPrice",params:[]})})).json();return s.result?BigInt(s.result):50000000n}catch{return 50000000n}}async readNonce(t,e){try{let a=await(await fetch(t,{method:"POST",headers:{"Content-Type":"application/json"},body:JSON.stringify({jsonrpc:"2.0",id:1,method:"eth_getTransactionCount",params:[e,"latest"]})})).json();return a.result?parseInt(a.result,16):0}catch{return 0}}calculateApprovalAmount(t,e,s){let a=BigInt(t)+BigInt(e);if(!s||s.mode==="exact")return a;let n=BigInt(s.defaultMultiple??10),i=a*n;if(s.maxCapUsd){let o=this.inferDecimals(t),r=BigInt(Math.floor(s.maxCapUsd*Math.pow(10,o)));if(i>r)return r}if(s.exactAboveUsd){let o=this.inferDecimals(t),r=BigInt(Math.floor(s.exactAboveUsd*Math.pow(10,o)));if(BigInt(t)>r)return a}return i}inferDecimals(t){return t.length>12?18:6}};function lt(u){return new ct(u)}function $t(u){if(u===St||u===ht)return!0;let t=u.toLowerCase();for(let e of Object.values(ot))if(e.toLowerCase()===t)return!0;for(let e of Object.keys(st))if(e.toLowerCase()===t)return!0;return!1}function qt(u=!1){return[ft({verbose:u}),lt({verbose:u})]}function Ht(u,t){return u.find(e=>e.canHandle(t))}export{M as ARBITRUM_ONE,k as AVALANCHE,B as BASE_MAINNET,nt as BASE_SEPOLIA,W as BSC_MAINNET,st as BSC_STABLECOIN_ADDRESSES,rt as BSC_USDC,dt as BSC_USDT,j as ETHEREUM_MAINNET,ct as EvmAdapter,U as OPTIMISM,w as PERMIT2_ADDRESS,L as POLYGON,K as SKALE_BASE,F as SKALE_BASE_SEPOLIA,D as SOLANA_DEVNET,b as SOLANA_MAINNET,O as SOLANA_TESTNET,it as SolanaAdapter,ot as USDC_ADDRESSES,V as X402_EXACT_PERMIT2_PROXY,qt as createDefaultAdapters,lt as createEvmAdapter,ft as createSolanaAdapter,Ht as findAdapter,q as isEvmWallet,$t as isKnownUSDC,$ as isSolanaWallet};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
"use strict";var nt=Object.create;var k=Object.defineProperty;var at=Object.getOwnPropertyDescriptor;var rt=Object.getOwnPropertyNames;var it=Object.getPrototypeOf,ot=Object.prototype.hasOwnProperty;var st=(t,e)=>{for(var n in e)k(t,n,{get:e[n],enumerable:!0})},$=(t,e,n,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let r of rt(e))!ot.call(t,r)&&r!==n&&k(t,r,{get:()=>e[r],enumerable:!(a=at(e,r))||a.enumerable});return t};var lt=(t,e,n)=>(n=t!=null?nt(it(t)):{},$(e||!t||!t.__esModule?k(n,"default",{value:t,enumerable:!0}):n,t)),ct=t=>$(k({},"__esModule",{value:!0}),t);var ft={};st(ft,{InsufficientBalanceError:()=>A,UnsupportedNetworkError:()=>f,WithdrawNotReadyError:()=>y,createFileChannelStore:()=>P,createLocalStorageChannelStore:()=>B,getDefaultChannelStore:()=>W,openBatchChannel:()=>Q,resumeBatchChannel:()=>Y});module.exports=ct(ft);var s=require("viem"),C=require("viem/chains"),X=require("@x402/evm"),I=require("@x402/evm/batch-settlement/client"),T=require("@x402/core/client");var O=require("os"),N=require("path"),j=require("@x402/evm/batch-settlement/client");function P(t){return new j.FileClientChannelStorage({directory:t})}var E="dexter-x402-channel:";function B(t){return{async get(e){let n=t.getItem(E+e);return n?JSON.parse(n):void 0},async set(e,n){t.setItem(E+e,JSON.stringify(n))},async delete(e){t.removeItem(E+e)}}}function W(){let t=globalThis.localStorage;return t?B(t):P((0,N.join)((0,O.homedir)(),".dexter-x402","channels"))}var A=class extends Error{constructor(e){super(e),this.name="InsufficientBalanceError"}},f=class extends Error{constructor(e){super(e),this.name="UnsupportedNetworkError"}};var L=require("viem"),z=require("@x402/evm/batch-settlement/client"),g=require("@x402/evm"),y=class extends Error{constructor(e){super(e),this.name="WithdrawNotReadyError"}};function M(t,e){return t===0?0:t+e}var dt=6,H=[{name:"payer",type:"address"},{name:"payerAuthorizer",type:"address"},{name:"receiver",type:"address"},{name:"receiverAuthorizer",type:"address"},{name:"token",type:"address"},{name:"withdrawDelay",type:"uint40"},{name:"salt",type:"bytes32"}],R=[{type:"function",name:"initiateWithdraw",inputs:[{name:"config",type:"tuple",components:H},{name:"amount",type:"uint128"}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"finalizeWithdraw",inputs:[{name:"config",type:"tuple",components:H}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"pendingWithdrawals",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"amount",type:"uint128"},{name:"initiatedAt",type:"uint40"}],stateMutability:"view"},{type:"function",name:"channels",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"balance",type:"uint128"},{name:"totalClaimed",type:"uint128"}],stateMutability:"view"}];function ht(t){if(Array.isArray(t))return{balance:BigInt(t[0]),totalClaimed:BigInt(t[1])};let e=t;return{balance:BigInt(e.balance),totalClaimed:BigInt(e.totalClaimed)}}function q(t){if(Array.isArray(t))return{amount:BigInt(t[0]),initiatedAt:Number(t[1])};let e=t;return{amount:BigInt(e.amount),initiatedAt:Number(e.initiatedAt)}}async function _(t){let{config:e,network:n,client:a,withdrawDelaySecs:r}=t,i=(0,z.computeChannelId)(e,n),l=t.amountAtomic;l===void 0&&(l=ht(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"channels",args:[i]})).balance);let c=await a.writeContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"initiateWithdraw",args:[e,l]});await a.waitForTransactionReceipt({hash:c});let d=q(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"pendingWithdrawals",args:[i]}));return{initiateTx:c,finalizableAt:M(d.initiatedAt,r)}}async function J(t){let{config:e,network:n,client:a,withdrawDelaySecs:r}=t,i=(0,z.computeChannelId)(e,n),l=q(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"pendingWithdrawals",args:[i]}));if(l.initiatedAt===0)throw new y(`no withdrawal is pending for channel ${i} \u2014 call forceWithdraw first`);let c=M(l.initiatedAt,r),d=Math.floor(Date.now()/1e3);if(d<c)throw new y(`withdrawal for channel ${i} is not finalizable until ${c} (unix seconds) \u2014 ${c-d}s remaining`);let o=await a.writeContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"finalizeWithdraw",args:[e]});return await a.waitForTransactionReceipt({hash:o}),{finalizeTx:o,withdrawnAmount:(0,L.formatUnits)(l.amount,dt)}}var V={"eip155:8453":{chain:C.base,defaultRpc:"https://
|
|
1
|
+
"use strict";var nt=Object.create;var k=Object.defineProperty;var at=Object.getOwnPropertyDescriptor;var rt=Object.getOwnPropertyNames;var it=Object.getPrototypeOf,ot=Object.prototype.hasOwnProperty;var st=(t,e)=>{for(var n in e)k(t,n,{get:e[n],enumerable:!0})},$=(t,e,n,a)=>{if(e&&typeof e=="object"||typeof e=="function")for(let r of rt(e))!ot.call(t,r)&&r!==n&&k(t,r,{get:()=>e[r],enumerable:!(a=at(e,r))||a.enumerable});return t};var lt=(t,e,n)=>(n=t!=null?nt(it(t)):{},$(e||!t||!t.__esModule?k(n,"default",{value:t,enumerable:!0}):n,t)),ct=t=>$(k({},"__esModule",{value:!0}),t);var ft={};st(ft,{InsufficientBalanceError:()=>A,UnsupportedNetworkError:()=>f,WithdrawNotReadyError:()=>y,createFileChannelStore:()=>P,createLocalStorageChannelStore:()=>B,getDefaultChannelStore:()=>W,openBatchChannel:()=>Q,resumeBatchChannel:()=>Y});module.exports=ct(ft);var s=require("viem"),C=require("viem/chains"),X=require("@x402/evm"),I=require("@x402/evm/batch-settlement/client"),T=require("@x402/core/client");var O=require("os"),N=require("path"),j=require("@x402/evm/batch-settlement/client");function P(t){return new j.FileClientChannelStorage({directory:t})}var E="dexter-x402-channel:";function B(t){return{async get(e){let n=t.getItem(E+e);return n?JSON.parse(n):void 0},async set(e,n){t.setItem(E+e,JSON.stringify(n))},async delete(e){t.removeItem(E+e)}}}function W(){let t=globalThis.localStorage;return t?B(t):P((0,N.join)((0,O.homedir)(),".dexter-x402","channels"))}var A=class extends Error{constructor(e){super(e),this.name="InsufficientBalanceError"}},f=class extends Error{constructor(e){super(e),this.name="UnsupportedNetworkError"}};var L=require("viem"),z=require("@x402/evm/batch-settlement/client"),g=require("@x402/evm"),y=class extends Error{constructor(e){super(e),this.name="WithdrawNotReadyError"}};function M(t,e){return t===0?0:t+e}var dt=6,H=[{name:"payer",type:"address"},{name:"payerAuthorizer",type:"address"},{name:"receiver",type:"address"},{name:"receiverAuthorizer",type:"address"},{name:"token",type:"address"},{name:"withdrawDelay",type:"uint40"},{name:"salt",type:"bytes32"}],R=[{type:"function",name:"initiateWithdraw",inputs:[{name:"config",type:"tuple",components:H},{name:"amount",type:"uint128"}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"finalizeWithdraw",inputs:[{name:"config",type:"tuple",components:H}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"pendingWithdrawals",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"amount",type:"uint128"},{name:"initiatedAt",type:"uint40"}],stateMutability:"view"},{type:"function",name:"channels",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"balance",type:"uint128"},{name:"totalClaimed",type:"uint128"}],stateMutability:"view"}];function ht(t){if(Array.isArray(t))return{balance:BigInt(t[0]),totalClaimed:BigInt(t[1])};let e=t;return{balance:BigInt(e.balance),totalClaimed:BigInt(e.totalClaimed)}}function q(t){if(Array.isArray(t))return{amount:BigInt(t[0]),initiatedAt:Number(t[1])};let e=t;return{amount:BigInt(e.amount),initiatedAt:Number(e.initiatedAt)}}async function _(t){let{config:e,network:n,client:a,withdrawDelaySecs:r}=t,i=(0,z.computeChannelId)(e,n),l=t.amountAtomic;l===void 0&&(l=ht(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"channels",args:[i]})).balance);let c=await a.writeContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"initiateWithdraw",args:[e,l]});await a.waitForTransactionReceipt({hash:c});let d=q(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"pendingWithdrawals",args:[i]}));return{initiateTx:c,finalizableAt:M(d.initiatedAt,r)}}async function J(t){let{config:e,network:n,client:a,withdrawDelaySecs:r}=t,i=(0,z.computeChannelId)(e,n),l=q(await a.readContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"pendingWithdrawals",args:[i]}));if(l.initiatedAt===0)throw new y(`no withdrawal is pending for channel ${i} \u2014 call forceWithdraw first`);let c=M(l.initiatedAt,r),d=Math.floor(Date.now()/1e3);if(d<c)throw new y(`withdrawal for channel ${i} is not finalizable until ${c} (unix seconds) \u2014 ${c-d}s remaining`);let o=await a.writeContract({address:g.BATCH_SETTLEMENT_ADDRESS,abi:R,functionName:"finalizeWithdraw",args:[e]});return await a.waitForTransactionReceipt({hash:o}),{finalizeTx:o,withdrawnAmount:(0,L.formatUnits)(l.amount,dt)}}var V={"eip155:8453":{chain:C.base,defaultRpc:"https://api.dexter.cash/api/evm/base/rpc"},"eip155:42161":{chain:C.arbitrum,defaultRpc:"https://api.dexter.cash/api/evm/arbitrum/rpc"},"eip155:137":{chain:C.polygon,defaultRpc:"https://api.dexter.cash/api/evm/polygon/rpc"}};function G(t){let e=V[t.network];if(!e)throw new f(`batch-settlement is not available on network "${t.network}" \u2014 supported: ${Object.keys(V).join(", ")}`);let n=t.rpcUrl??e.defaultRpc,a=t.wallet.signTypedData;if(typeof a!="function")throw new Error("batch-settlement requires an EvmWallet that supports signTypedData (EIP-712)");let r=(0,s.createWalletClient)({account:{address:t.wallet.address,type:"json-rpc"},chain:e.chain,transport:(0,s.http)(n)}).extend(s.publicActions),i=t.wallet.sendTransaction,l=typeof i=="function",c=async u=>{let b=u.to??void 0;if(!b)throw new Error("batch-settlement escape hatch: transaction is missing a `to` address");let p=u.data??"0x";if(typeof i=="function")return await i.call(t.wallet,{to:b,data:p,value:u.value});throw new Error("batch-settlement: wallet has no sendTransaction")},d=Object.assign(r,{address:t.wallet.address,signTypedData:u=>a.call(t.wallet,u),sendTransaction:c}),o=(0,X.toClientEvmSigner)(d),m=new I.BatchSettlementEvmScheme(o,{storage:t.store,depositPolicy:{depositMultiplier:4},depositStrategy:()=>t.depositAtomic,salt:t.salt}),h=new T.x402Client;h.register(t.network,m);let w=new T.x402HTTPClient(h);return{scheme:m,x402Cli:h,httpClient:w,rpcUrl:n,withdrawClient:d,canSubmitTransactions:l}}var v=6;function ut(t,e){let n=e>t?0n:t-e;return{deposited:(0,s.formatUnits)(t,v),spent:(0,s.formatUnits)(e,v),remaining:(0,s.formatUnits)(n,v)}}function pt(t){let e=t.payload;if(!e||typeof e!="object")return"";let n=e.voucher;if(!n||typeof n!="object")return"";let a=n.channelId;return typeof a=="string"?a:""}function mt(t){let e=t.payload;if(!e||typeof e!="object")return;let n=e.channelConfig;if(!(!n||typeof n!="object"))return n}function K(t){let{stack:e,store:n,network:a}=t,r="",i,l=t.depositedAtomic,c=0n;async function d(){if(!r)return;let o=await(0,I.getChannel)(n,r);o&&(o.chargedCumulativeAmount!==void 0&&(c=BigInt(o.chargedCumulativeAmount)),l===0n&&o.balance!==void 0&&(l=BigInt(o.balance)))}return{get channelId(){return r},get salt(){return t.salt},get network(){return a},get state(){return ut(l,c)},async fetch(o,m){let{httpClient:h}=e,w=await fetch(o,m);if(w.status!==402)return w;let F=h.getPaymentRequiredResponse(p=>w.headers.get(p),await w.clone().json().catch(()=>{})),u=2,b;for(let p=1;p<=u;p+=1){let S=await h.createPaymentPayload(F),D=pt(S);D&&(r=D);let U=mt(S);U&&(i=U);let Z=h.encodePaymentSignatureHeader(S),x=await fetch(o,{...m,headers:{...m?.headers??{},...Z}});b=x;let{recovered:tt}=await h.processPaymentResult(S,et=>x.headers.get(et),x.status);if(await d(),!(x.status===402&&tt&&p<u))return x}return b},async close(){return r&&await n.delete(r.toLowerCase()),{closed:!0}},async forceWithdraw(){if(!e.canSubmitTransactions)throw new Error("batch-settlement forceWithdraw requires a wallet with a sendTransaction method (the withdrawal escape hatch submits an on-chain transaction and the buyer pays gas; a signature-only wallet cannot use it).");if(!i)throw new Error("forceWithdraw is unavailable until the channel has resolved on-chain \u2014 make at least one fetch() against the channel first");return _({config:i,network:a,client:e.withdrawClient,withdrawDelaySecs:i.withdrawDelay})},async finalizeWithdraw(){if(!e.canSubmitTransactions)throw new Error("batch-settlement finalizeWithdraw requires a wallet with a sendTransaction method (the withdrawal escape hatch submits an on-chain transaction and the buyer pays gas; a signature-only wallet cannot use it).");if(!i)throw new Error("finalizeWithdraw is unavailable until the channel has resolved on-chain \u2014 make at least one fetch() against the channel first");return J({config:i,network:a,client:e.withdrawClient,withdrawDelaySecs:i.withdrawDelay})}}}async function wt(){let t=globalThis.crypto??(await import("crypto")).webcrypto,e=new Uint8Array(32);return t.getRandomValues(e),(0,s.toHex)(e)}async function Q(t){let e=t.store??W(),n=t.salt??await wt(),a;try{a=(0,s.parseUnits)(t.deposit,v)}catch{throw new Error(`deposit must be a valid USDC amount in decimal units (e.g. "0.30"), got "${t.deposit}"`)}if(a<=0n)throw new Error(`deposit must be a positive amount, got "${t.deposit}"`);let r=G({wallet:t.wallet,network:t.network,rpcUrl:t.rpcUrl,store:e,depositAtomic:a.toString(),salt:n});return K({stack:r,store:e,network:t.network,depositedAtomic:a,salt:n})}async function Y(t){let e=t.store??W(),n=G({wallet:t.wallet,network:t.network,rpcUrl:t.rpcUrl,store:e,depositAtomic:"0",salt:t.salt});return K({stack:n,store:e,network:t.network,depositedAtomic:0n,salt:t.salt})}0&&(module.exports={InsufficientBalanceError,UnsupportedNetworkError,WithdrawNotReadyError,createFileChannelStore,createLocalStorageChannelStore,getDefaultChannelStore,openBatchChannel,resumeBatchChannel});
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import{createWalletClient as K,http as Q,publicActions as Y,parseUnits as Z,formatUnits as A,toHex as tt}from"viem";import{base as et,arbitrum as nt,polygon as at}from"viem/chains";import{toClientEvmSigner as rt}from"@x402/evm";import{BatchSettlementEvmScheme as it,getChannel as ot}from"@x402/evm/batch-settlement/client";import{x402Client as st,x402HTTPClient as lt}from"@x402/core/client";import{homedir as q}from"os";import{join as _}from"path";import{FileClientChannelStorage as J}from"@x402/evm/batch-settlement/client";function E(t){return new J({directory:t})}var R="dexter-x402-channel:";function P(t){return{async get(e){let n=t.getItem(R+e);return n?JSON.parse(n):void 0},async set(e,n){t.setItem(R+e,JSON.stringify(n))},async delete(e){t.removeItem(R+e)}}}function x(){let t=globalThis.localStorage;return t?P(t):E(_(q(),".dexter-x402","channels"))}var k=class extends Error{constructor(e){super(e),this.name="InsufficientBalanceError"}},y=class extends Error{constructor(e){super(e),this.name="UnsupportedNetworkError"}};import{formatUnits as V}from"viem";import{computeChannelId as z}from"@x402/evm/batch-settlement/client";import{BATCH_SETTLEMENT_ADDRESS as C}from"@x402/evm";var S=class extends Error{constructor(e){super(e),this.name="WithdrawNotReadyError"}};function F(t,e){return t===0?0:t+e}var X=6,B=[{name:"payer",type:"address"},{name:"payerAuthorizer",type:"address"},{name:"receiver",type:"address"},{name:"receiverAuthorizer",type:"address"},{name:"token",type:"address"},{name:"withdrawDelay",type:"uint40"},{name:"salt",type:"bytes32"}],b=[{type:"function",name:"initiateWithdraw",inputs:[{name:"config",type:"tuple",components:B},{name:"amount",type:"uint128"}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"finalizeWithdraw",inputs:[{name:"config",type:"tuple",components:B}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"pendingWithdrawals",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"amount",type:"uint128"},{name:"initiatedAt",type:"uint40"}],stateMutability:"view"},{type:"function",name:"channels",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"balance",type:"uint128"},{name:"totalClaimed",type:"uint128"}],stateMutability:"view"}];function G(t){if(Array.isArray(t))return{balance:BigInt(t[0]),totalClaimed:BigInt(t[1])};let e=t;return{balance:BigInt(e.balance),totalClaimed:BigInt(e.totalClaimed)}}function D(t){if(Array.isArray(t))return{amount:BigInt(t[0]),initiatedAt:Number(t[1])};let e=t;return{amount:BigInt(e.amount),initiatedAt:Number(e.initiatedAt)}}async function U(t){let{config:e,network:n,client:a,withdrawDelaySecs:i}=t,r=z(e,n),s=t.amountAtomic;s===void 0&&(s=G(await a.readContract({address:C,abi:b,functionName:"channels",args:[r]})).balance);let l=await a.writeContract({address:C,abi:b,functionName:"initiateWithdraw",args:[e,s]});await a.waitForTransactionReceipt({hash:l});let c=D(await a.readContract({address:C,abi:b,functionName:"pendingWithdrawals",args:[r]}));return{initiateTx:l,finalizableAt:F(c.initiatedAt,i)}}async function $(t){let{config:e,network:n,client:a,withdrawDelaySecs:i}=t,r=z(e,n),s=D(await a.readContract({address:C,abi:b,functionName:"pendingWithdrawals",args:[r]}));if(s.initiatedAt===0)throw new S(`no withdrawal is pending for channel ${r} \u2014 call forceWithdraw first`);let l=F(s.initiatedAt,i),c=Math.floor(Date.now()/1e3);if(c<l)throw new S(`withdrawal for channel ${r} is not finalizable until ${l} (unix seconds) \u2014 ${l-c}s remaining`);let o=await a.writeContract({address:C,abi:b,functionName:"finalizeWithdraw",args:[e]});return await a.waitForTransactionReceipt({hash:o}),{finalizeTx:o,withdrawnAmount:V(s.amount,X)}}var O={"eip155:8453":{chain:et,defaultRpc:"https://
|
|
1
|
+
import{createWalletClient as K,http as Q,publicActions as Y,parseUnits as Z,formatUnits as A,toHex as tt}from"viem";import{base as et,arbitrum as nt,polygon as at}from"viem/chains";import{toClientEvmSigner as rt}from"@x402/evm";import{BatchSettlementEvmScheme as it,getChannel as ot}from"@x402/evm/batch-settlement/client";import{x402Client as st,x402HTTPClient as lt}from"@x402/core/client";import{homedir as q}from"os";import{join as _}from"path";import{FileClientChannelStorage as J}from"@x402/evm/batch-settlement/client";function E(t){return new J({directory:t})}var R="dexter-x402-channel:";function P(t){return{async get(e){let n=t.getItem(R+e);return n?JSON.parse(n):void 0},async set(e,n){t.setItem(R+e,JSON.stringify(n))},async delete(e){t.removeItem(R+e)}}}function x(){let t=globalThis.localStorage;return t?P(t):E(_(q(),".dexter-x402","channels"))}var k=class extends Error{constructor(e){super(e),this.name="InsufficientBalanceError"}},y=class extends Error{constructor(e){super(e),this.name="UnsupportedNetworkError"}};import{formatUnits as V}from"viem";import{computeChannelId as z}from"@x402/evm/batch-settlement/client";import{BATCH_SETTLEMENT_ADDRESS as C}from"@x402/evm";var S=class extends Error{constructor(e){super(e),this.name="WithdrawNotReadyError"}};function F(t,e){return t===0?0:t+e}var X=6,B=[{name:"payer",type:"address"},{name:"payerAuthorizer",type:"address"},{name:"receiver",type:"address"},{name:"receiverAuthorizer",type:"address"},{name:"token",type:"address"},{name:"withdrawDelay",type:"uint40"},{name:"salt",type:"bytes32"}],b=[{type:"function",name:"initiateWithdraw",inputs:[{name:"config",type:"tuple",components:B},{name:"amount",type:"uint128"}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"finalizeWithdraw",inputs:[{name:"config",type:"tuple",components:B}],outputs:[],stateMutability:"nonpayable"},{type:"function",name:"pendingWithdrawals",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"amount",type:"uint128"},{name:"initiatedAt",type:"uint40"}],stateMutability:"view"},{type:"function",name:"channels",inputs:[{name:"channelId",type:"bytes32"}],outputs:[{name:"balance",type:"uint128"},{name:"totalClaimed",type:"uint128"}],stateMutability:"view"}];function G(t){if(Array.isArray(t))return{balance:BigInt(t[0]),totalClaimed:BigInt(t[1])};let e=t;return{balance:BigInt(e.balance),totalClaimed:BigInt(e.totalClaimed)}}function D(t){if(Array.isArray(t))return{amount:BigInt(t[0]),initiatedAt:Number(t[1])};let e=t;return{amount:BigInt(e.amount),initiatedAt:Number(e.initiatedAt)}}async function U(t){let{config:e,network:n,client:a,withdrawDelaySecs:i}=t,r=z(e,n),s=t.amountAtomic;s===void 0&&(s=G(await a.readContract({address:C,abi:b,functionName:"channels",args:[r]})).balance);let l=await a.writeContract({address:C,abi:b,functionName:"initiateWithdraw",args:[e,s]});await a.waitForTransactionReceipt({hash:l});let c=D(await a.readContract({address:C,abi:b,functionName:"pendingWithdrawals",args:[r]}));return{initiateTx:l,finalizableAt:F(c.initiatedAt,i)}}async function $(t){let{config:e,network:n,client:a,withdrawDelaySecs:i}=t,r=z(e,n),s=D(await a.readContract({address:C,abi:b,functionName:"pendingWithdrawals",args:[r]}));if(s.initiatedAt===0)throw new S(`no withdrawal is pending for channel ${r} \u2014 call forceWithdraw first`);let l=F(s.initiatedAt,i),c=Math.floor(Date.now()/1e3);if(c<l)throw new S(`withdrawal for channel ${r} is not finalizable until ${l} (unix seconds) \u2014 ${l-c}s remaining`);let o=await a.writeContract({address:C,abi:b,functionName:"finalizeWithdraw",args:[e]});return await a.waitForTransactionReceipt({hash:o}),{finalizeTx:o,withdrawnAmount:V(s.amount,X)}}var O={"eip155:8453":{chain:et,defaultRpc:"https://api.dexter.cash/api/evm/base/rpc"},"eip155:42161":{chain:nt,defaultRpc:"https://api.dexter.cash/api/evm/arbitrum/rpc"},"eip155:137":{chain:at,defaultRpc:"https://api.dexter.cash/api/evm/polygon/rpc"}};function N(t){let e=O[t.network];if(!e)throw new y(`batch-settlement is not available on network "${t.network}" \u2014 supported: ${Object.keys(O).join(", ")}`);let n=t.rpcUrl??e.defaultRpc,a=t.wallet.signTypedData;if(typeof a!="function")throw new Error("batch-settlement requires an EvmWallet that supports signTypedData (EIP-712)");let i=K({account:{address:t.wallet.address,type:"json-rpc"},chain:e.chain,transport:Q(n)}).extend(Y),r=t.wallet.sendTransaction,s=typeof r=="function",l=async h=>{let w=h.to??void 0;if(!w)throw new Error("batch-settlement escape hatch: transaction is missing a `to` address");let u=h.data??"0x";if(typeof r=="function")return await r.call(t.wallet,{to:w,data:u,value:h.value});throw new Error("batch-settlement: wallet has no sendTransaction")},c=Object.assign(i,{address:t.wallet.address,signTypedData:h=>a.call(t.wallet,h),sendTransaction:l}),o=rt(c),p=new it(o,{storage:t.store,depositPolicy:{depositMultiplier:4},depositStrategy:()=>t.depositAtomic,salt:t.salt}),d=new st;d.register(t.network,p);let m=new lt(d);return{scheme:p,x402Cli:d,httpClient:m,rpcUrl:n,withdrawClient:c,canSubmitTransactions:s}}var W=6;function ct(t,e){let n=e>t?0n:t-e;return{deposited:A(t,W),spent:A(e,W),remaining:A(n,W)}}function dt(t){let e=t.payload;if(!e||typeof e!="object")return"";let n=e.voucher;if(!n||typeof n!="object")return"";let a=n.channelId;return typeof a=="string"?a:""}function ht(t){let e=t.payload;if(!e||typeof e!="object")return;let n=e.channelConfig;if(!(!n||typeof n!="object"))return n}function j(t){let{stack:e,store:n,network:a}=t,i="",r,s=t.depositedAtomic,l=0n;async function c(){if(!i)return;let o=await ot(n,i);o&&(o.chargedCumulativeAmount!==void 0&&(l=BigInt(o.chargedCumulativeAmount)),s===0n&&o.balance!==void 0&&(s=BigInt(o.balance)))}return{get channelId(){return i},get salt(){return t.salt},get network(){return a},get state(){return ct(s,l)},async fetch(o,p){let{httpClient:d}=e,m=await fetch(o,p);if(m.status!==402)return m;let v=d.getPaymentRequiredResponse(u=>m.headers.get(u),await m.clone().json().catch(()=>{})),h=2,w;for(let u=1;u<=h;u+=1){let f=await d.createPaymentPayload(v),I=dt(f);I&&(i=I);let T=ht(f);T&&(r=T);let H=d.encodePaymentSignatureHeader(f),g=await fetch(o,{...p,headers:{...p?.headers??{},...H}});w=g;let{recovered:L}=await d.processPaymentResult(f,M=>g.headers.get(M),g.status);if(await c(),!(g.status===402&&L&&u<h))return g}return w},async close(){return i&&await n.delete(i.toLowerCase()),{closed:!0}},async forceWithdraw(){if(!e.canSubmitTransactions)throw new Error("batch-settlement forceWithdraw requires a wallet with a sendTransaction method (the withdrawal escape hatch submits an on-chain transaction and the buyer pays gas; a signature-only wallet cannot use it).");if(!r)throw new Error("forceWithdraw is unavailable until the channel has resolved on-chain \u2014 make at least one fetch() against the channel first");return U({config:r,network:a,client:e.withdrawClient,withdrawDelaySecs:r.withdrawDelay})},async finalizeWithdraw(){if(!e.canSubmitTransactions)throw new Error("batch-settlement finalizeWithdraw requires a wallet with a sendTransaction method (the withdrawal escape hatch submits an on-chain transaction and the buyer pays gas; a signature-only wallet cannot use it).");if(!r)throw new Error("finalizeWithdraw is unavailable until the channel has resolved on-chain \u2014 make at least one fetch() against the channel first");return $({config:r,network:a,client:e.withdrawClient,withdrawDelaySecs:r.withdrawDelay})}}}async function ut(){let t=globalThis.crypto??(await import("crypto")).webcrypto,e=new Uint8Array(32);return t.getRandomValues(e),tt(e)}async function pt(t){let e=t.store??x(),n=t.salt??await ut(),a;try{a=Z(t.deposit,W)}catch{throw new Error(`deposit must be a valid USDC amount in decimal units (e.g. "0.30"), got "${t.deposit}"`)}if(a<=0n)throw new Error(`deposit must be a positive amount, got "${t.deposit}"`);let i=N({wallet:t.wallet,network:t.network,rpcUrl:t.rpcUrl,store:e,depositAtomic:a.toString(),salt:n});return j({stack:i,store:e,network:t.network,depositedAtomic:a,salt:n})}async function mt(t){let e=t.store??x(),n=N({wallet:t.wallet,network:t.network,rpcUrl:t.rpcUrl,store:e,depositAtomic:"0",salt:t.salt});return j({stack:n,store:e,network:t.network,depositedAtomic:0n,salt:t.salt})}export{k as InsufficientBalanceError,y as UnsupportedNetworkError,S as WithdrawNotReadyError,E as createFileChannelStore,P as createLocalStorageChannelStore,x as getDefaultChannelStore,pt as openBatchChannel,mt as resumeBatchChannel};
|