moltspay 1.2.0 → 1.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (54) hide show
  1. package/README.md +78 -9
  2. package/dist/cdp/index.d.mts +1 -1
  3. package/dist/cdp/index.d.ts +1 -1
  4. package/dist/cdp/index.js +16 -49
  5. package/dist/cdp/index.js.map +1 -1
  6. package/dist/cdp/index.mjs +16 -49
  7. package/dist/cdp/index.mjs.map +1 -1
  8. package/dist/chains/index.d.mts +1 -1
  9. package/dist/chains/index.d.ts +1 -1
  10. package/dist/chains/index.js +16 -49
  11. package/dist/chains/index.js.map +1 -1
  12. package/dist/chains/index.mjs +16 -49
  13. package/dist/chains/index.mjs.map +1 -1
  14. package/dist/cli/index.js +180 -111
  15. package/dist/cli/index.js.map +1 -1
  16. package/dist/cli/index.mjs +180 -111
  17. package/dist/cli/index.mjs.map +1 -1
  18. package/dist/client/index.d.mts +3 -3
  19. package/dist/client/index.d.ts +3 -3
  20. package/dist/client/index.js +32 -58
  21. package/dist/client/index.js.map +1 -1
  22. package/dist/client/index.mjs +32 -58
  23. package/dist/client/index.mjs.map +1 -1
  24. package/dist/facilitators/index.d.mts +12 -6
  25. package/dist/facilitators/index.d.ts +12 -6
  26. package/dist/facilitators/index.js +39 -33
  27. package/dist/facilitators/index.js.map +1 -1
  28. package/dist/facilitators/index.mjs +39 -33
  29. package/dist/facilitators/index.mjs.map +1 -1
  30. package/dist/{index-B3v8IWjM.d.mts → index-DgJPZMBG.d.mts} +2 -1
  31. package/dist/{index-B3v8IWjM.d.ts → index-DgJPZMBG.d.ts} +2 -1
  32. package/dist/index.d.mts +1 -1
  33. package/dist/index.d.ts +1 -1
  34. package/dist/index.js +102 -101
  35. package/dist/index.js.map +1 -1
  36. package/dist/index.mjs +102 -101
  37. package/dist/index.mjs.map +1 -1
  38. package/dist/server/index.js +70 -43
  39. package/dist/server/index.js.map +1 -1
  40. package/dist/server/index.mjs +70 -43
  41. package/dist/server/index.mjs.map +1 -1
  42. package/dist/verify/index.d.mts +1 -1
  43. package/dist/verify/index.d.ts +1 -1
  44. package/dist/verify/index.js +16 -49
  45. package/dist/verify/index.js.map +1 -1
  46. package/dist/verify/index.mjs +16 -49
  47. package/dist/verify/index.mjs.map +1 -1
  48. package/dist/wallet/index.d.mts +1 -1
  49. package/dist/wallet/index.d.ts +1 -1
  50. package/dist/wallet/index.js +16 -49
  51. package/dist/wallet/index.js.map +1 -1
  52. package/dist/wallet/index.mjs +16 -49
  53. package/dist/wallet/index.mjs.map +1 -1
  54. package/package.json +1 -1
@@ -1,4 +1,4 @@
1
- import { T as TokenSymbol } from '../index-B3v8IWjM.mjs';
1
+ import { T as TokenSymbol } from '../index-DgJPZMBG.mjs';
2
2
 
3
3
  /**
4
4
  * MoltsPay Client Types
@@ -78,8 +78,8 @@ interface PayOptions {
78
78
  token?: TokenSymbol;
79
79
  /** Auto-select token based on balance (default: false) */
80
80
  autoSelect?: boolean;
81
- /** Chain to pay on (base or polygon, default: base) */
82
- chain?: 'base' | 'polygon';
81
+ /** Chain to pay on (base, polygon, or base_sepolia, default: base) */
82
+ chain?: 'base' | 'polygon' | 'base_sepolia';
83
83
  }
84
84
  declare class MoltsPayClient {
85
85
  private configDir;
@@ -1,4 +1,4 @@
1
- import { T as TokenSymbol } from '../index-B3v8IWjM.js';
1
+ import { T as TokenSymbol } from '../index-DgJPZMBG.js';
2
2
 
3
3
  /**
4
4
  * MoltsPay Client Types
@@ -78,8 +78,8 @@ interface PayOptions {
78
78
  token?: TokenSymbol;
79
79
  /** Auto-select token based on balance (default: false) */
80
80
  autoSelect?: boolean;
81
- /** Chain to pay on (base or polygon, default: base) */
82
- chain?: 'base' | 'polygon';
81
+ /** Chain to pay on (base, polygon, or base_sepolia, default: base) */
82
+ chain?: 'base' | 'polygon' | 'base_sepolia';
83
83
  }
84
84
  declare class MoltsPayClient {
85
85
  private configDir;
@@ -39,12 +39,15 @@ var CHAINS = {
39
39
  USDC: {
40
40
  address: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",
41
41
  decimals: 6,
42
- symbol: "USDC"
42
+ symbol: "USDC",
43
+ eip712Name: "USD Coin"
44
+ // EIP-712 domain name
43
45
  },
44
46
  USDT: {
45
47
  address: "0xfde4C96c8593536E31F229EA8f37b2ADa2699bb2",
46
48
  decimals: 6,
47
- symbol: "USDT"
49
+ symbol: "USDT",
50
+ eip712Name: "Tether USD"
48
51
  }
49
52
  },
50
53
  usdc: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",
@@ -61,12 +64,15 @@ var CHAINS = {
61
64
  USDC: {
62
65
  address: "0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",
63
66
  decimals: 6,
64
- symbol: "USDC"
67
+ symbol: "USDC",
68
+ eip712Name: "USD Coin"
65
69
  },
66
70
  USDT: {
67
71
  address: "0xc2132D05D31c914a87C6611C10748AEb04B58e8F",
68
72
  decimals: 6,
69
- symbol: "USDT"
73
+ symbol: "USDT",
74
+ eip712Name: "(PoS) Tether USD"
75
+ // Polygon uses this name
70
76
  }
71
77
  },
72
78
  usdc: "0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",
@@ -74,27 +80,6 @@ var CHAINS = {
74
80
  explorerTx: "https://polygonscan.com/tx/",
75
81
  avgBlockTime: 2
76
82
  },
77
- ethereum: {
78
- name: "Ethereum",
79
- chainId: 1,
80
- rpc: "https://eth.llamarpc.com",
81
- tokens: {
82
- USDC: {
83
- address: "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
84
- decimals: 6,
85
- symbol: "USDC"
86
- },
87
- USDT: {
88
- address: "0xdAC17F958D2ee523a2206206994597C13D831ec7",
89
- decimals: 6,
90
- symbol: "USDT"
91
- }
92
- },
93
- usdc: "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
94
- explorer: "https://etherscan.io/address/",
95
- explorerTx: "https://etherscan.io/tx/",
96
- avgBlockTime: 12
97
- },
98
83
  // ============ Testnet ============
99
84
  base_sepolia: {
100
85
  name: "Base Sepolia",
@@ -104,41 +89,23 @@ var CHAINS = {
104
89
  USDC: {
105
90
  address: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
106
91
  decimals: 6,
107
- symbol: "USDC"
92
+ symbol: "USDC",
93
+ eip712Name: "USDC"
94
+ // Testnet USDC uses 'USDC' not 'USD Coin'
108
95
  },
109
96
  USDT: {
110
97
  address: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
111
98
  // Same as USDC on testnet (no official USDT)
112
99
  decimals: 6,
113
- symbol: "USDT"
100
+ symbol: "USDT",
101
+ eip712Name: "USDC"
102
+ // Uses same contract as USDC
114
103
  }
115
104
  },
116
105
  usdc: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
117
106
  explorer: "https://sepolia.basescan.org/address/",
118
107
  explorerTx: "https://sepolia.basescan.org/tx/",
119
108
  avgBlockTime: 2
120
- },
121
- sepolia: {
122
- name: "Sepolia",
123
- chainId: 11155111,
124
- rpc: "https://rpc.sepolia.org",
125
- tokens: {
126
- USDC: {
127
- address: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
128
- decimals: 6,
129
- symbol: "USDC"
130
- },
131
- USDT: {
132
- address: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
133
- // Same as USDC on testnet
134
- decimals: 6,
135
- symbol: "USDT"
136
- }
137
- },
138
- usdc: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
139
- explorer: "https://sepolia.etherscan.io/address/",
140
- explorerTx: "https://sepolia.etherscan.io/tx/",
141
- avgBlockTime: 12
142
109
  }
143
110
  };
144
111
  function getChain(name) {
@@ -301,7 +268,7 @@ Server accepts: ${serverChains.join(", ")}`
301
268
  } else {
302
269
  throw new Error(
303
270
  `Server accepts: ${serverChains.join(", ")}
304
- Please specify: --chain base or --chain polygon`
271
+ Please specify: --chain base, --chain polygon, or --chain base_sepolia`
305
272
  );
306
273
  }
307
274
  }
@@ -343,13 +310,19 @@ Please specify: --chain base or --chain polygon`
343
310
  if (!payTo) {
344
311
  throw new Error("Missing payTo address in payment requirements");
345
312
  }
346
- const authorization = await this.signEIP3009(payTo, amount, chain, token);
313
+ const domainOverride = req.extra && typeof req.extra === "object" && req.extra.name ? { name: req.extra.name, version: req.extra.version || "2" } : void 0;
314
+ const authorization = await this.signEIP3009(payTo, amount, chain, token, domainOverride);
347
315
  const tokenConfig = chain.tokens[token];
348
- const tokenName = token === "USDC" ? "USD Coin" : "Tether USD";
316
+ const extra = req.extra && typeof req.extra === "object" ? req.extra : {
317
+ name: tokenConfig.eip712Name || "USD Coin",
318
+ version: "2"
319
+ };
349
320
  const payload = {
350
321
  x402Version: X402_VERSION,
322
+ scheme: "exact",
323
+ network,
351
324
  payload: authorization,
352
- // v2 requires 'accepted' field with the requirements being fulfilled
325
+ // { authorization: {...}, signature: "0x..." }
353
326
  accepted: {
354
327
  scheme: "exact",
355
328
  network,
@@ -357,7 +330,7 @@ Please specify: --chain base or --chain polygon`
357
330
  amount: amountRaw,
358
331
  payTo,
359
332
  maxTimeoutSeconds: req.maxTimeoutSeconds || 300,
360
- extra: req.extra || { name: tokenName, version: "2" }
333
+ extra
361
334
  }
362
335
  };
363
336
  const paymentHeader = Buffer.from(JSON.stringify(payload)).toString("base64");
@@ -387,7 +360,7 @@ Please specify: --chain base or --chain polygon`
387
360
  * This only signs - no on-chain transaction, no gas needed.
388
361
  * Supports both USDC and USDT.
389
362
  */
390
- async signEIP3009(to, amount, chain, token = "USDC") {
363
+ async signEIP3009(to, amount, chain, token = "USDC", domainOverride) {
391
364
  const validAfter = 0;
392
365
  const validBefore = Math.floor(Date.now() / 1e3) + 3600;
393
366
  const nonce = import_ethers.ethers.hexlify(import_ethers.ethers.randomBytes(32));
@@ -401,10 +374,11 @@ Please specify: --chain base or --chain polygon`
401
374
  validBefore: validBefore.toString(),
402
375
  nonce
403
376
  };
404
- const tokenName = token === "USDC" ? "USD Coin" : "Tether USD";
377
+ const tokenName = domainOverride?.name || tokenConfig.eip712Name || (token === "USDC" ? "USD Coin" : "Tether USD");
378
+ const tokenVersion = domainOverride?.version || "2";
405
379
  const domain = {
406
380
  name: tokenName,
407
- version: "2",
381
+ version: tokenVersion,
408
382
  chainId: chain.chainId,
409
383
  verifyingContract: tokenConfig.address
410
384
  };
@@ -573,7 +547,7 @@ Please specify: --chain base or --chain polygon`
573
547
  if (!this.wallet) {
574
548
  throw new Error("Client not initialized");
575
549
  }
576
- const supportedChains = ["base", "polygon"];
550
+ const supportedChains = ["base", "polygon", "base_sepolia"];
577
551
  const tokenAbi = ["function balanceOf(address) view returns (uint256)"];
578
552
  const results = {};
579
553
  await Promise.all(
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/client/index.ts","../../src/chains/index.ts"],"sourcesContent":["/**\n * MoltsPay Client - Pay for AI Agent services\n * \n * Uses x402 protocol for gasless, pay-for-success payments.\n * \n * Usage:\n * const client = new MoltsPayClient(); // Loads from ~/.moltspay/\n * const services = await client.getServices('http://provider:3000');\n * const result = await client.pay('http://provider:3000', 'text-to-video', { prompt: '...' });\n */\n\nimport { existsSync, readFileSync, writeFileSync, mkdirSync, statSync, chmodSync } from 'fs';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { Wallet, ethers } from 'ethers';\nimport { getChain, type ChainName, type TokenSymbol } from '../chains/index.js';\nimport {\n ClientConfig,\n WalletData,\n ServicesResponse,\n MoltsPayClientOptions,\n} from './types.js';\n\nexport * from './types.js';\n\nexport interface PayOptions {\n /** Token to pay with (default: USDC, or auto-select based on balance) */\n token?: TokenSymbol;\n /** Auto-select token based on balance (default: false) */\n autoSelect?: boolean;\n /** Chain to pay on (base or polygon, default: base) */\n chain?: 'base' | 'polygon';\n}\n\n// x402 constants\nconst X402_VERSION = 2;\nconst PAYMENT_REQUIRED_HEADER = 'x-payment-required';\nconst PAYMENT_HEADER = 'x-payment';\n\ninterface X402PaymentRequirements {\n scheme: string;\n network: string;\n // v2 fields\n amount?: string;\n asset?: string;\n payTo?: string;\n maxTimeoutSeconds?: number;\n extra?: Record<string, unknown>;\n // v1 fields (legacy)\n maxAmountRequired?: string;\n resource?: string;\n description?: string;\n}\n\ninterface EIP3009Authorization {\n from: string;\n to: string;\n value: string;\n validAfter: string;\n validBefore: string;\n nonce: string;\n}\n\nconst DEFAULT_CONFIG: ClientConfig = {\n chain: 'base',\n limits: {\n maxPerTx: 100,\n maxPerDay: 1000,\n },\n};\n\nexport class MoltsPayClient {\n private configDir: string;\n private config: ClientConfig;\n private walletData: WalletData | null = null;\n private wallet: Wallet | null = null;\n private todaySpending: number = 0;\n private lastSpendingReset: number = 0;\n\n constructor(options: MoltsPayClientOptions = {}) {\n this.configDir = options.configDir || join(homedir(), '.moltspay');\n this.config = this.loadConfig();\n this.walletData = this.loadWallet();\n this.loadSpending(); // Load persisted spending data\n \n if (this.walletData) {\n this.wallet = new Wallet(this.walletData.privateKey);\n }\n }\n\n /**\n * Check if client is initialized (has wallet)\n */\n get isInitialized(): boolean {\n return this.wallet !== null;\n }\n\n /**\n * Get wallet address\n */\n get address(): string | null {\n return this.wallet?.address || null;\n }\n\n /**\n * Get current config\n */\n getConfig(): ClientConfig {\n return { ...this.config };\n }\n\n /**\n * Update config\n */\n updateConfig(updates: Partial<ClientConfig['limits']>): void {\n if (updates.maxPerTx !== undefined) {\n this.config.limits.maxPerTx = updates.maxPerTx;\n }\n if (updates.maxPerDay !== undefined) {\n this.config.limits.maxPerDay = updates.maxPerDay;\n }\n this.saveConfig();\n }\n\n /**\n * Get services from a provider\n */\n async getServices(serverUrl: string): Promise<ServicesResponse> {\n // Normalize URL - don't append /services if already present\n const normalizedUrl = serverUrl.replace(/\\/(services|api\\/services|registry\\/services)\\/?$/, '');\n \n // Try /services first (standard provider endpoint)\n const endpoints = ['/services', '/api/services', '/registry/services'];\n \n for (const endpoint of endpoints) {\n try {\n const res = await fetch(`${normalizedUrl}${endpoint}`);\n if (!res.ok) continue;\n \n const contentType = res.headers.get('content-type') || '';\n if (!contentType.includes('application/json')) continue;\n \n return await res.json() as ServicesResponse;\n } catch {\n continue;\n }\n }\n \n throw new Error(`Failed to get services: no valid endpoint found at ${normalizedUrl}`);\n }\n\n /**\n * Pay for a service and get the result (x402 protocol)\n * \n * This is GASLESS for the client - server pays gas to claim payment.\n * This is PAY-FOR-SUCCESS - payment only claimed if service succeeds.\n * \n * @param serverUrl - Server URL\n * @param service - Service ID\n * @param params - Service parameters\n * @param options - Payment options (token selection)\n */\n async pay(\n serverUrl: string,\n service: string,\n params: Record<string, any>,\n options: PayOptions = {}\n ): Promise<Record<string, any>> {\n if (!this.wallet || !this.walletData) {\n throw new Error('Client not initialized. Run: npx moltspay init');\n }\n\n // Step 1: Make initial request without payment\n console.log(`[MoltsPay] Requesting service: ${service}`);\n const requestBody: any = { service, params };\n if (options.chain) {\n requestBody.chain = options.chain;\n }\n const initialRes = await fetch(`${serverUrl}/execute`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(requestBody),\n });\n\n // If not 402, check for success or error\n if (initialRes.status !== 402) {\n const data = await initialRes.json() as any;\n if (initialRes.ok && data.result) {\n return data.result;\n }\n throw new Error(data.error || 'Unexpected response');\n }\n\n // Step 2: Parse payment requirements from 402 response\n const paymentRequiredHeader = initialRes.headers.get(PAYMENT_REQUIRED_HEADER);\n if (!paymentRequiredHeader) {\n throw new Error('Missing x-payment-required header');\n }\n\n let requirements: X402PaymentRequirements[];\n try {\n const decoded = Buffer.from(paymentRequiredHeader, 'base64').toString('utf-8');\n const parsed = JSON.parse(decoded);\n \n // Handle both v1 (array) and v2 (object with accepts) formats\n if (Array.isArray(parsed)) {\n // v1 format: direct array of requirements\n requirements = parsed;\n } else if (parsed.accepts && Array.isArray(parsed.accepts)) {\n // v2 format: { x402Version: 2, accepts: [...] }\n requirements = parsed.accepts;\n } else {\n // Single requirement object\n requirements = [parsed];\n }\n } catch {\n throw new Error('Invalid x-payment-required header');\n }\n\n // Helper to convert network ID to chain name\n const networkToChainName = (network: string): string | null => {\n const match = network.match(/^eip155:(\\d+)$/);\n if (!match) return null;\n const chainId = parseInt(match[1]);\n if (chainId === 8453) return 'base';\n if (chainId === 137) return 'polygon';\n if (chainId === 84532) return 'base_sepolia';\n return null;\n };\n\n // Get server's accepted chains\n const serverChains = requirements\n .map(r => networkToChainName(r.network))\n .filter((c): c is string => c !== null);\n\n // Determine which chain to use\n let chainName: ChainName;\n const userSpecifiedChain = options.chain;\n\n if (userSpecifiedChain) {\n // User specified --chain, validate it's accepted by server\n if (!serverChains.includes(userSpecifiedChain)) {\n throw new Error(\n `Server doesn't accept '${userSpecifiedChain}'.\\n` +\n `Server accepts: ${serverChains.join(', ')}`\n );\n }\n chainName = userSpecifiedChain as ChainName;\n } else {\n // No --chain provided\n if (serverChains.length === 1 && serverChains[0] === 'base') {\n // Only default to base if server ONLY accepts base\n chainName = 'base';\n } else {\n throw new Error(\n `Server accepts: ${serverChains.join(', ')}\\n` +\n `Please specify: --chain base or --chain polygon`\n );\n }\n }\n\n const chain = getChain(chainName);\n const network = `eip155:${chain.chainId}`;\n const req = requirements.find(r => r.scheme === 'exact' && r.network === network);\n\n if (!req) {\n throw new Error(`Failed to find payment requirement for ${chainName}`);\n }\n\n // Step 3: Check limits\n // v2 uses 'amount', v1 uses 'maxAmountRequired'\n const amountRaw = req.amount || req.maxAmountRequired;\n if (!amountRaw) {\n throw new Error('Missing amount in payment requirements');\n }\n const amount = Number(amountRaw) / 1e6;\n this.checkLimits(amount);\n\n // Determine which token to use\n let token: TokenSymbol = options.token || 'USDC';\n \n // Auto-select token based on balance if requested\n if (options.autoSelect) {\n const balances = await this.getBalance();\n if (balances.usdc >= amount) {\n token = 'USDC';\n } else if (balances.usdt >= amount) {\n token = 'USDT';\n } else {\n throw new Error(`Insufficient balance: need $${amount}, have ${balances.usdc} USDC / ${balances.usdt} USDT`);\n }\n }\n\n // USDT does not support gasless transfers (no EIP-2612 permit)\n // It requires on-chain approve + transfer, meaning the user pays gas\n if (token === 'USDT') {\n const balances = await this.getBalance();\n if (balances.native < 0.0001) {\n throw new Error(\n `USDT requires ETH for gas (~$0.01 on Base). ` +\n `Your ETH balance: ${balances.native.toFixed(6)} ETH. ` +\n `Please add a small amount of ETH to your wallet, or use USDC (gasless).`\n );\n }\n console.log(`[MoltsPay] ⚠️ USDT requires gas (~$0.01). Proceeding with payment...`);\n } else {\n console.log(`[MoltsPay] Signing payment: $${amount} ${token} (gasless)`);\n }\n\n // Step 4: Sign EIP-3009 authorization (GASLESS - just signing)\n // payTo is the recipient address (v2 format)\n const payTo = req.payTo || req.resource; // fallback for v1 compatibility\n if (!payTo) {\n throw new Error('Missing payTo address in payment requirements');\n }\n const authorization = await this.signEIP3009(payTo, amount, chain, token);\n\n // Get token-specific info\n const tokenConfig = chain.tokens[token];\n const tokenName = token === 'USDC' ? 'USD Coin' : 'Tether USD';\n\n // Step 5: Create x402 payment payload (v2 format requires 'accepted')\n const payload = {\n x402Version: X402_VERSION,\n payload: authorization,\n // v2 requires 'accepted' field with the requirements being fulfilled\n accepted: {\n scheme: 'exact',\n network,\n asset: tokenConfig.address,\n amount: amountRaw,\n payTo,\n maxTimeoutSeconds: req.maxTimeoutSeconds || 300,\n extra: req.extra || { name: tokenName, version: '2' },\n },\n };\n const paymentHeader = Buffer.from(JSON.stringify(payload)).toString('base64');\n\n // Step 6: Retry with payment header\n console.log(`[MoltsPay] Sending request with payment...`);\n const paidRequestBody: any = { service, params };\n if (options.chain) {\n paidRequestBody.chain = options.chain;\n }\n const paidRes = await fetch(`${serverUrl}/execute`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n [PAYMENT_HEADER]: paymentHeader,\n },\n body: JSON.stringify(paidRequestBody),\n });\n\n const result = await paidRes.json() as any;\n\n if (!paidRes.ok) {\n throw new Error(result.error || 'Service execution failed');\n }\n\n // Update spending tracking\n this.recordSpending(amount);\n\n console.log(`[MoltsPay] Success! Payment: ${result.payment?.status || 'claimed'}`);\n \n return result.result;\n }\n\n /**\n * Sign EIP-3009 transferWithAuthorization (GASLESS)\n * This only signs - no on-chain transaction, no gas needed.\n * Supports both USDC and USDT.\n */\n private async signEIP3009(\n to: string,\n amount: number,\n chain: { chainId: number; tokens: Record<TokenSymbol, { address: string; decimals: number }> },\n token: TokenSymbol = 'USDC'\n ): Promise<{ authorization: EIP3009Authorization; signature: string }> {\n const validAfter = 0;\n const validBefore = Math.floor(Date.now() / 1000) + 3600; // 1 hour\n const nonce = ethers.hexlify(ethers.randomBytes(32));\n \n const tokenConfig = chain.tokens[token];\n const value = BigInt(Math.floor(amount * (10 ** tokenConfig.decimals))).toString();\n\n const authorization: EIP3009Authorization = {\n from: this.wallet!.address,\n to,\n value,\n validAfter: validAfter.toString(),\n validBefore: validBefore.toString(),\n nonce,\n };\n\n // EIP-712 domain - token specific\n const tokenName = token === 'USDC' ? 'USD Coin' : 'Tether USD';\n const domain = {\n name: tokenName,\n version: '2',\n chainId: chain.chainId,\n verifyingContract: tokenConfig.address,\n };\n\n // EIP-3009 types\n const types = {\n TransferWithAuthorization: [\n { name: 'from', type: 'address' },\n { name: 'to', type: 'address' },\n { name: 'value', type: 'uint256' },\n { name: 'validAfter', type: 'uint256' },\n { name: 'validBefore', type: 'uint256' },\n { name: 'nonce', type: 'bytes32' },\n ],\n };\n\n const signature = await this.wallet!.signTypedData(domain, types, authorization);\n\n return { authorization, signature };\n }\n\n /**\n * Check spending limits\n */\n private checkLimits(amount: number): void {\n // Check per-tx limit\n if (amount > this.config.limits.maxPerTx) {\n throw new Error(\n `Amount $${amount} exceeds max per transaction ($${this.config.limits.maxPerTx})`\n );\n }\n\n // Reset daily spending if new day\n const today = new Date().setHours(0, 0, 0, 0);\n if (today > this.lastSpendingReset) {\n this.todaySpending = 0;\n this.lastSpendingReset = today;\n this.saveSpending(); // Persist reset\n }\n\n // Check daily limit\n if (this.todaySpending + amount > this.config.limits.maxPerDay) {\n throw new Error(\n `Would exceed daily limit ($${this.todaySpending} + $${amount} > $${this.config.limits.maxPerDay})`\n );\n }\n }\n\n /**\n * Record spending and persist to disk\n */\n private recordSpending(amount: number): void {\n this.todaySpending += amount;\n this.saveSpending();\n }\n\n // --- Config & Wallet Management ---\n\n private loadConfig(): ClientConfig {\n const configPath = join(this.configDir, 'config.json');\n if (existsSync(configPath)) {\n const content = readFileSync(configPath, 'utf-8');\n return { ...DEFAULT_CONFIG, ...JSON.parse(content) };\n }\n return { ...DEFAULT_CONFIG };\n }\n\n private saveConfig(): void {\n mkdirSync(this.configDir, { recursive: true });\n const configPath = join(this.configDir, 'config.json');\n writeFileSync(configPath, JSON.stringify(this.config, null, 2));\n }\n\n /**\n * Load spending data from disk\n */\n private loadSpending(): void {\n const spendingPath = join(this.configDir, 'spending.json');\n if (existsSync(spendingPath)) {\n try {\n const data = JSON.parse(readFileSync(spendingPath, 'utf-8'));\n const today = new Date().setHours(0, 0, 0, 0);\n \n // Only load if it's from today\n if (data.date && data.date === today) {\n this.todaySpending = data.amount || 0;\n this.lastSpendingReset = data.date;\n } else {\n // Data is from a previous day, reset\n this.todaySpending = 0;\n this.lastSpendingReset = today;\n }\n } catch {\n // Ignore parse errors, start fresh\n this.todaySpending = 0;\n this.lastSpendingReset = new Date().setHours(0, 0, 0, 0);\n }\n }\n }\n\n /**\n * Save spending data to disk\n */\n private saveSpending(): void {\n mkdirSync(this.configDir, { recursive: true });\n const spendingPath = join(this.configDir, 'spending.json');\n const data = {\n date: this.lastSpendingReset || new Date().setHours(0, 0, 0, 0),\n amount: this.todaySpending,\n updatedAt: Date.now(),\n };\n writeFileSync(spendingPath, JSON.stringify(data, null, 2));\n }\n\n private loadWallet(): WalletData | null {\n const walletPath = join(this.configDir, 'wallet.json');\n if (existsSync(walletPath)) {\n // Security check: warn and fix if permissions are too open\n try {\n const stats = statSync(walletPath);\n const mode = stats.mode & 0o777;\n if (mode !== 0o600) {\n console.warn(`[MoltsPay] WARNING: wallet.json has insecure permissions (${mode.toString(8)})`);\n console.warn(`[MoltsPay] Fixing permissions to 0600...`);\n chmodSync(walletPath, 0o600);\n }\n } catch (err) {\n // Ignore permission check errors on Windows\n }\n \n const content = readFileSync(walletPath, 'utf-8');\n return JSON.parse(content);\n }\n return null;\n }\n\n /**\n * Initialize a new wallet (called by CLI)\n */\n static init(\n configDir: string,\n options: { chain: string; maxPerTx: number; maxPerDay: number }\n ): { address: string; configDir: string } {\n mkdirSync(configDir, { recursive: true });\n\n // Create wallet\n const wallet = Wallet.createRandom();\n const walletData: WalletData = {\n address: wallet.address,\n privateKey: wallet.privateKey,\n createdAt: Date.now(),\n };\n\n // Save wallet with secure permissions (0o600 = owner read/write only)\n const walletPath = join(configDir, 'wallet.json');\n writeFileSync(walletPath, JSON.stringify(walletData, null, 2), { mode: 0o600 });\n\n // Save config\n const config: ClientConfig = {\n chain: options.chain,\n limits: {\n maxPerTx: options.maxPerTx,\n maxPerDay: options.maxPerDay,\n },\n };\n const configPath = join(configDir, 'config.json');\n writeFileSync(configPath, JSON.stringify(config, null, 2));\n\n return { address: wallet.address, configDir };\n }\n\n /**\n * Get wallet balance (USDC, USDT, and native token) on default chain\n */\n async getBalance(): Promise<{ usdc: number; usdt: number; native: number }> {\n if (!this.wallet) {\n throw new Error('Client not initialized');\n }\n\n let chain;\n try {\n chain = getChain(this.config.chain as ChainName);\n } catch {\n throw new Error(`Unknown chain: ${this.config.chain}`);\n }\n\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n const tokenAbi = ['function balanceOf(address) view returns (uint256)'];\n\n // Get all balances in parallel\n const [nativeBalance, usdcBalance, usdtBalance] = await Promise.all([\n provider.getBalance(this.wallet.address),\n new ethers.Contract(chain.tokens.USDC.address, tokenAbi, provider).balanceOf(this.wallet.address),\n new ethers.Contract(chain.tokens.USDT.address, tokenAbi, provider).balanceOf(this.wallet.address),\n ]);\n\n return {\n usdc: parseFloat(ethers.formatUnits(usdcBalance, chain.tokens.USDC.decimals)),\n usdt: parseFloat(ethers.formatUnits(usdtBalance, chain.tokens.USDT.decimals)),\n native: parseFloat(ethers.formatEther(nativeBalance)),\n };\n }\n\n /**\n * Get wallet balances on all supported chains (Base + Polygon)\n */\n async getAllBalances(): Promise<Record<string, { usdc: number; usdt: number; native: number }>> {\n if (!this.wallet) {\n throw new Error('Client not initialized');\n }\n\n const supportedChains: ChainName[] = ['base', 'polygon'];\n const tokenAbi = ['function balanceOf(address) view returns (uint256)'];\n const results: Record<string, { usdc: number; usdt: number; native: number }> = {};\n\n // Query all chains in parallel\n await Promise.all(\n supportedChains.map(async (chainName) => {\n try {\n const chain = getChain(chainName);\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n\n const [nativeBalance, usdcBalance, usdtBalance] = await Promise.all([\n provider.getBalance(this.wallet!.address),\n new ethers.Contract(chain.tokens.USDC.address, tokenAbi, provider).balanceOf(this.wallet!.address),\n new ethers.Contract(chain.tokens.USDT.address, tokenAbi, provider).balanceOf(this.wallet!.address),\n ]);\n\n results[chainName] = {\n usdc: parseFloat(ethers.formatUnits(usdcBalance, chain.tokens.USDC.decimals)),\n usdt: parseFloat(ethers.formatUnits(usdtBalance, chain.tokens.USDT.decimals)),\n native: parseFloat(ethers.formatEther(nativeBalance)),\n };\n } catch (err) {\n // If chain query fails, show zeros\n results[chainName] = { usdc: 0, usdt: 0, native: 0 };\n }\n })\n );\n\n return results;\n }\n}\n","/**\n * Blockchain Configuration\n */\n\nimport type { ChainConfig, ChainName, TokenSymbol } from '../types/index.js';\n\nexport const CHAINS: Record<ChainName, ChainConfig> = {\n // ============ Mainnet ============\n base: {\n name: 'Base',\n chainId: 8453,\n rpc: 'https://mainnet.base.org',\n tokens: {\n USDC: {\n address: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913',\n decimals: 6,\n symbol: 'USDC',\n },\n USDT: {\n address: '0xfde4C96c8593536E31F229EA8f37b2ADa2699bb2',\n decimals: 6,\n symbol: 'USDT',\n },\n },\n usdc: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913', // deprecated, for backward compat\n explorer: 'https://basescan.org/address/',\n explorerTx: 'https://basescan.org/tx/',\n avgBlockTime: 2,\n },\n polygon: {\n name: 'Polygon',\n chainId: 137,\n rpc: 'https://polygon-bor-rpc.publicnode.com',\n tokens: {\n USDC: {\n address: '0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359',\n decimals: 6,\n symbol: 'USDC',\n },\n USDT: {\n address: '0xc2132D05D31c914a87C6611C10748AEb04B58e8F',\n decimals: 6,\n symbol: 'USDT',\n },\n },\n usdc: '0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359',\n explorer: 'https://polygonscan.com/address/',\n explorerTx: 'https://polygonscan.com/tx/',\n avgBlockTime: 2,\n },\n ethereum: {\n name: 'Ethereum',\n chainId: 1,\n rpc: 'https://eth.llamarpc.com',\n tokens: {\n USDC: {\n address: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48',\n decimals: 6,\n symbol: 'USDC',\n },\n USDT: {\n address: '0xdAC17F958D2ee523a2206206994597C13D831ec7',\n decimals: 6,\n symbol: 'USDT',\n },\n },\n usdc: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48',\n explorer: 'https://etherscan.io/address/',\n explorerTx: 'https://etherscan.io/tx/',\n avgBlockTime: 12,\n },\n\n // ============ Testnet ============\n base_sepolia: {\n name: 'Base Sepolia',\n chainId: 84532,\n rpc: 'https://sepolia.base.org',\n tokens: {\n USDC: {\n address: '0x036CbD53842c5426634e7929541eC2318f3dCF7e',\n decimals: 6,\n symbol: 'USDC',\n },\n USDT: {\n address: '0x036CbD53842c5426634e7929541eC2318f3dCF7e', // Same as USDC on testnet (no official USDT)\n decimals: 6,\n symbol: 'USDT',\n },\n },\n usdc: '0x036CbD53842c5426634e7929541eC2318f3dCF7e',\n explorer: 'https://sepolia.basescan.org/address/',\n explorerTx: 'https://sepolia.basescan.org/tx/',\n avgBlockTime: 2,\n },\n sepolia: {\n name: 'Sepolia',\n chainId: 11155111,\n rpc: 'https://rpc.sepolia.org',\n tokens: {\n USDC: {\n address: '0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238',\n decimals: 6,\n symbol: 'USDC',\n },\n USDT: {\n address: '0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238', // Same as USDC on testnet\n decimals: 6,\n symbol: 'USDT',\n },\n },\n usdc: '0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238',\n explorer: 'https://sepolia.etherscan.io/address/',\n explorerTx: 'https://sepolia.etherscan.io/tx/',\n avgBlockTime: 12,\n },\n};\n\n/**\n * Get token address for a chain\n */\nexport function getTokenAddress(chainName: ChainName, token: TokenSymbol): string {\n const chain = CHAINS[chainName];\n if (!chain) {\n throw new Error(`Unsupported chain: ${chainName}`);\n }\n const tokenConfig = chain.tokens[token];\n if (!tokenConfig) {\n throw new Error(`Token ${token} not supported on ${chainName}`);\n }\n return tokenConfig.address;\n}\n\n/**\n * Get token config for a chain\n */\nexport function getTokenConfig(chainName: ChainName, token: TokenSymbol) {\n const chain = CHAINS[chainName];\n if (!chain) {\n throw new Error(`Unsupported chain: ${chainName}`);\n }\n return chain.tokens[token];\n}\n\n/**\n * Get chain configuration\n */\nexport function getChain(name: ChainName): ChainConfig {\n const config = CHAINS[name];\n if (!config) {\n throw new Error(`Unsupported chain: ${name}. Supported: ${Object.keys(CHAINS).join(', ')}`);\n }\n return config;\n}\n\n/**\n * List all supported chains\n */\nexport function listChains(): ChainName[] {\n return Object.keys(CHAINS) as ChainName[];\n}\n\n/**\n * Get chain config by chainId\n */\nexport function getChainById(chainId: number): ChainConfig | undefined {\n return Object.values(CHAINS).find(c => c.chainId === chainId);\n}\n\n/**\n * ERC20 ABI (minimal, only required methods)\n */\nexport const ERC20_ABI = [\n 'function balanceOf(address owner) view returns (uint256)',\n 'function transfer(address to, uint256 amount) returns (bool)',\n 'function approve(address spender, uint256 amount) returns (bool)',\n 'function allowance(address owner, address spender) view returns (uint256)',\n 'function decimals() view returns (uint8)',\n 'function symbol() view returns (string)',\n 'function name() view returns (string)',\n 'function nonces(address owner) view returns (uint256)',\n 'function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s)',\n 'event Transfer(address indexed from, address indexed to, uint256 value)',\n 'event Approval(address indexed owner, address indexed spender, uint256 value)',\n];\n\nexport type { ChainConfig, ChainName, TokenSymbol };\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAWA,gBAAwF;AACxF,gBAAwB;AACxB,kBAAqB;AACrB,oBAA+B;;;ACRxB,IAAM,SAAyC;AAAA;AAAA,EAEpD,MAAM;AAAA,IACJ,MAAM;AAAA,IACN,SAAS;AAAA,IACT,KAAK;AAAA,IACL,QAAQ;AAAA,MACN,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,MACA,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,IACA,MAAM;AAAA;AAAA,IACN,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,cAAc;AAAA,EAChB;AAAA,EACA,SAAS;AAAA,IACP,MAAM;AAAA,IACN,SAAS;AAAA,IACT,KAAK;AAAA,IACL,QAAQ;AAAA,MACN,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,MACA,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,IACA,MAAM;AAAA,IACN,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,cAAc;AAAA,EAChB;AAAA,EACA,UAAU;AAAA,IACR,MAAM;AAAA,IACN,SAAS;AAAA,IACT,KAAK;AAAA,IACL,QAAQ;AAAA,MACN,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,MACA,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,IACA,MAAM;AAAA,IACN,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,cAAc;AAAA,EAChB;AAAA;AAAA,EAGA,cAAc;AAAA,IACZ,MAAM;AAAA,IACN,SAAS;AAAA,IACT,KAAK;AAAA,IACL,QAAQ;AAAA,MACN,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,MACA,MAAM;AAAA,QACJ,SAAS;AAAA;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,IACA,MAAM;AAAA,IACN,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,cAAc;AAAA,EAChB;AAAA,EACA,SAAS;AAAA,IACP,MAAM;AAAA,IACN,SAAS;AAAA,IACT,KAAK;AAAA,IACL,QAAQ;AAAA,MACN,MAAM;AAAA,QACJ,SAAS;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,MACA,MAAM;AAAA,QACJ,SAAS;AAAA;AAAA,QACT,UAAU;AAAA,QACV,QAAQ;AAAA,MACV;AAAA,IACF;AAAA,IACA,MAAM;AAAA,IACN,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,cAAc;AAAA,EAChB;AACF;AA+BO,SAAS,SAAS,MAA8B;AACrD,QAAM,SAAS,OAAO,IAAI;AAC1B,MAAI,CAAC,QAAQ;AACX,UAAM,IAAI,MAAM,sBAAsB,IAAI,gBAAgB,OAAO,KAAK,MAAM,EAAE,KAAK,IAAI,CAAC,EAAE;AAAA,EAC5F;AACA,SAAO;AACT;;;ADrHA,IAAM,eAAe;AACrB,IAAM,0BAA0B;AAChC,IAAM,iBAAiB;AA0BvB,IAAM,iBAA+B;AAAA,EACnC,OAAO;AAAA,EACP,QAAQ;AAAA,IACN,UAAU;AAAA,IACV,WAAW;AAAA,EACb;AACF;AAEO,IAAM,iBAAN,MAAqB;AAAA,EAClB;AAAA,EACA;AAAA,EACA,aAAgC;AAAA,EAChC,SAAwB;AAAA,EACxB,gBAAwB;AAAA,EACxB,oBAA4B;AAAA,EAEpC,YAAY,UAAiC,CAAC,GAAG;AAC/C,SAAK,YAAY,QAAQ,iBAAa,sBAAK,mBAAQ,GAAG,WAAW;AACjE,SAAK,SAAS,KAAK,WAAW;AAC9B,SAAK,aAAa,KAAK,WAAW;AAClC,SAAK,aAAa;AAElB,QAAI,KAAK,YAAY;AACnB,WAAK,SAAS,IAAI,qBAAO,KAAK,WAAW,UAAU;AAAA,IACrD;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,gBAAyB;AAC3B,WAAO,KAAK,WAAW;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,UAAyB;AAC3B,WAAO,KAAK,QAAQ,WAAW;AAAA,EACjC;AAAA;AAAA;AAAA;AAAA,EAKA,YAA0B;AACxB,WAAO,EAAE,GAAG,KAAK,OAAO;AAAA,EAC1B;AAAA;AAAA;AAAA;AAAA,EAKA,aAAa,SAAgD;AAC3D,QAAI,QAAQ,aAAa,QAAW;AAClC,WAAK,OAAO,OAAO,WAAW,QAAQ;AAAA,IACxC;AACA,QAAI,QAAQ,cAAc,QAAW;AACnC,WAAK,OAAO,OAAO,YAAY,QAAQ;AAAA,IACzC;AACA,SAAK,WAAW;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,YAAY,WAA8C;AAE9D,UAAM,gBAAgB,UAAU,QAAQ,qDAAqD,EAAE;AAG/F,UAAM,YAAY,CAAC,aAAa,iBAAiB,oBAAoB;AAErE,eAAW,YAAY,WAAW;AAChC,UAAI;AACF,cAAM,MAAM,MAAM,MAAM,GAAG,aAAa,GAAG,QAAQ,EAAE;AACrD,YAAI,CAAC,IAAI,GAAI;AAEb,cAAM,cAAc,IAAI,QAAQ,IAAI,cAAc,KAAK;AACvD,YAAI,CAAC,YAAY,SAAS,kBAAkB,EAAG;AAE/C,eAAO,MAAM,IAAI,KAAK;AAAA,MACxB,QAAQ;AACN;AAAA,MACF;AAAA,IACF;AAEA,UAAM,IAAI,MAAM,sDAAsD,aAAa,EAAE;AAAA,EACvF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAaA,MAAM,IACJ,WACA,SACA,QACA,UAAsB,CAAC,GACO;AAC9B,QAAI,CAAC,KAAK,UAAU,CAAC,KAAK,YAAY;AACpC,YAAM,IAAI,MAAM,gDAAgD;AAAA,IAClE;AAGA,YAAQ,IAAI,kCAAkC,OAAO,EAAE;AACvD,UAAM,cAAmB,EAAE,SAAS,OAAO;AAC3C,QAAI,QAAQ,OAAO;AACjB,kBAAY,QAAQ,QAAQ;AAAA,IAC9B;AACA,UAAM,aAAa,MAAM,MAAM,GAAG,SAAS,YAAY;AAAA,MACrD,QAAQ;AAAA,MACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,MAC9C,MAAM,KAAK,UAAU,WAAW;AAAA,IAClC,CAAC;AAGD,QAAI,WAAW,WAAW,KAAK;AAC7B,YAAM,OAAO,MAAM,WAAW,KAAK;AACnC,UAAI,WAAW,MAAM,KAAK,QAAQ;AAChC,eAAO,KAAK;AAAA,MACd;AACA,YAAM,IAAI,MAAM,KAAK,SAAS,qBAAqB;AAAA,IACrD;AAGA,UAAM,wBAAwB,WAAW,QAAQ,IAAI,uBAAuB;AAC5E,QAAI,CAAC,uBAAuB;AAC1B,YAAM,IAAI,MAAM,mCAAmC;AAAA,IACrD;AAEA,QAAI;AACJ,QAAI;AACF,YAAM,UAAU,OAAO,KAAK,uBAAuB,QAAQ,EAAE,SAAS,OAAO;AAC7E,YAAM,SAAS,KAAK,MAAM,OAAO;AAGjC,UAAI,MAAM,QAAQ,MAAM,GAAG;AAEzB,uBAAe;AAAA,MACjB,WAAW,OAAO,WAAW,MAAM,QAAQ,OAAO,OAAO,GAAG;AAE1D,uBAAe,OAAO;AAAA,MACxB,OAAO;AAEL,uBAAe,CAAC,MAAM;AAAA,MACxB;AAAA,IACF,QAAQ;AACN,YAAM,IAAI,MAAM,mCAAmC;AAAA,IACrD;AAGA,UAAM,qBAAqB,CAACA,aAAmC;AAC7D,YAAM,QAAQA,SAAQ,MAAM,gBAAgB;AAC5C,UAAI,CAAC,MAAO,QAAO;AACnB,YAAM,UAAU,SAAS,MAAM,CAAC,CAAC;AACjC,UAAI,YAAY,KAAM,QAAO;AAC7B,UAAI,YAAY,IAAK,QAAO;AAC5B,UAAI,YAAY,MAAO,QAAO;AAC9B,aAAO;AAAA,IACT;AAGA,UAAM,eAAe,aAClB,IAAI,OAAK,mBAAmB,EAAE,OAAO,CAAC,EACtC,OAAO,CAAC,MAAmB,MAAM,IAAI;AAGxC,QAAI;AACJ,UAAM,qBAAqB,QAAQ;AAEnC,QAAI,oBAAoB;AAEtB,UAAI,CAAC,aAAa,SAAS,kBAAkB,GAAG;AAC9C,cAAM,IAAI;AAAA,UACR,0BAA0B,kBAAkB;AAAA,kBACzB,aAAa,KAAK,IAAI,CAAC;AAAA,QAC5C;AAAA,MACF;AACA,kBAAY;AAAA,IACd,OAAO;AAEL,UAAI,aAAa,WAAW,KAAK,aAAa,CAAC,MAAM,QAAQ;AAE3D,oBAAY;AAAA,MACd,OAAO;AACL,cAAM,IAAI;AAAA,UACR,mBAAmB,aAAa,KAAK,IAAI,CAAC;AAAA;AAAA,QAE5C;AAAA,MACF;AAAA,IACF;AAEA,UAAM,QAAQ,SAAS,SAAS;AAChC,UAAM,UAAU,UAAU,MAAM,OAAO;AACvC,UAAM,MAAM,aAAa,KAAK,OAAK,EAAE,WAAW,WAAW,EAAE,YAAY,OAAO;AAEhF,QAAI,CAAC,KAAK;AACR,YAAM,IAAI,MAAM,0CAA0C,SAAS,EAAE;AAAA,IACvE;AAIA,UAAM,YAAY,IAAI,UAAU,IAAI;AACpC,QAAI,CAAC,WAAW;AACd,YAAM,IAAI,MAAM,wCAAwC;AAAA,IAC1D;AACA,UAAM,SAAS,OAAO,SAAS,IAAI;AACnC,SAAK,YAAY,MAAM;AAGvB,QAAI,QAAqB,QAAQ,SAAS;AAG1C,QAAI,QAAQ,YAAY;AACtB,YAAM,WAAW,MAAM,KAAK,WAAW;AACvC,UAAI,SAAS,QAAQ,QAAQ;AAC3B,gBAAQ;AAAA,MACV,WAAW,SAAS,QAAQ,QAAQ;AAClC,gBAAQ;AAAA,MACV,OAAO;AACL,cAAM,IAAI,MAAM,+BAA+B,MAAM,UAAU,SAAS,IAAI,WAAW,SAAS,IAAI,OAAO;AAAA,MAC7G;AAAA,IACF;AAIA,QAAI,UAAU,QAAQ;AACpB,YAAM,WAAW,MAAM,KAAK,WAAW;AACvC,UAAI,SAAS,SAAS,MAAQ;AAC5B,cAAM,IAAI;AAAA,UACR,iEACqB,SAAS,OAAO,QAAQ,CAAC,CAAC;AAAA,QAEjD;AAAA,MACF;AACA,cAAQ,IAAI,iFAAuE;AAAA,IACrF,OAAO;AACL,cAAQ,IAAI,gCAAgC,MAAM,IAAI,KAAK,YAAY;AAAA,IACzE;AAIA,UAAM,QAAQ,IAAI,SAAS,IAAI;AAC/B,QAAI,CAAC,OAAO;AACV,YAAM,IAAI,MAAM,+CAA+C;AAAA,IACjE;AACA,UAAM,gBAAgB,MAAM,KAAK,YAAY,OAAO,QAAQ,OAAO,KAAK;AAGxE,UAAM,cAAc,MAAM,OAAO,KAAK;AACtC,UAAM,YAAY,UAAU,SAAS,aAAa;AAGlD,UAAM,UAAU;AAAA,MACd,aAAa;AAAA,MACb,SAAS;AAAA;AAAA,MAET,UAAU;AAAA,QACR,QAAQ;AAAA,QACR;AAAA,QACA,OAAO,YAAY;AAAA,QACnB,QAAQ;AAAA,QACR;AAAA,QACA,mBAAmB,IAAI,qBAAqB;AAAA,QAC5C,OAAO,IAAI,SAAS,EAAE,MAAM,WAAW,SAAS,IAAI;AAAA,MACtD;AAAA,IACF;AACA,UAAM,gBAAgB,OAAO,KAAK,KAAK,UAAU,OAAO,CAAC,EAAE,SAAS,QAAQ;AAG5E,YAAQ,IAAI,4CAA4C;AACxD,UAAM,kBAAuB,EAAE,SAAS,OAAO;AAC/C,QAAI,QAAQ,OAAO;AACjB,sBAAgB,QAAQ,QAAQ;AAAA,IAClC;AACA,UAAM,UAAU,MAAM,MAAM,GAAG,SAAS,YAAY;AAAA,MAClD,QAAQ;AAAA,MACR,SAAS;AAAA,QACP,gBAAgB;AAAA,QAChB,CAAC,cAAc,GAAG;AAAA,MACpB;AAAA,MACA,MAAM,KAAK,UAAU,eAAe;AAAA,IACtC,CAAC;AAED,UAAM,SAAS,MAAM,QAAQ,KAAK;AAElC,QAAI,CAAC,QAAQ,IAAI;AACf,YAAM,IAAI,MAAM,OAAO,SAAS,0BAA0B;AAAA,IAC5D;AAGA,SAAK,eAAe,MAAM;AAE1B,YAAQ,IAAI,gCAAgC,OAAO,SAAS,UAAU,SAAS,EAAE;AAEjF,WAAO,OAAO;AAAA,EAChB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAc,YACZ,IACA,QACA,OACA,QAAqB,QACgD;AACrE,UAAM,aAAa;AACnB,UAAM,cAAc,KAAK,MAAM,KAAK,IAAI,IAAI,GAAI,IAAI;AACpD,UAAM,QAAQ,qBAAO,QAAQ,qBAAO,YAAY,EAAE,CAAC;AAEnD,UAAM,cAAc,MAAM,OAAO,KAAK;AACtC,UAAM,QAAQ,OAAO,KAAK,MAAM,SAAU,MAAM,YAAY,QAAS,CAAC,EAAE,SAAS;AAEjF,UAAM,gBAAsC;AAAA,MAC1C,MAAM,KAAK,OAAQ;AAAA,MACnB;AAAA,MACA;AAAA,MACA,YAAY,WAAW,SAAS;AAAA,MAChC,aAAa,YAAY,SAAS;AAAA,MAClC;AAAA,IACF;AAGA,UAAM,YAAY,UAAU,SAAS,aAAa;AAClD,UAAM,SAAS;AAAA,MACb,MAAM;AAAA,MACN,SAAS;AAAA,MACT,SAAS,MAAM;AAAA,MACf,mBAAmB,YAAY;AAAA,IACjC;AAGA,UAAM,QAAQ;AAAA,MACZ,2BAA2B;AAAA,QACzB,EAAE,MAAM,QAAQ,MAAM,UAAU;AAAA,QAChC,EAAE,MAAM,MAAM,MAAM,UAAU;AAAA,QAC9B,EAAE,MAAM,SAAS,MAAM,UAAU;AAAA,QACjC,EAAE,MAAM,cAAc,MAAM,UAAU;AAAA,QACtC,EAAE,MAAM,eAAe,MAAM,UAAU;AAAA,QACvC,EAAE,MAAM,SAAS,MAAM,UAAU;AAAA,MACnC;AAAA,IACF;AAEA,UAAM,YAAY,MAAM,KAAK,OAAQ,cAAc,QAAQ,OAAO,aAAa;AAE/E,WAAO,EAAE,eAAe,UAAU;AAAA,EACpC;AAAA;AAAA;AAAA;AAAA,EAKQ,YAAY,QAAsB;AAExC,QAAI,SAAS,KAAK,OAAO,OAAO,UAAU;AACxC,YAAM,IAAI;AAAA,QACR,WAAW,MAAM,kCAAkC,KAAK,OAAO,OAAO,QAAQ;AAAA,MAChF;AAAA,IACF;AAGA,UAAM,SAAQ,oBAAI,KAAK,GAAE,SAAS,GAAG,GAAG,GAAG,CAAC;AAC5C,QAAI,QAAQ,KAAK,mBAAmB;AAClC,WAAK,gBAAgB;AACrB,WAAK,oBAAoB;AACzB,WAAK,aAAa;AAAA,IACpB;AAGA,QAAI,KAAK,gBAAgB,SAAS,KAAK,OAAO,OAAO,WAAW;AAC9D,YAAM,IAAI;AAAA,QACR,8BAA8B,KAAK,aAAa,OAAO,MAAM,OAAO,KAAK,OAAO,OAAO,SAAS;AAAA,MAClG;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,eAAe,QAAsB;AAC3C,SAAK,iBAAiB;AACtB,SAAK,aAAa;AAAA,EACpB;AAAA;AAAA,EAIQ,aAA2B;AACjC,UAAM,iBAAa,kBAAK,KAAK,WAAW,aAAa;AACrD,YAAI,sBAAW,UAAU,GAAG;AAC1B,YAAM,cAAU,wBAAa,YAAY,OAAO;AAChD,aAAO,EAAE,GAAG,gBAAgB,GAAG,KAAK,MAAM,OAAO,EAAE;AAAA,IACrD;AACA,WAAO,EAAE,GAAG,eAAe;AAAA,EAC7B;AAAA,EAEQ,aAAmB;AACzB,6BAAU,KAAK,WAAW,EAAE,WAAW,KAAK,CAAC;AAC7C,UAAM,iBAAa,kBAAK,KAAK,WAAW,aAAa;AACrD,iCAAc,YAAY,KAAK,UAAU,KAAK,QAAQ,MAAM,CAAC,CAAC;AAAA,EAChE;AAAA;AAAA;AAAA;AAAA,EAKQ,eAAqB;AAC3B,UAAM,mBAAe,kBAAK,KAAK,WAAW,eAAe;AACzD,YAAI,sBAAW,YAAY,GAAG;AAC5B,UAAI;AACF,cAAM,OAAO,KAAK,UAAM,wBAAa,cAAc,OAAO,CAAC;AAC3D,cAAM,SAAQ,oBAAI,KAAK,GAAE,SAAS,GAAG,GAAG,GAAG,CAAC;AAG5C,YAAI,KAAK,QAAQ,KAAK,SAAS,OAAO;AACpC,eAAK,gBAAgB,KAAK,UAAU;AACpC,eAAK,oBAAoB,KAAK;AAAA,QAChC,OAAO;AAEL,eAAK,gBAAgB;AACrB,eAAK,oBAAoB;AAAA,QAC3B;AAAA,MACF,QAAQ;AAEN,aAAK,gBAAgB;AACrB,aAAK,qBAAoB,oBAAI,KAAK,GAAE,SAAS,GAAG,GAAG,GAAG,CAAC;AAAA,MACzD;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,eAAqB;AAC3B,6BAAU,KAAK,WAAW,EAAE,WAAW,KAAK,CAAC;AAC7C,UAAM,mBAAe,kBAAK,KAAK,WAAW,eAAe;AACzD,UAAM,OAAO;AAAA,MACX,MAAM,KAAK,sBAAqB,oBAAI,KAAK,GAAE,SAAS,GAAG,GAAG,GAAG,CAAC;AAAA,MAC9D,QAAQ,KAAK;AAAA,MACb,WAAW,KAAK,IAAI;AAAA,IACtB;AACA,iCAAc,cAAc,KAAK,UAAU,MAAM,MAAM,CAAC,CAAC;AAAA,EAC3D;AAAA,EAEQ,aAAgC;AACtC,UAAM,iBAAa,kBAAK,KAAK,WAAW,aAAa;AACrD,YAAI,sBAAW,UAAU,GAAG;AAE1B,UAAI;AACF,cAAM,YAAQ,oBAAS,UAAU;AACjC,cAAM,OAAO,MAAM,OAAO;AAC1B,YAAI,SAAS,KAAO;AAClB,kBAAQ,KAAK,6DAA6D,KAAK,SAAS,CAAC,CAAC,GAAG;AAC7F,kBAAQ,KAAK,0CAA0C;AACvD,mCAAU,YAAY,GAAK;AAAA,QAC7B;AAAA,MACF,SAAS,KAAK;AAAA,MAEd;AAEA,YAAM,cAAU,wBAAa,YAAY,OAAO;AAChD,aAAO,KAAK,MAAM,OAAO;AAAA,IAC3B;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,OAAO,KACL,WACA,SACwC;AACxC,6BAAU,WAAW,EAAE,WAAW,KAAK,CAAC;AAGxC,UAAM,SAAS,qBAAO,aAAa;AACnC,UAAM,aAAyB;AAAA,MAC7B,SAAS,OAAO;AAAA,MAChB,YAAY,OAAO;AAAA,MACnB,WAAW,KAAK,IAAI;AAAA,IACtB;AAGA,UAAM,iBAAa,kBAAK,WAAW,aAAa;AAChD,iCAAc,YAAY,KAAK,UAAU,YAAY,MAAM,CAAC,GAAG,EAAE,MAAM,IAAM,CAAC;AAG9E,UAAM,SAAuB;AAAA,MAC3B,OAAO,QAAQ;AAAA,MACf,QAAQ;AAAA,QACN,UAAU,QAAQ;AAAA,QAClB,WAAW,QAAQ;AAAA,MACrB;AAAA,IACF;AACA,UAAM,iBAAa,kBAAK,WAAW,aAAa;AAChD,iCAAc,YAAY,KAAK,UAAU,QAAQ,MAAM,CAAC,CAAC;AAEzD,WAAO,EAAE,SAAS,OAAO,SAAS,UAAU;AAAA,EAC9C;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,aAAsE;AAC1E,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,wBAAwB;AAAA,IAC1C;AAEA,QAAI;AACJ,QAAI;AACF,cAAQ,SAAS,KAAK,OAAO,KAAkB;AAAA,IACjD,QAAQ;AACN,YAAM,IAAI,MAAM,kBAAkB,KAAK,OAAO,KAAK,EAAE;AAAA,IACvD;AAEA,UAAM,WAAW,IAAI,qBAAO,gBAAgB,MAAM,GAAG;AACrD,UAAM,WAAW,CAAC,oDAAoD;AAGtE,UAAM,CAAC,eAAe,aAAa,WAAW,IAAI,MAAM,QAAQ,IAAI;AAAA,MAClE,SAAS,WAAW,KAAK,OAAO,OAAO;AAAA,MACvC,IAAI,qBAAO,SAAS,MAAM,OAAO,KAAK,SAAS,UAAU,QAAQ,EAAE,UAAU,KAAK,OAAO,OAAO;AAAA,MAChG,IAAI,qBAAO,SAAS,MAAM,OAAO,KAAK,SAAS,UAAU,QAAQ,EAAE,UAAU,KAAK,OAAO,OAAO;AAAA,IAClG,CAAC;AAED,WAAO;AAAA,MACL,MAAM,WAAW,qBAAO,YAAY,aAAa,MAAM,OAAO,KAAK,QAAQ,CAAC;AAAA,MAC5E,MAAM,WAAW,qBAAO,YAAY,aAAa,MAAM,OAAO,KAAK,QAAQ,CAAC;AAAA,MAC5E,QAAQ,WAAW,qBAAO,YAAY,aAAa,CAAC;AAAA,IACtD;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,iBAA0F;AAC9F,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,wBAAwB;AAAA,IAC1C;AAEA,UAAM,kBAA+B,CAAC,QAAQ,SAAS;AACvD,UAAM,WAAW,CAAC,oDAAoD;AACtE,UAAM,UAA0E,CAAC;AAGjF,UAAM,QAAQ;AAAA,MACZ,gBAAgB,IAAI,OAAO,cAAc;AACvC,YAAI;AACF,gBAAM,QAAQ,SAAS,SAAS;AAChC,gBAAM,WAAW,IAAI,qBAAO,gBAAgB,MAAM,GAAG;AAErD,gBAAM,CAAC,eAAe,aAAa,WAAW,IAAI,MAAM,QAAQ,IAAI;AAAA,YAClE,SAAS,WAAW,KAAK,OAAQ,OAAO;AAAA,YACxC,IAAI,qBAAO,SAAS,MAAM,OAAO,KAAK,SAAS,UAAU,QAAQ,EAAE,UAAU,KAAK,OAAQ,OAAO;AAAA,YACjG,IAAI,qBAAO,SAAS,MAAM,OAAO,KAAK,SAAS,UAAU,QAAQ,EAAE,UAAU,KAAK,OAAQ,OAAO;AAAA,UACnG,CAAC;AAED,kBAAQ,SAAS,IAAI;AAAA,YACnB,MAAM,WAAW,qBAAO,YAAY,aAAa,MAAM,OAAO,KAAK,QAAQ,CAAC;AAAA,YAC5E,MAAM,WAAW,qBAAO,YAAY,aAAa,MAAM,OAAO,KAAK,QAAQ,CAAC;AAAA,YAC5E,QAAQ,WAAW,qBAAO,YAAY,aAAa,CAAC;AAAA,UACtD;AAAA,QACF,SAAS,KAAK;AAEZ,kBAAQ,SAAS,IAAI,EAAE,MAAM,GAAG,MAAM,GAAG,QAAQ,EAAE;AAAA,QACrD;AAAA,MACF,CAAC;AAAA,IACH;AAEA,WAAO;AAAA,EACT;AACF;","names":["network"]}
1
+ {"version":3,"sources":["../../src/client/index.ts","../../src/chains/index.ts"],"sourcesContent":["/**\n * MoltsPay Client - Pay for AI Agent services\n * \n * Uses x402 protocol for gasless, pay-for-success payments.\n * \n * Usage:\n * const client = new MoltsPayClient(); // Loads from ~/.moltspay/\n * const services = await client.getServices('http://provider:3000');\n * const result = await client.pay('http://provider:3000', 'text-to-video', { prompt: '...' });\n */\n\nimport { existsSync, readFileSync, writeFileSync, mkdirSync, statSync, chmodSync } from 'fs';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { Wallet, ethers } from 'ethers';\nimport { getChain, type ChainName, type TokenSymbol } from '../chains/index.js';\nimport {\n ClientConfig,\n WalletData,\n ServicesResponse,\n MoltsPayClientOptions,\n} from './types.js';\n\nexport * from './types.js';\n\nexport interface PayOptions {\n /** Token to pay with (default: USDC, or auto-select based on balance) */\n token?: TokenSymbol;\n /** Auto-select token based on balance (default: false) */\n autoSelect?: boolean;\n /** Chain to pay on (base, polygon, or base_sepolia, default: base) */\n chain?: 'base' | 'polygon' | 'base_sepolia';\n}\n\n// x402 constants\nconst X402_VERSION = 2;\nconst PAYMENT_REQUIRED_HEADER = 'x-payment-required';\nconst PAYMENT_HEADER = 'x-payment';\n\ninterface X402PaymentRequirements {\n scheme: string;\n network: string;\n // v2 fields\n amount?: string;\n asset?: string;\n payTo?: string;\n maxTimeoutSeconds?: number;\n extra?: Record<string, unknown>;\n // v1 fields (legacy)\n maxAmountRequired?: string;\n resource?: string;\n description?: string;\n}\n\ninterface EIP3009Authorization {\n from: string;\n to: string;\n value: string;\n validAfter: string;\n validBefore: string;\n nonce: string;\n}\n\nconst DEFAULT_CONFIG: ClientConfig = {\n chain: 'base',\n limits: {\n maxPerTx: 100,\n maxPerDay: 1000,\n },\n};\n\nexport class MoltsPayClient {\n private configDir: string;\n private config: ClientConfig;\n private walletData: WalletData | null = null;\n private wallet: Wallet | null = null;\n private todaySpending: number = 0;\n private lastSpendingReset: number = 0;\n\n constructor(options: MoltsPayClientOptions = {}) {\n this.configDir = options.configDir || join(homedir(), '.moltspay');\n this.config = this.loadConfig();\n this.walletData = this.loadWallet();\n this.loadSpending(); // Load persisted spending data\n \n if (this.walletData) {\n this.wallet = new Wallet(this.walletData.privateKey);\n }\n }\n\n /**\n * Check if client is initialized (has wallet)\n */\n get isInitialized(): boolean {\n return this.wallet !== null;\n }\n\n /**\n * Get wallet address\n */\n get address(): string | null {\n return this.wallet?.address || null;\n }\n\n /**\n * Get current config\n */\n getConfig(): ClientConfig {\n return { ...this.config };\n }\n\n /**\n * Update config\n */\n updateConfig(updates: Partial<ClientConfig['limits']>): void {\n if (updates.maxPerTx !== undefined) {\n this.config.limits.maxPerTx = updates.maxPerTx;\n }\n if (updates.maxPerDay !== undefined) {\n this.config.limits.maxPerDay = updates.maxPerDay;\n }\n this.saveConfig();\n }\n\n /**\n * Get services from a provider\n */\n async getServices(serverUrl: string): Promise<ServicesResponse> {\n // Normalize URL - don't append /services if already present\n const normalizedUrl = serverUrl.replace(/\\/(services|api\\/services|registry\\/services)\\/?$/, '');\n \n // Try /services first (standard provider endpoint)\n const endpoints = ['/services', '/api/services', '/registry/services'];\n \n for (const endpoint of endpoints) {\n try {\n const res = await fetch(`${normalizedUrl}${endpoint}`);\n if (!res.ok) continue;\n \n const contentType = res.headers.get('content-type') || '';\n if (!contentType.includes('application/json')) continue;\n \n return await res.json() as ServicesResponse;\n } catch {\n continue;\n }\n }\n \n throw new Error(`Failed to get services: no valid endpoint found at ${normalizedUrl}`);\n }\n\n /**\n * Pay for a service and get the result (x402 protocol)\n * \n * This is GASLESS for the client - server pays gas to claim payment.\n * This is PAY-FOR-SUCCESS - payment only claimed if service succeeds.\n * \n * @param serverUrl - Server URL\n * @param service - Service ID\n * @param params - Service parameters\n * @param options - Payment options (token selection)\n */\n async pay(\n serverUrl: string,\n service: string,\n params: Record<string, any>,\n options: PayOptions = {}\n ): Promise<Record<string, any>> {\n if (!this.wallet || !this.walletData) {\n throw new Error('Client not initialized. Run: npx moltspay init');\n }\n\n // Step 1: Make initial request without payment\n console.log(`[MoltsPay] Requesting service: ${service}`);\n const requestBody: any = { service, params };\n if (options.chain) {\n requestBody.chain = options.chain;\n }\n const initialRes = await fetch(`${serverUrl}/execute`, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(requestBody),\n });\n\n // If not 402, check for success or error\n if (initialRes.status !== 402) {\n const data = await initialRes.json() as any;\n if (initialRes.ok && data.result) {\n return data.result;\n }\n throw new Error(data.error || 'Unexpected response');\n }\n\n // Step 2: Parse payment requirements from 402 response\n const paymentRequiredHeader = initialRes.headers.get(PAYMENT_REQUIRED_HEADER);\n if (!paymentRequiredHeader) {\n throw new Error('Missing x-payment-required header');\n }\n\n let requirements: X402PaymentRequirements[];\n try {\n const decoded = Buffer.from(paymentRequiredHeader, 'base64').toString('utf-8');\n const parsed = JSON.parse(decoded);\n \n // Handle both v1 (array) and v2 (object with accepts) formats\n if (Array.isArray(parsed)) {\n // v1 format: direct array of requirements\n requirements = parsed;\n } else if (parsed.accepts && Array.isArray(parsed.accepts)) {\n // v2 format: { x402Version: 2, accepts: [...] }\n requirements = parsed.accepts;\n } else {\n // Single requirement object\n requirements = [parsed];\n }\n } catch {\n throw new Error('Invalid x-payment-required header');\n }\n\n // Helper to convert network ID to chain name\n const networkToChainName = (network: string): string | null => {\n const match = network.match(/^eip155:(\\d+)$/);\n if (!match) return null;\n const chainId = parseInt(match[1]);\n if (chainId === 8453) return 'base';\n if (chainId === 137) return 'polygon';\n if (chainId === 84532) return 'base_sepolia';\n return null;\n };\n\n // Get server's accepted chains\n const serverChains = requirements\n .map(r => networkToChainName(r.network))\n .filter((c): c is string => c !== null);\n\n // Determine which chain to use\n let chainName: ChainName;\n const userSpecifiedChain = options.chain;\n\n if (userSpecifiedChain) {\n // User specified --chain, validate it's accepted by server\n if (!serverChains.includes(userSpecifiedChain)) {\n throw new Error(\n `Server doesn't accept '${userSpecifiedChain}'.\\n` +\n `Server accepts: ${serverChains.join(', ')}`\n );\n }\n chainName = userSpecifiedChain as ChainName;\n } else {\n // No --chain provided\n if (serverChains.length === 1 && serverChains[0] === 'base') {\n // Only default to base if server ONLY accepts base\n chainName = 'base';\n } else {\n throw new Error(\n `Server accepts: ${serverChains.join(', ')}\\n` +\n `Please specify: --chain base, --chain polygon, or --chain base_sepolia`\n );\n }\n }\n\n const chain = getChain(chainName);\n const network = `eip155:${chain.chainId}`;\n const req = requirements.find(r => r.scheme === 'exact' && r.network === network);\n\n if (!req) {\n throw new Error(`Failed to find payment requirement for ${chainName}`);\n }\n\n // Step 3: Check limits\n // v2 uses 'amount', v1 uses 'maxAmountRequired'\n const amountRaw = req.amount || req.maxAmountRequired;\n if (!amountRaw) {\n throw new Error('Missing amount in payment requirements');\n }\n const amount = Number(amountRaw) / 1e6;\n this.checkLimits(amount);\n\n // Determine which token to use\n let token: TokenSymbol = options.token || 'USDC';\n \n // Auto-select token based on balance if requested\n if (options.autoSelect) {\n const balances = await this.getBalance();\n if (balances.usdc >= amount) {\n token = 'USDC';\n } else if (balances.usdt >= amount) {\n token = 'USDT';\n } else {\n throw new Error(`Insufficient balance: need $${amount}, have ${balances.usdc} USDC / ${balances.usdt} USDT`);\n }\n }\n\n // USDT does not support gasless transfers (no EIP-2612 permit)\n // It requires on-chain approve + transfer, meaning the user pays gas\n if (token === 'USDT') {\n const balances = await this.getBalance();\n if (balances.native < 0.0001) {\n throw new Error(\n `USDT requires ETH for gas (~$0.01 on Base). ` +\n `Your ETH balance: ${balances.native.toFixed(6)} ETH. ` +\n `Please add a small amount of ETH to your wallet, or use USDC (gasless).`\n );\n }\n console.log(`[MoltsPay] ⚠️ USDT requires gas (~$0.01). Proceeding with payment...`);\n } else {\n console.log(`[MoltsPay] Signing payment: $${amount} ${token} (gasless)`);\n }\n\n // Step 4: Sign EIP-3009 authorization (GASLESS - just signing)\n // payTo is the recipient address (v2 format)\n const payTo = req.payTo || req.resource; // fallback for v1 compatibility\n if (!payTo) {\n throw new Error('Missing payTo address in payment requirements');\n }\n \n // Use server's extra field for domain info (contains correct EIP-712 domain for the token on this network)\n const domainOverride = (req.extra && typeof req.extra === 'object' && req.extra.name) \n ? { name: req.extra.name as string, version: (req.extra.version as string) || '2' }\n : undefined;\n \n const authorization = await this.signEIP3009(payTo, amount, chain, token, domainOverride);\n\n // Get token-specific info for accepted field\n const tokenConfig = chain.tokens[token];\n\n // Step 5: Create x402 payment payload (v2 requires scheme, network, payload, AND accepted)\n // Use server's extra field if provided (contains correct EIP-712 domain for the token on this network)\n // Fall back to local config for backward compatibility\n const extra = (req.extra && typeof req.extra === 'object') \n ? req.extra \n : {\n name: (tokenConfig as any).eip712Name || 'USD Coin',\n version: '2',\n };\n \n const payload = {\n x402Version: X402_VERSION,\n scheme: 'exact',\n network,\n payload: authorization, // { authorization: {...}, signature: \"0x...\" }\n accepted: {\n scheme: 'exact',\n network,\n asset: tokenConfig.address,\n amount: amountRaw,\n payTo,\n maxTimeoutSeconds: req.maxTimeoutSeconds || 300,\n extra,\n },\n };\n const paymentHeader = Buffer.from(JSON.stringify(payload)).toString('base64');\n\n // Step 6: Retry with payment header\n console.log(`[MoltsPay] Sending request with payment...`);\n const paidRequestBody: any = { service, params };\n if (options.chain) {\n paidRequestBody.chain = options.chain;\n }\n const paidRes = await fetch(`${serverUrl}/execute`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n [PAYMENT_HEADER]: paymentHeader,\n },\n body: JSON.stringify(paidRequestBody),\n });\n\n const result = await paidRes.json() as any;\n\n if (!paidRes.ok) {\n throw new Error(result.error || 'Service execution failed');\n }\n\n // Update spending tracking\n this.recordSpending(amount);\n\n console.log(`[MoltsPay] Success! Payment: ${result.payment?.status || 'claimed'}`);\n \n return result.result;\n }\n\n /**\n * Sign EIP-3009 transferWithAuthorization (GASLESS)\n * This only signs - no on-chain transaction, no gas needed.\n * Supports both USDC and USDT.\n */\n private async signEIP3009(\n to: string,\n amount: number,\n chain: { chainId: number; tokens: Record<TokenSymbol, { address: string; decimals: number }> },\n token: TokenSymbol = 'USDC',\n domainOverride?: { name: string; version: string }\n ): Promise<{ authorization: EIP3009Authorization; signature: string }> {\n const validAfter = 0;\n const validBefore = Math.floor(Date.now() / 1000) + 3600; // 1 hour\n const nonce = ethers.hexlify(ethers.randomBytes(32));\n \n const tokenConfig = chain.tokens[token];\n const value = BigInt(Math.floor(amount * (10 ** tokenConfig.decimals))).toString();\n\n const authorization: EIP3009Authorization = {\n from: this.wallet!.address,\n to,\n value,\n validAfter: validAfter.toString(),\n validBefore: validBefore.toString(),\n nonce,\n };\n\n // EIP-712 domain - use server's domain info if provided (handles mainnet vs testnet differences)\n // Fall back to local token config for backward compatibility\n const tokenName = domainOverride?.name || (tokenConfig as any).eip712Name || (token === 'USDC' ? 'USD Coin' : 'Tether USD');\n const tokenVersion = domainOverride?.version || '2';\n const domain = {\n name: tokenName,\n version: tokenVersion,\n chainId: chain.chainId,\n verifyingContract: tokenConfig.address,\n };\n\n // EIP-3009 types\n const types = {\n TransferWithAuthorization: [\n { name: 'from', type: 'address' },\n { name: 'to', type: 'address' },\n { name: 'value', type: 'uint256' },\n { name: 'validAfter', type: 'uint256' },\n { name: 'validBefore', type: 'uint256' },\n { name: 'nonce', type: 'bytes32' },\n ],\n };\n\n const signature = await this.wallet!.signTypedData(domain, types, authorization);\n\n return { authorization, signature };\n }\n\n /**\n * Check spending limits\n */\n private checkLimits(amount: number): void {\n // Check per-tx limit\n if (amount > this.config.limits.maxPerTx) {\n throw new Error(\n `Amount $${amount} exceeds max per transaction ($${this.config.limits.maxPerTx})`\n );\n }\n\n // Reset daily spending if new day\n const today = new Date().setHours(0, 0, 0, 0);\n if (today > this.lastSpendingReset) {\n this.todaySpending = 0;\n this.lastSpendingReset = today;\n this.saveSpending(); // Persist reset\n }\n\n // Check daily limit\n if (this.todaySpending + amount > this.config.limits.maxPerDay) {\n throw new Error(\n `Would exceed daily limit ($${this.todaySpending} + $${amount} > $${this.config.limits.maxPerDay})`\n );\n }\n }\n\n /**\n * Record spending and persist to disk\n */\n private recordSpending(amount: number): void {\n this.todaySpending += amount;\n this.saveSpending();\n }\n\n // --- Config & Wallet Management ---\n\n private loadConfig(): ClientConfig {\n const configPath = join(this.configDir, 'config.json');\n if (existsSync(configPath)) {\n const content = readFileSync(configPath, 'utf-8');\n return { ...DEFAULT_CONFIG, ...JSON.parse(content) };\n }\n return { ...DEFAULT_CONFIG };\n }\n\n private saveConfig(): void {\n mkdirSync(this.configDir, { recursive: true });\n const configPath = join(this.configDir, 'config.json');\n writeFileSync(configPath, JSON.stringify(this.config, null, 2));\n }\n\n /**\n * Load spending data from disk\n */\n private loadSpending(): void {\n const spendingPath = join(this.configDir, 'spending.json');\n if (existsSync(spendingPath)) {\n try {\n const data = JSON.parse(readFileSync(spendingPath, 'utf-8'));\n const today = new Date().setHours(0, 0, 0, 0);\n \n // Only load if it's from today\n if (data.date && data.date === today) {\n this.todaySpending = data.amount || 0;\n this.lastSpendingReset = data.date;\n } else {\n // Data is from a previous day, reset\n this.todaySpending = 0;\n this.lastSpendingReset = today;\n }\n } catch {\n // Ignore parse errors, start fresh\n this.todaySpending = 0;\n this.lastSpendingReset = new Date().setHours(0, 0, 0, 0);\n }\n }\n }\n\n /**\n * Save spending data to disk\n */\n private saveSpending(): void {\n mkdirSync(this.configDir, { recursive: true });\n const spendingPath = join(this.configDir, 'spending.json');\n const data = {\n date: this.lastSpendingReset || new Date().setHours(0, 0, 0, 0),\n amount: this.todaySpending,\n updatedAt: Date.now(),\n };\n writeFileSync(spendingPath, JSON.stringify(data, null, 2));\n }\n\n private loadWallet(): WalletData | null {\n const walletPath = join(this.configDir, 'wallet.json');\n if (existsSync(walletPath)) {\n // Security check: warn and fix if permissions are too open\n try {\n const stats = statSync(walletPath);\n const mode = stats.mode & 0o777;\n if (mode !== 0o600) {\n console.warn(`[MoltsPay] WARNING: wallet.json has insecure permissions (${mode.toString(8)})`);\n console.warn(`[MoltsPay] Fixing permissions to 0600...`);\n chmodSync(walletPath, 0o600);\n }\n } catch (err) {\n // Ignore permission check errors on Windows\n }\n \n const content = readFileSync(walletPath, 'utf-8');\n return JSON.parse(content);\n }\n return null;\n }\n\n /**\n * Initialize a new wallet (called by CLI)\n */\n static init(\n configDir: string,\n options: { chain: string; maxPerTx: number; maxPerDay: number }\n ): { address: string; configDir: string } {\n mkdirSync(configDir, { recursive: true });\n\n // Create wallet\n const wallet = Wallet.createRandom();\n const walletData: WalletData = {\n address: wallet.address,\n privateKey: wallet.privateKey,\n createdAt: Date.now(),\n };\n\n // Save wallet with secure permissions (0o600 = owner read/write only)\n const walletPath = join(configDir, 'wallet.json');\n writeFileSync(walletPath, JSON.stringify(walletData, null, 2), { mode: 0o600 });\n\n // Save config\n const config: ClientConfig = {\n chain: options.chain,\n limits: {\n maxPerTx: options.maxPerTx,\n maxPerDay: options.maxPerDay,\n },\n };\n const configPath = join(configDir, 'config.json');\n writeFileSync(configPath, JSON.stringify(config, null, 2));\n\n return { address: wallet.address, configDir };\n }\n\n /**\n * Get wallet balance (USDC, USDT, and native token) on default chain\n */\n async getBalance(): Promise<{ usdc: number; usdt: number; native: number }> {\n if (!this.wallet) {\n throw new Error('Client not initialized');\n }\n\n let chain;\n try {\n chain = getChain(this.config.chain as ChainName);\n } catch {\n throw new Error(`Unknown chain: ${this.config.chain}`);\n }\n\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n const tokenAbi = ['function balanceOf(address) view returns (uint256)'];\n\n // Get all balances in parallel\n const [nativeBalance, usdcBalance, usdtBalance] = await Promise.all([\n provider.getBalance(this.wallet.address),\n new ethers.Contract(chain.tokens.USDC.address, tokenAbi, provider).balanceOf(this.wallet.address),\n new ethers.Contract(chain.tokens.USDT.address, tokenAbi, provider).balanceOf(this.wallet.address),\n ]);\n\n return {\n usdc: parseFloat(ethers.formatUnits(usdcBalance, chain.tokens.USDC.decimals)),\n usdt: parseFloat(ethers.formatUnits(usdtBalance, chain.tokens.USDT.decimals)),\n native: parseFloat(ethers.formatEther(nativeBalance)),\n };\n }\n\n /**\n * Get wallet balances on all supported chains (Base + Polygon)\n */\n async getAllBalances(): Promise<Record<string, { usdc: number; usdt: number; native: number }>> {\n if (!this.wallet) {\n throw new Error('Client not initialized');\n }\n\n const supportedChains: ChainName[] = ['base', 'polygon', 'base_sepolia'];\n const tokenAbi = ['function balanceOf(address) view returns (uint256)'];\n const results: Record<string, { usdc: number; usdt: number; native: number }> = {};\n\n // Query all chains in parallel\n await Promise.all(\n supportedChains.map(async (chainName) => {\n try {\n const chain = getChain(chainName);\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n\n const [nativeBalance, usdcBalance, usdtBalance] = await Promise.all([\n provider.getBalance(this.wallet!.address),\n new ethers.Contract(chain.tokens.USDC.address, tokenAbi, provider).balanceOf(this.wallet!.address),\n new ethers.Contract(chain.tokens.USDT.address, tokenAbi, provider).balanceOf(this.wallet!.address),\n ]);\n\n results[chainName] = {\n usdc: parseFloat(ethers.formatUnits(usdcBalance, chain.tokens.USDC.decimals)),\n usdt: parseFloat(ethers.formatUnits(usdtBalance, chain.tokens.USDT.decimals)),\n native: parseFloat(ethers.formatEther(nativeBalance)),\n };\n } catch (err) {\n // If chain query fails, show zeros\n results[chainName] = { usdc: 0, usdt: 0, native: 0 };\n }\n })\n );\n\n return results;\n }\n}\n","/**\n * Blockchain Configuration\n */\n\nimport type { ChainConfig, ChainName, TokenSymbol } from '../types/index.js';\n\nexport const CHAINS: Record<ChainName, ChainConfig> = {\n // ============ Mainnet ============\n base: {\n name: 'Base',\n chainId: 8453,\n rpc: 'https://mainnet.base.org',\n tokens: {\n USDC: {\n address: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913',\n decimals: 6,\n symbol: 'USDC',\n eip712Name: 'USD Coin', // EIP-712 domain name\n },\n USDT: {\n address: '0xfde4C96c8593536E31F229EA8f37b2ADa2699bb2',\n decimals: 6,\n symbol: 'USDT',\n eip712Name: 'Tether USD',\n },\n },\n usdc: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913', // deprecated, for backward compat\n explorer: 'https://basescan.org/address/',\n explorerTx: 'https://basescan.org/tx/',\n avgBlockTime: 2,\n },\n polygon: {\n name: 'Polygon',\n chainId: 137,\n rpc: 'https://polygon-bor-rpc.publicnode.com',\n tokens: {\n USDC: {\n address: '0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359',\n decimals: 6,\n symbol: 'USDC',\n eip712Name: 'USD Coin',\n },\n USDT: {\n address: '0xc2132D05D31c914a87C6611C10748AEb04B58e8F',\n decimals: 6,\n symbol: 'USDT',\n eip712Name: '(PoS) Tether USD', // Polygon uses this name\n },\n },\n usdc: '0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359',\n explorer: 'https://polygonscan.com/address/',\n explorerTx: 'https://polygonscan.com/tx/',\n avgBlockTime: 2,\n },\n // ============ Testnet ============\n base_sepolia: {\n name: 'Base Sepolia',\n chainId: 84532,\n rpc: 'https://sepolia.base.org',\n tokens: {\n USDC: {\n address: '0x036CbD53842c5426634e7929541eC2318f3dCF7e',\n decimals: 6,\n symbol: 'USDC',\n eip712Name: 'USDC', // Testnet USDC uses 'USDC' not 'USD Coin'\n },\n USDT: {\n address: '0x036CbD53842c5426634e7929541eC2318f3dCF7e', // Same as USDC on testnet (no official USDT)\n decimals: 6,\n symbol: 'USDT',\n eip712Name: 'USDC', // Uses same contract as USDC\n },\n },\n usdc: '0x036CbD53842c5426634e7929541eC2318f3dCF7e',\n explorer: 'https://sepolia.basescan.org/address/',\n explorerTx: 'https://sepolia.basescan.org/tx/',\n avgBlockTime: 2,\n },\n};\n\n/**\n * Get token address for a chain\n */\nexport function getTokenAddress(chainName: ChainName, token: TokenSymbol): string {\n const chain = CHAINS[chainName];\n if (!chain) {\n throw new Error(`Unsupported chain: ${chainName}`);\n }\n const tokenConfig = chain.tokens[token];\n if (!tokenConfig) {\n throw new Error(`Token ${token} not supported on ${chainName}`);\n }\n return tokenConfig.address;\n}\n\n/**\n * Get token config for a chain\n */\nexport function getTokenConfig(chainName: ChainName, token: TokenSymbol) {\n const chain = CHAINS[chainName];\n if (!chain) {\n throw new Error(`Unsupported chain: ${chainName}`);\n }\n return chain.tokens[token];\n}\n\n/**\n * Get chain configuration\n */\nexport function getChain(name: ChainName): ChainConfig {\n const config = CHAINS[name];\n if (!config) {\n throw new Error(`Unsupported chain: ${name}. Supported: ${Object.keys(CHAINS).join(', ')}`);\n }\n return config;\n}\n\n/**\n * List all supported chains\n */\nexport function listChains(): ChainName[] {\n return Object.keys(CHAINS) as ChainName[];\n}\n\n/**\n * Get chain config by chainId\n */\nexport function getChainById(chainId: number): ChainConfig | undefined {\n return Object.values(CHAINS).find(c => c.chainId === chainId);\n}\n\n/**\n * ERC20 ABI (minimal, only required methods)\n */\nexport const ERC20_ABI = [\n 'function balanceOf(address owner) view returns (uint256)',\n 'function transfer(address to, uint256 amount) returns (bool)',\n 'function approve(address spender, uint256 amount) returns (bool)',\n 'function allowance(address owner, address spender) view returns (uint256)',\n 'function decimals() view returns (uint8)',\n 'function symbol() view returns (string)',\n 'function name() view returns (string)',\n 'function nonces(address owner) view returns (uint256)',\n 'function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s)',\n 'event Transfer(address indexed from, address indexed to, uint256 value)',\n 'event Approval(address indexed owner, address indexed spender, uint256 value)',\n];\n\nexport type { ChainConfig, ChainName, TokenSymbol };\n"],"mappings":"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","names":["network"]}
@@ -15,12 +15,15 @@ var CHAINS = {
15
15
  USDC: {
16
16
  address: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",
17
17
  decimals: 6,
18
- symbol: "USDC"
18
+ symbol: "USDC",
19
+ eip712Name: "USD Coin"
20
+ // EIP-712 domain name
19
21
  },
20
22
  USDT: {
21
23
  address: "0xfde4C96c8593536E31F229EA8f37b2ADa2699bb2",
22
24
  decimals: 6,
23
- symbol: "USDT"
25
+ symbol: "USDT",
26
+ eip712Name: "Tether USD"
24
27
  }
25
28
  },
26
29
  usdc: "0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913",
@@ -37,12 +40,15 @@ var CHAINS = {
37
40
  USDC: {
38
41
  address: "0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",
39
42
  decimals: 6,
40
- symbol: "USDC"
43
+ symbol: "USDC",
44
+ eip712Name: "USD Coin"
41
45
  },
42
46
  USDT: {
43
47
  address: "0xc2132D05D31c914a87C6611C10748AEb04B58e8F",
44
48
  decimals: 6,
45
- symbol: "USDT"
49
+ symbol: "USDT",
50
+ eip712Name: "(PoS) Tether USD"
51
+ // Polygon uses this name
46
52
  }
47
53
  },
48
54
  usdc: "0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359",
@@ -50,27 +56,6 @@ var CHAINS = {
50
56
  explorerTx: "https://polygonscan.com/tx/",
51
57
  avgBlockTime: 2
52
58
  },
53
- ethereum: {
54
- name: "Ethereum",
55
- chainId: 1,
56
- rpc: "https://eth.llamarpc.com",
57
- tokens: {
58
- USDC: {
59
- address: "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
60
- decimals: 6,
61
- symbol: "USDC"
62
- },
63
- USDT: {
64
- address: "0xdAC17F958D2ee523a2206206994597C13D831ec7",
65
- decimals: 6,
66
- symbol: "USDT"
67
- }
68
- },
69
- usdc: "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48",
70
- explorer: "https://etherscan.io/address/",
71
- explorerTx: "https://etherscan.io/tx/",
72
- avgBlockTime: 12
73
- },
74
59
  // ============ Testnet ============
75
60
  base_sepolia: {
76
61
  name: "Base Sepolia",
@@ -80,41 +65,23 @@ var CHAINS = {
80
65
  USDC: {
81
66
  address: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
82
67
  decimals: 6,
83
- symbol: "USDC"
68
+ symbol: "USDC",
69
+ eip712Name: "USDC"
70
+ // Testnet USDC uses 'USDC' not 'USD Coin'
84
71
  },
85
72
  USDT: {
86
73
  address: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
87
74
  // Same as USDC on testnet (no official USDT)
88
75
  decimals: 6,
89
- symbol: "USDT"
76
+ symbol: "USDT",
77
+ eip712Name: "USDC"
78
+ // Uses same contract as USDC
90
79
  }
91
80
  },
92
81
  usdc: "0x036CbD53842c5426634e7929541eC2318f3dCF7e",
93
82
  explorer: "https://sepolia.basescan.org/address/",
94
83
  explorerTx: "https://sepolia.basescan.org/tx/",
95
84
  avgBlockTime: 2
96
- },
97
- sepolia: {
98
- name: "Sepolia",
99
- chainId: 11155111,
100
- rpc: "https://rpc.sepolia.org",
101
- tokens: {
102
- USDC: {
103
- address: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
104
- decimals: 6,
105
- symbol: "USDC"
106
- },
107
- USDT: {
108
- address: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
109
- // Same as USDC on testnet
110
- decimals: 6,
111
- symbol: "USDT"
112
- }
113
- },
114
- usdc: "0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238",
115
- explorer: "https://sepolia.etherscan.io/address/",
116
- explorerTx: "https://sepolia.etherscan.io/tx/",
117
- avgBlockTime: 12
118
85
  }
119
86
  };
120
87
  function getChain(name) {
@@ -277,7 +244,7 @@ Server accepts: ${serverChains.join(", ")}`
277
244
  } else {
278
245
  throw new Error(
279
246
  `Server accepts: ${serverChains.join(", ")}
280
- Please specify: --chain base or --chain polygon`
247
+ Please specify: --chain base, --chain polygon, or --chain base_sepolia`
281
248
  );
282
249
  }
283
250
  }
@@ -319,13 +286,19 @@ Please specify: --chain base or --chain polygon`
319
286
  if (!payTo) {
320
287
  throw new Error("Missing payTo address in payment requirements");
321
288
  }
322
- const authorization = await this.signEIP3009(payTo, amount, chain, token);
289
+ const domainOverride = req.extra && typeof req.extra === "object" && req.extra.name ? { name: req.extra.name, version: req.extra.version || "2" } : void 0;
290
+ const authorization = await this.signEIP3009(payTo, amount, chain, token, domainOverride);
323
291
  const tokenConfig = chain.tokens[token];
324
- const tokenName = token === "USDC" ? "USD Coin" : "Tether USD";
292
+ const extra = req.extra && typeof req.extra === "object" ? req.extra : {
293
+ name: tokenConfig.eip712Name || "USD Coin",
294
+ version: "2"
295
+ };
325
296
  const payload = {
326
297
  x402Version: X402_VERSION,
298
+ scheme: "exact",
299
+ network,
327
300
  payload: authorization,
328
- // v2 requires 'accepted' field with the requirements being fulfilled
301
+ // { authorization: {...}, signature: "0x..." }
329
302
  accepted: {
330
303
  scheme: "exact",
331
304
  network,
@@ -333,7 +306,7 @@ Please specify: --chain base or --chain polygon`
333
306
  amount: amountRaw,
334
307
  payTo,
335
308
  maxTimeoutSeconds: req.maxTimeoutSeconds || 300,
336
- extra: req.extra || { name: tokenName, version: "2" }
309
+ extra
337
310
  }
338
311
  };
339
312
  const paymentHeader = Buffer.from(JSON.stringify(payload)).toString("base64");
@@ -363,7 +336,7 @@ Please specify: --chain base or --chain polygon`
363
336
  * This only signs - no on-chain transaction, no gas needed.
364
337
  * Supports both USDC and USDT.
365
338
  */
366
- async signEIP3009(to, amount, chain, token = "USDC") {
339
+ async signEIP3009(to, amount, chain, token = "USDC", domainOverride) {
367
340
  const validAfter = 0;
368
341
  const validBefore = Math.floor(Date.now() / 1e3) + 3600;
369
342
  const nonce = ethers.hexlify(ethers.randomBytes(32));
@@ -377,10 +350,11 @@ Please specify: --chain base or --chain polygon`
377
350
  validBefore: validBefore.toString(),
378
351
  nonce
379
352
  };
380
- const tokenName = token === "USDC" ? "USD Coin" : "Tether USD";
353
+ const tokenName = domainOverride?.name || tokenConfig.eip712Name || (token === "USDC" ? "USD Coin" : "Tether USD");
354
+ const tokenVersion = domainOverride?.version || "2";
381
355
  const domain = {
382
356
  name: tokenName,
383
- version: "2",
357
+ version: tokenVersion,
384
358
  chainId: chain.chainId,
385
359
  verifyingContract: tokenConfig.address
386
360
  };
@@ -549,7 +523,7 @@ Please specify: --chain base or --chain polygon`
549
523
  if (!this.wallet) {
550
524
  throw new Error("Client not initialized");
551
525
  }
552
- const supportedChains = ["base", "polygon"];
526
+ const supportedChains = ["base", "polygon", "base_sepolia"];
553
527
  const tokenAbi = ["function balanceOf(address) view returns (uint256)"];
554
528
  const results = {};
555
529
  await Promise.all(