moltspay 0.1.3 → 0.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.
- package/README.md +126 -0
- package/dist/cli.js +17 -3
- package/dist/cli.js.map +1 -1
- package/dist/cli.mjs +17 -3
- package/dist/cli.mjs.map +1 -1
- package/dist/guide/index.d.mts +7 -7
- package/dist/guide/index.d.ts +7 -7
- package/dist/guide/index.js +50 -50
- package/dist/guide/index.js.map +1 -1
- package/dist/guide/index.mjs +50 -50
- package/dist/guide/index.mjs.map +1 -1
- package/dist/index.d.mts +249 -2
- package/dist/index.d.ts +249 -2
- package/dist/index.js +874 -63
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +858 -62
- package/dist/index.mjs.map +1 -1
- package/dist/wallet/index.d.mts +188 -1
- package/dist/wallet/index.d.ts +188 -1
- package/dist/wallet/index.js +378 -2
- package/dist/wallet/index.js.map +1 -1
- package/dist/wallet/index.mjs +371 -1
- package/dist/wallet/index.mjs.map +1 -1
- package/package.json +1 -1
package/README.md
CHANGED
|
@@ -10,6 +10,8 @@ Blockchain payment infrastructure for AI Agents on Moltbook.
|
|
|
10
10
|
- 🔒 **Secure Wallet** - Limits, whitelist, and audit logging
|
|
11
11
|
- 📝 **EIP-2612 Permit** - Gasless user payments
|
|
12
12
|
- ⛓️ **Multi-chain** - Base, Polygon, Ethereum (mainnet & testnet)
|
|
13
|
+
- 🤖 **Agent-to-Agent** - Complete A2A payment flow support (v0.2.0+)
|
|
14
|
+
- 🧾 **Receipt Generation** - Transaction receipts for audit/accounting
|
|
13
15
|
|
|
14
16
|
## Installation
|
|
15
17
|
|
|
@@ -68,6 +70,130 @@ Payment Agent is designed to be called by AI Agents (like Clawdbot) to handle pa
|
|
|
68
70
|
└─────────────────────────────────────────────────────────┘
|
|
69
71
|
```
|
|
70
72
|
|
|
73
|
+
## Agent-to-Agent Payment Flow (v0.2.0+)
|
|
74
|
+
|
|
75
|
+
Complete support for pure-conversation payment between AI Agents.
|
|
76
|
+
|
|
77
|
+
### Flow Overview
|
|
78
|
+
|
|
79
|
+
```
|
|
80
|
+
START → 能力识别 → 能力协商 → Onboarding → 服务请求 → 报价 → 支付 → 验证 → 交付 → 收据 → END
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
### Buyer Agent: Create Wallet & Pay
|
|
84
|
+
|
|
85
|
+
```typescript
|
|
86
|
+
import { createWallet, loadWallet, PermitWallet } from 'moltspay';
|
|
87
|
+
|
|
88
|
+
// Step 1: Create wallet (first time)
|
|
89
|
+
const result = createWallet({ password: 'secure123' });
|
|
90
|
+
console.log('Wallet address:', result.address);
|
|
91
|
+
// Send this address to Boss for Permit authorization
|
|
92
|
+
|
|
93
|
+
// Step 2: After Boss signs Permit, load wallet and pay
|
|
94
|
+
const { privateKey } = loadWallet({ password: 'secure123' });
|
|
95
|
+
const wallet = new PermitWallet({
|
|
96
|
+
chain: 'base',
|
|
97
|
+
privateKey
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
// Step 3: Pay using Boss's Permit
|
|
101
|
+
const payment = await wallet.transferWithPermit({
|
|
102
|
+
to: '0xSELLER...',
|
|
103
|
+
amount: 3.99,
|
|
104
|
+
permit: bossSignedPermit // { owner, spender, value, deadline, v, r, s }
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
console.log('Payment tx:', payment.tx_hash);
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
### Seller Agent: Verify & Deliver
|
|
111
|
+
|
|
112
|
+
```typescript
|
|
113
|
+
import {
|
|
114
|
+
PaymentAgent,
|
|
115
|
+
generateReceipt,
|
|
116
|
+
formatReceiptText,
|
|
117
|
+
SellerTemplates
|
|
118
|
+
} from 'moltspay';
|
|
119
|
+
|
|
120
|
+
const agent = new PaymentAgent({
|
|
121
|
+
chain: 'base',
|
|
122
|
+
walletAddress: '0xSELLER...',
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
// Step 1: Create invoice & quote
|
|
126
|
+
const invoice = agent.createInvoice({
|
|
127
|
+
orderId: 'vo_123',
|
|
128
|
+
amount: 3.99,
|
|
129
|
+
service: 'Video Generation 5s 720p',
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
// Use template for natural conversation
|
|
133
|
+
console.log(SellerTemplates.quote({
|
|
134
|
+
service: 'Video Generation 5s 720p',
|
|
135
|
+
price: 3.99,
|
|
136
|
+
recipientAddress: agent.walletAddress,
|
|
137
|
+
}));
|
|
138
|
+
|
|
139
|
+
// Step 2: Verify payment
|
|
140
|
+
const result = await agent.verifyPayment(txHash, { expectedAmount: 3.99 });
|
|
141
|
+
|
|
142
|
+
if (result.verified) {
|
|
143
|
+
console.log(SellerTemplates.verificationPassed(result.amount!));
|
|
144
|
+
|
|
145
|
+
// Step 3: Deliver service
|
|
146
|
+
// ... generate video ...
|
|
147
|
+
|
|
148
|
+
// Step 4: Generate receipt
|
|
149
|
+
const receipt = generateReceipt({
|
|
150
|
+
orderId: 'vo_123',
|
|
151
|
+
service: 'Video Generation 5s 720p',
|
|
152
|
+
amount: 3.99,
|
|
153
|
+
chain: 'base',
|
|
154
|
+
txHash: result.tx_hash!,
|
|
155
|
+
payerAddress: result.from!,
|
|
156
|
+
recipientAddress: agent.walletAddress,
|
|
157
|
+
delivery: {
|
|
158
|
+
url: 'https://download.link/video.mp4',
|
|
159
|
+
fileHash: 'sha256:abc123...',
|
|
160
|
+
},
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
console.log(formatReceiptText(receipt));
|
|
164
|
+
}
|
|
165
|
+
```
|
|
166
|
+
|
|
167
|
+
### Conversation Templates
|
|
168
|
+
|
|
169
|
+
Standard templates for natural A2A dialogue:
|
|
170
|
+
|
|
171
|
+
```typescript
|
|
172
|
+
import { SellerTemplates, BuyerTemplates, parseStatusMarker } from 'moltspay';
|
|
173
|
+
|
|
174
|
+
// Seller templates
|
|
175
|
+
SellerTemplates.askPaymentCapability() // "你是否具备链上支付 USDC 的能力?"
|
|
176
|
+
SellerTemplates.guideInstall() // "请安装 moltspay..."
|
|
177
|
+
SellerTemplates.guideFunding() // "A) 直接转账 B) Permit授权"
|
|
178
|
+
SellerTemplates.guidePermit(agentAddr, 10) // "请向 Boss 发送..."
|
|
179
|
+
SellerTemplates.quote({ service, price, recipientAddress })
|
|
180
|
+
SellerTemplates.verificationPassed(amount)
|
|
181
|
+
SellerTemplates.deliver({ downloadUrl, fileHash })
|
|
182
|
+
SellerTemplates.receipt(receipt)
|
|
183
|
+
|
|
184
|
+
// Buyer templates
|
|
185
|
+
BuyerTemplates.requestService('视频生成')
|
|
186
|
+
BuyerTemplates.noCapability() // "我没有钱包"
|
|
187
|
+
BuyerTemplates.walletCreated(address) // "[状态:已具备钱包地址]"
|
|
188
|
+
BuyerTemplates.choosePermit() // "我选择 B"
|
|
189
|
+
BuyerTemplates.permitReceived(10) // "[状态:已具备支付额度 USDC=10]"
|
|
190
|
+
BuyerTemplates.paymentSent(txHash, amount) // "[状态:已发起支付 tx=...]"
|
|
191
|
+
|
|
192
|
+
// Parse status markers from messages
|
|
193
|
+
const status = parseStatusMarker('[状态:已发起支付 tx=0xabc amount=3.99 USDC]');
|
|
194
|
+
// { type: 'payment_sent', data: { txHash: '0xabc', amount: '3.99' } }
|
|
195
|
+
```
|
|
196
|
+
|
|
71
197
|
## API Reference
|
|
72
198
|
|
|
73
199
|
### PaymentAgent
|
package/dist/cli.js
CHANGED
|
@@ -804,12 +804,26 @@ var SecureWallet = class {
|
|
|
804
804
|
}
|
|
805
805
|
};
|
|
806
806
|
|
|
807
|
-
// src/
|
|
807
|
+
// src/wallet/createWallet.ts
|
|
808
808
|
var import_ethers3 = require("ethers");
|
|
809
|
+
var import_path = require("path");
|
|
810
|
+
var DEFAULT_STORAGE_DIR = (0, import_path.join)(process.env.HOME || "~", ".moltspay");
|
|
809
811
|
|
|
810
|
-
// src/
|
|
812
|
+
// src/wallet/PermitWallet.ts
|
|
811
813
|
var import_ethers4 = require("ethers");
|
|
812
|
-
var
|
|
814
|
+
var PERMIT_ABI = [
|
|
815
|
+
...ERC20_ABI,
|
|
816
|
+
"function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s)",
|
|
817
|
+
"function transferFrom(address from, address to, uint256 amount) returns (bool)",
|
|
818
|
+
"function allowance(address owner, address spender) view returns (uint256)"
|
|
819
|
+
];
|
|
820
|
+
|
|
821
|
+
// src/permit/Permit.ts
|
|
822
|
+
var import_ethers5 = require("ethers");
|
|
823
|
+
|
|
824
|
+
// src/verify/index.ts
|
|
825
|
+
var import_ethers6 = require("ethers");
|
|
826
|
+
var TRANSFER_EVENT_TOPIC = import_ethers6.ethers.id("Transfer(address,address,uint256)");
|
|
813
827
|
|
|
814
828
|
// src/cli.ts
|
|
815
829
|
var program = new import_commander.Command();
|
package/dist/cli.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/cli.ts","../src/agent/PaymentAgent.ts","../src/chains/index.ts","../src/wallet/Wallet.ts","../src/audit/AuditLog.ts","../src/wallet/SecureWallet.ts","../src/permit/Permit.ts","../src/verify/index.ts"],"sourcesContent":["#!/usr/bin/env node\n/**\n * Payment Agent CLI\n */\n\nimport { Command } from 'commander';\nimport { PaymentAgent, Wallet, SecureWallet, CHAINS, listChains } from '../src/index.js';\n\nconst program = new Command();\n\nprogram\n .name('payment-agent')\n .description('Blockchain payment infrastructure for AI Agents')\n .version('0.1.0');\n\n// ============ balance ============\nprogram\n .command('balance')\n .description('Get wallet balance')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('-a, --address <address>', 'Wallet address')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: options.address || process.env.PAYMENT_AGENT_WALLET,\n });\n \n const balance = await agent.getBalance();\n console.log(JSON.stringify(balance, null, 2));\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ invoice ============\nprogram\n .command('invoice')\n .description('Generate payment invoice')\n .requiredOption('-o, --order <orderId>', 'Order ID')\n .requiredOption('-a, --amount <amount>', 'Amount in USDC')\n .option('-s, --service <service>', 'Service type', 'payment')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('--json', 'Output raw JSON')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: process.env.PAYMENT_AGENT_WALLET,\n });\n \n const invoice = agent.createInvoice({\n orderId: options.order,\n amount: parseFloat(options.amount),\n service: options.service,\n });\n\n if (options.json) {\n console.log(JSON.stringify(invoice, null, 2));\n } else {\n console.log(agent.formatInvoiceMessage(invoice));\n }\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ verify ============\nprogram\n .command('verify')\n .description('Verify a payment transaction')\n .requiredOption('-t, --tx <txHash>', 'Transaction hash')\n .option('-a, --amount <amount>', 'Expected amount')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: process.env.PAYMENT_AGENT_WALLET,\n });\n \n const result = await agent.verifyPayment(options.tx, {\n expectedAmount: options.amount ? parseFloat(options.amount) : undefined,\n });\n\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.verified ? 0 : 1);\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ transfer ============\nprogram\n .command('transfer')\n .description('Transfer USDC (requires private key)')\n .requiredOption('--to <address>', 'Recipient address')\n .requiredOption('-a, --amount <amount>', 'Amount in USDC')\n .option('-r, --reason <reason>', 'Transfer reason')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('--secure', 'Use secure wallet with limits')\n .action(async (options) => {\n try {\n if (options.secure) {\n const wallet = new SecureWallet({ chain: options.chain });\n const result = await wallet.transfer({\n to: options.to,\n amount: parseFloat(options.amount),\n reason: options.reason,\n });\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.success ? 0 : 1);\n } else {\n const wallet = new Wallet({ chain: options.chain });\n const result = await wallet.transfer(options.to, parseFloat(options.amount));\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.success ? 0 : 1);\n }\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ chains ============\nprogram\n .command('chains')\n .description('List supported chains')\n .action(() => {\n const chains = listChains();\n console.log('Supported chains:');\n for (const name of chains) {\n const config = CHAINS[name];\n console.log(` ${name}: ${config.name} (chainId: ${config.chainId})`);\n }\n });\n\nprogram.parse();\n","/**\n * PaymentAgent - 核心支付代理\n * \n * 功能:\n * - 生成 Invoice(支付请求)\n * - 验证链上支付\n * - 生成钱包深度链接\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n PaymentAgentConfig,\n ChainName,\n ChainConfig,\n Invoice,\n CreateInvoiceParams,\n VerifyResult,\n VerifyOptions,\n WalletBalance,\n} from '../types/index.js';\n\nexport class PaymentAgent {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly walletAddress: string;\n \n private provider: ethers.JsonRpcProvider;\n private usdcContract: ethers.Contract;\n \n static readonly PROTOCOL_VERSION = '1.0';\n\n constructor(config: PaymentAgentConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n this.walletAddress = config.walletAddress || process.env.PAYMENT_AGENT_WALLET || '';\n \n if (!this.walletAddress) {\n throw new Error('walletAddress is required. Set via config or PAYMENT_AGENT_WALLET env var.');\n }\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.provider\n );\n }\n\n /**\n * 生成支付请求(Invoice)\n */\n createInvoice(params: CreateInvoiceParams): Invoice {\n const expiresMinutes = params.expiresMinutes || 30;\n const expiresAt = new Date(Date.now() + expiresMinutes * 60 * 1000).toISOString();\n \n const invoice: Invoice = {\n type: 'payment_request',\n version: PaymentAgent.PROTOCOL_VERSION,\n order_id: params.orderId,\n service: params.service,\n description: params.description || `${params.service} service`,\n amount: params.amount.toFixed(2),\n token: 'USDC',\n chain: this.chain,\n chain_id: this.chainConfig.chainId,\n recipient: this.walletAddress,\n memo: params.orderId,\n expires_at: expiresAt,\n deep_link: this.generateDeepLink(params.amount, params.orderId),\n explorer_url: `${this.chainConfig.explorer}${this.walletAddress}`,\n };\n\n if (params.metadata) {\n invoice.metadata = params.metadata;\n }\n\n return invoice;\n }\n\n /**\n * 生成钱包深度链接(支持 MetaMask 等)\n */\n generateDeepLink(amount: number, memo: string): string {\n const amountWei = Math.floor(amount * 1e6); // USDC 6位小数\n return `https://metamask.app.link/send/${this.chainConfig.usdc}@${this.chainConfig.chainId}/transfer?address=${this.walletAddress}&uint256=${amountWei}`;\n }\n\n /**\n * 验证链上支付\n */\n async verifyPayment(txHash: string, options: VerifyOptions = {}): Promise<VerifyResult> {\n try {\n // 确保 txHash 格式正确\n if (!txHash.startsWith('0x')) {\n txHash = '0x' + txHash;\n }\n\n const receipt = await this.provider.getTransactionReceipt(txHash);\n \n if (!receipt) {\n return { verified: false, error: 'Transaction not found', pending: true };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, error: 'Transaction failed' };\n }\n\n // Transfer 事件签名\n const transferTopic = ethers.id('Transfer(address,address,uint256)');\n const usdcAddress = this.chainConfig.usdc.toLowerCase();\n\n for (const log of receipt.logs) {\n // 检查是否是 USDC 合约的 Transfer 事件\n if (\n log.address.toLowerCase() === usdcAddress &&\n log.topics.length >= 3 &&\n log.topics[0] === transferTopic\n ) {\n // 解析 from, to, amount\n const from = ethers.getAddress('0x' + log.topics[1].slice(-40));\n const to = ethers.getAddress('0x' + log.topics[2].slice(-40));\n const amountWei = BigInt(log.data);\n const amount = Number(amountWei) / 1e6;\n\n // 检查接收地址\n if (to.toLowerCase() !== this.walletAddress.toLowerCase()) {\n continue;\n }\n\n // 检查金额(允许误差)\n const tolerance = options.tolerance ?? 0.01;\n if (options.expectedAmount) {\n const diff = Math.abs(amount - options.expectedAmount);\n if (diff > options.expectedAmount * tolerance) {\n return {\n verified: false,\n error: `Amount mismatch: expected ${options.expectedAmount}, got ${amount}`,\n };\n }\n }\n\n const currentBlock = await this.provider.getBlockNumber();\n \n return {\n verified: true,\n tx_hash: txHash,\n amount: amount.toFixed(2),\n token: 'USDC',\n from,\n to,\n block_number: receipt.blockNumber,\n confirmations: currentBlock - receipt.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${txHash}`,\n };\n }\n }\n\n return { verified: false, error: 'No USDC transfer to recipient found in transaction' };\n } catch (error) {\n return { verified: false, error: (error as Error).message };\n }\n }\n\n /**\n * 扫描最近转账(按金额匹配)\n */\n async scanRecentTransfers(expectedAmount: number, timeoutMinutes: number = 30): Promise<VerifyResult> {\n try {\n const currentBlock = await this.provider.getBlockNumber();\n const blocksPerMinute = Math.ceil(60 / this.chainConfig.avgBlockTime);\n const fromBlock = currentBlock - (timeoutMinutes * blocksPerMinute);\n\n // 使用 getLogs 扫描 Transfer 事件\n const transferTopic = ethers.id('Transfer(address,address,uint256)');\n const recipientTopic = ethers.zeroPadValue(this.walletAddress, 32);\n\n const logs = await this.provider.getLogs({\n address: this.chainConfig.usdc,\n topics: [transferTopic, null, recipientTopic],\n fromBlock,\n toBlock: 'latest',\n });\n\n for (const log of logs) {\n const amountWei = BigInt(log.data);\n const amount = Number(amountWei) / 1e6;\n\n // 按金额匹配\n if (Math.abs(amount - expectedAmount) < 0.01) {\n const from = ethers.getAddress('0x' + log.topics[1].slice(-40));\n \n return {\n verified: true,\n tx_hash: log.transactionHash,\n amount: amount.toFixed(2),\n token: 'USDC',\n from,\n to: this.walletAddress,\n block_number: log.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${log.transactionHash}`,\n };\n }\n }\n\n return { verified: false, error: 'No matching payment found' };\n } catch (error) {\n return { verified: false, error: (error as Error).message };\n }\n }\n\n /**\n * 获取钱包余额\n */\n async getBalance(address?: string): Promise<WalletBalance> {\n const addr = address || this.walletAddress;\n \n const [ethBalance, usdcBalance] = await Promise.all([\n this.provider.getBalance(addr),\n this.usdcContract.balanceOf(addr),\n ]);\n\n return {\n address: addr,\n eth: ethers.formatEther(ethBalance),\n usdc: (Number(usdcBalance) / 1e6).toFixed(2),\n chain: this.chain,\n };\n }\n\n /**\n * 格式化 Invoice 为人类可读消息\n */\n formatInvoiceMessage(invoice: Invoice, includeJson: boolean = true): string {\n let msg = `🎬 **Payment Request**\n\n**Service:** ${invoice.service}\n**Price:** ${invoice.amount} USDC (${this.chainConfig.name})\n\n**💳 Payment Options:**\n\n1️⃣ **Direct Transfer:**\n Send exactly \\`${invoice.amount} USDC\\` to:\n \\`${invoice.recipient}\\`\n (Network: ${this.chainConfig.name})\n\n2️⃣ **One-Click Pay (MetaMask):**\n ${invoice.deep_link}\n\n⏱️ Expires: ${invoice.expires_at}`;\n\n if (includeJson) {\n msg += `\n\n3️⃣ **For AI Agents:**\n\\`\\`\\`json\n${JSON.stringify(invoice, null, 2)}\n\\`\\`\\``;\n }\n\n msg += `\n\nAfter payment, reply with your tx hash:\n\\`paid: 0x...\\``;\n\n return msg;\n }\n}\n","/**\n * 区块链配置\n */\n\nimport type { ChainConfig, ChainName } from '../types/index.js';\n\nexport const CHAINS: Record<ChainName, ChainConfig> = {\n // ============ 主网 ============\n base: {\n name: 'Base',\n chainId: 8453,\n rpc: 'https://mainnet.base.org',\n usdc: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913',\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-rpc.com',\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 usdc: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48',\n explorer: 'https://etherscan.io/address/',\n explorerTx: 'https://etherscan.io/tx/',\n avgBlockTime: 12,\n },\n\n // ============ 测试网 ============\n base_sepolia: {\n name: 'Base Sepolia',\n chainId: 84532,\n rpc: 'https://sepolia.base.org',\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 usdc: '0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238',\n explorer: 'https://sepolia.etherscan.io/address/',\n explorerTx: 'https://sepolia.etherscan.io/tx/',\n avgBlockTime: 12,\n },\n};\n\n/**\n * 获取链配置\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 * 列出所有支持的链\n */\nexport function listChains(): ChainName[] {\n return Object.keys(CHAINS) as ChainName[];\n}\n\n/**\n * 根据 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(最小化,仅包含需要的方法)\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 };\n","/**\n * Wallet - 基础托管钱包\n * \n * 功能:\n * - 查询余额\n * - 发送 USDC 转账\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n ChainName,\n ChainConfig,\n WalletBalance,\n TransferResult,\n} from '../types/index.js';\n\nexport interface WalletConfig {\n chain?: ChainName;\n privateKey?: string;\n rpcUrl?: string;\n}\n\nexport class Wallet {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly address: string;\n \n private wallet: ethers.Wallet;\n private provider: ethers.JsonRpcProvider;\n private usdcContract: ethers.Contract;\n\n constructor(config: WalletConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n \n const privateKey = config.privateKey || process.env.PAYMENT_AGENT_PRIVATE_KEY;\n if (!privateKey) {\n throw new Error('privateKey is required. Set via config or PAYMENT_AGENT_PRIVATE_KEY env var.');\n }\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n this.wallet = new ethers.Wallet(privateKey, this.provider);\n this.address = this.wallet.address;\n \n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.wallet\n );\n }\n\n /**\n * 获取钱包余额\n */\n async getBalance(): Promise<WalletBalance> {\n const [ethBalance, usdcBalance] = await Promise.all([\n this.provider.getBalance(this.address),\n this.usdcContract.balanceOf(this.address),\n ]);\n\n return {\n address: this.address,\n eth: ethers.formatEther(ethBalance),\n usdc: (Number(usdcBalance) / 1e6).toFixed(2),\n chain: this.chain,\n };\n }\n\n /**\n * 发送 USDC 转账\n */\n async transfer(to: string, amount: number): Promise<TransferResult> {\n try {\n // 验证地址\n to = ethers.getAddress(to);\n \n // 转换金额(USDC 6位小数)\n const amountWei = BigInt(Math.floor(amount * 1e6));\n\n // 检查余额\n const balance = await this.usdcContract.balanceOf(this.address);\n if (BigInt(balance) < amountWei) {\n return {\n success: false,\n error: `Insufficient USDC balance: ${Number(balance) / 1e6} < ${amount}`,\n };\n }\n\n // 发送交易\n const tx = await this.usdcContract.transfer(to, amountWei);\n const receipt = await tx.wait();\n\n if (receipt.status === 1) {\n return {\n success: true,\n tx_hash: tx.hash,\n from: this.address,\n to,\n amount,\n gas_used: Number(receipt.gasUsed),\n block_number: receipt.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${tx.hash}`,\n };\n } else {\n return {\n success: false,\n tx_hash: tx.hash,\n error: 'Transaction reverted',\n };\n }\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 获取 ETH 余额\n */\n async getEthBalance(): Promise<string> {\n const balance = await this.provider.getBalance(this.address);\n return ethers.formatEther(balance);\n }\n\n /**\n * 获取 USDC 余额\n */\n async getUsdcBalance(): Promise<string> {\n const balance = await this.usdcContract.balanceOf(this.address);\n return (Number(balance) / 1e6).toFixed(2);\n }\n}\n","/**\n * AuditLog - 不可篡改审计日志\n * \n * 特点:\n * - 链式哈希,任何修改都会破坏链条\n * - 按日期分文件存储\n * - JSONL 格式便于追加和解析\n */\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as crypto from 'crypto';\nimport type { AuditEntry, AuditAction } from '../types/index.js';\n\nexport interface LogParams {\n action: AuditAction;\n request_id: string;\n from?: string;\n to?: string;\n amount?: number;\n tx_hash?: string;\n reason?: string;\n requester?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport class AuditLog {\n private basePath: string;\n private lastHash: string = '0000000000000000';\n\n constructor(basePath?: string) {\n this.basePath = basePath || path.join(process.cwd(), 'data', 'audit');\n this.ensureDir();\n this.loadLastHash();\n }\n\n /**\n * 记录审计日志\n */\n async log(params: LogParams): Promise<AuditEntry> {\n const now = new Date();\n \n const entry: AuditEntry = {\n timestamp: now.getTime() / 1000,\n datetime: now.toISOString(),\n action: params.action,\n request_id: params.request_id,\n from: params.from,\n to: params.to,\n amount: params.amount,\n tx_hash: params.tx_hash,\n reason: params.reason,\n requester: params.requester,\n prev_hash: this.lastHash,\n hash: '', // 计算后填充\n metadata: params.metadata,\n };\n\n // 计算哈希(不包含 hash 字段本身)\n entry.hash = this.calculateHash(entry);\n this.lastHash = entry.hash;\n\n // 写入文件\n const filePath = this.getFilePath(now);\n const line = JSON.stringify(entry) + '\\n';\n fs.appendFileSync(filePath, line, 'utf-8');\n\n return entry;\n }\n\n /**\n * 读取指定日期的日志\n */\n read(date?: Date): AuditEntry[] {\n const filePath = this.getFilePath(date || new Date());\n \n if (!fs.existsSync(filePath)) {\n return [];\n }\n\n const content = fs.readFileSync(filePath, 'utf-8');\n const lines = content.trim().split('\\n').filter(Boolean);\n \n return lines.map(line => JSON.parse(line) as AuditEntry);\n }\n\n /**\n * 验证日志完整性\n */\n verify(date?: Date): { valid: boolean; errors: string[] } {\n const entries = this.read(date);\n const errors: string[] = [];\n\n for (let i = 0; i < entries.length; i++) {\n const entry = entries[i];\n \n // 验证哈希\n const expectedHash = this.calculateHash(entry);\n if (entry.hash !== expectedHash) {\n errors.push(`Entry ${i}: hash mismatch (expected ${expectedHash}, got ${entry.hash})`);\n }\n\n // 验证链接\n if (i > 0 && entry.prev_hash !== entries[i - 1].hash) {\n errors.push(`Entry ${i}: prev_hash mismatch`);\n }\n }\n\n return { valid: errors.length === 0, errors };\n }\n\n /**\n * 搜索日志\n */\n search(filter: Partial<{\n action: AuditAction;\n request_id: string;\n from: string;\n to: string;\n startDate: Date;\n endDate: Date;\n }>): AuditEntry[] {\n const results: AuditEntry[] = [];\n const startDate = filter.startDate || new Date(Date.now() - 30 * 24 * 60 * 60 * 1000);\n const endDate = filter.endDate || new Date();\n\n // 遍历日期范围\n const current = new Date(startDate);\n while (current <= endDate) {\n const entries = this.read(current);\n \n for (const entry of entries) {\n let match = true;\n\n if (filter.action && entry.action !== filter.action) match = false;\n if (filter.request_id && entry.request_id !== filter.request_id) match = false;\n if (filter.from && entry.from?.toLowerCase() !== filter.from.toLowerCase()) match = false;\n if (filter.to && entry.to?.toLowerCase() !== filter.to.toLowerCase()) match = false;\n\n if (match) {\n results.push(entry);\n }\n }\n\n current.setDate(current.getDate() + 1);\n }\n\n return results;\n }\n\n /**\n * 获取日志文件路径\n */\n private getFilePath(date: Date): string {\n const dateStr = date.toISOString().slice(0, 10);\n return path.join(this.basePath, `audit_${dateStr}.jsonl`);\n }\n\n /**\n * 计算条目哈希\n */\n private calculateHash(entry: AuditEntry): string {\n const data = {\n timestamp: entry.timestamp,\n action: entry.action,\n request_id: entry.request_id,\n from: entry.from,\n to: entry.to,\n amount: entry.amount,\n tx_hash: entry.tx_hash,\n prev_hash: entry.prev_hash,\n };\n \n const str = JSON.stringify(data);\n return crypto.createHash('sha256').update(str).digest('hex').slice(0, 16);\n }\n\n /**\n * 加载最后一条日志的哈希\n */\n private loadLastHash(): void {\n const today = new Date();\n \n // 检查今天和昨天的日志\n for (let i = 0; i < 2; i++) {\n const date = new Date(today);\n date.setDate(date.getDate() - i);\n \n const entries = this.read(date);\n if (entries.length > 0) {\n this.lastHash = entries[entries.length - 1].hash;\n return;\n }\n }\n }\n\n /**\n * 确保目录存在\n */\n private ensureDir(): void {\n if (!fs.existsSync(this.basePath)) {\n fs.mkdirSync(this.basePath, { recursive: true });\n }\n }\n}\n","/**\n * SecureWallet - 安全托管钱包\n * \n * 在基础 Wallet 之上增加:\n * - 单笔限额控制\n * - 日限额控制\n * - 白名单机制\n * - 审计日志\n * - 超限审批队列\n */\n\nimport { Wallet, type WalletConfig } from './Wallet.js';\nimport { AuditLog } from '../audit/AuditLog.js';\nimport type {\n SecurityLimits,\n SecureWalletConfig,\n TransferResult,\n TransferParams,\n PendingTransfer,\n} from '../types/index.js';\n\nconst DEFAULT_LIMITS: SecurityLimits = {\n singleMax: 100, // 单笔最大 $100\n dailyMax: 1000, // 日最大 $1000\n requireWhitelist: true,\n};\n\nexport class SecureWallet {\n private wallet: Wallet;\n private limits: SecurityLimits;\n private whitelist: Set<string>;\n private auditLog: AuditLog;\n private dailyTotal: number = 0;\n private dailyDate: string = '';\n private pendingTransfers: Map<string, PendingTransfer> = new Map();\n\n constructor(config: SecureWalletConfig = {}) {\n this.wallet = new Wallet({\n chain: config.chain,\n privateKey: config.privateKey,\n });\n\n this.limits = { ...DEFAULT_LIMITS, ...config.limits };\n this.whitelist = new Set((config.whitelist || []).map(a => a.toLowerCase()));\n this.auditLog = new AuditLog(config.auditPath);\n }\n\n /**\n * 获取钱包地址\n */\n get address(): string {\n return this.wallet.address;\n }\n\n /**\n * 获取余额\n */\n async getBalance() {\n return this.wallet.getBalance();\n }\n\n /**\n * 安全转账(带限额和白名单检查)\n * \n * 支持两种调用方式:\n * - transfer({ to, amount, reason?, requester? })\n * - transfer(to, amount)\n */\n async transfer(paramsOrTo: TransferParams | string, amountArg?: number | string): Promise<TransferResult> {\n // 支持两种调用方式\n let params: TransferParams;\n if (typeof paramsOrTo === 'string') {\n params = { \n to: paramsOrTo, \n amount: typeof amountArg === 'string' ? parseFloat(amountArg) : (amountArg || 0)\n };\n } else {\n params = paramsOrTo;\n }\n \n const { to, amount, reason, requester } = params;\n const toAddress = to.toLowerCase();\n const requestId = `tr_${Date.now().toString(36)}${Math.random().toString(36).slice(2, 8)}`;\n\n // 记录请求\n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n from: this.wallet.address,\n to,\n amount,\n reason,\n requester,\n });\n\n // 1. 白名单检查\n if (this.limits.requireWhitelist && !this.whitelist.has(toAddress)) {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: 'Address not in whitelist' },\n });\n return { success: false, error: `Address not in whitelist: ${to}` };\n }\n\n // 2. 单笔限额检查\n if (amount > this.limits.singleMax) {\n // 加入审批队列\n const pending: PendingTransfer = {\n id: requestId,\n to,\n amount,\n reason,\n requester,\n created_at: new Date().toISOString(),\n status: 'pending',\n };\n this.pendingTransfers.set(requestId, pending);\n \n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n metadata: { pending: true, reason: 'Exceeds single limit' },\n });\n \n return {\n success: false,\n error: `Amount ${amount} exceeds single limit ${this.limits.singleMax}. Pending approval: ${requestId}`,\n };\n }\n\n // 3. 日限额检查\n this.updateDailyTotal();\n if (this.dailyTotal + amount > this.limits.dailyMax) {\n const pending: PendingTransfer = {\n id: requestId,\n to,\n amount,\n reason,\n requester,\n created_at: new Date().toISOString(),\n status: 'pending',\n };\n this.pendingTransfers.set(requestId, pending);\n \n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n metadata: { pending: true, reason: 'Exceeds daily limit' },\n });\n \n return {\n success: false,\n error: `Daily limit would be exceeded (${this.dailyTotal} + ${amount} > ${this.limits.dailyMax}). Pending approval: ${requestId}`,\n };\n }\n\n // 4. 执行转账\n const result = await this.wallet.transfer(to, amount);\n\n // 5. 记录结果\n if (result.success) {\n this.dailyTotal += amount;\n await this.auditLog.log({\n action: 'transfer_executed',\n request_id: requestId,\n from: this.wallet.address,\n to,\n amount,\n tx_hash: result.tx_hash,\n reason,\n requester,\n });\n } else {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: result.error },\n });\n }\n\n return result;\n }\n\n /**\n * 审批待处理转账\n */\n async approve(requestId: string, approver: string): Promise<TransferResult> {\n const pending = this.pendingTransfers.get(requestId);\n if (!pending) {\n return { success: false, error: `Pending transfer not found: ${requestId}` };\n }\n\n if (pending.status !== 'pending') {\n return { success: false, error: `Transfer already ${pending.status}` };\n }\n\n await this.auditLog.log({\n action: 'transfer_approved',\n request_id: requestId,\n metadata: { approver },\n });\n\n pending.status = 'approved';\n\n // 执行转账(跳过限额检查)\n const result = await this.wallet.transfer(pending.to, pending.amount);\n\n if (result.success) {\n pending.status = 'executed';\n this.dailyTotal += pending.amount;\n \n await this.auditLog.log({\n action: 'transfer_executed',\n request_id: requestId,\n from: this.wallet.address,\n to: pending.to,\n amount: pending.amount,\n tx_hash: result.tx_hash,\n reason: pending.reason,\n requester: pending.requester,\n metadata: { approved_by: approver },\n });\n } else {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: result.error },\n });\n }\n\n return result;\n }\n\n /**\n * 拒绝待处理转账\n */\n async reject(requestId: string, rejecter: string, reason?: string): Promise<void> {\n const pending = this.pendingTransfers.get(requestId);\n if (!pending) {\n throw new Error(`Pending transfer not found: ${requestId}`);\n }\n\n pending.status = 'rejected';\n\n await this.auditLog.log({\n action: 'transfer_rejected',\n request_id: requestId,\n metadata: { rejected_by: rejecter, reason },\n });\n }\n\n /**\n * 添加白名单地址\n */\n async addToWhitelist(address: string, addedBy: string): Promise<void> {\n const addr = address.toLowerCase();\n this.whitelist.add(addr);\n\n await this.auditLog.log({\n action: 'whitelist_add',\n request_id: `wl_${Date.now()}`,\n to: address,\n metadata: { added_by: addedBy },\n });\n }\n\n /**\n * 移除白名单地址\n */\n async removeFromWhitelist(address: string, removedBy: string): Promise<void> {\n const addr = address.toLowerCase();\n this.whitelist.delete(addr);\n\n await this.auditLog.log({\n action: 'whitelist_remove',\n request_id: `wl_${Date.now()}`,\n to: address,\n metadata: { removed_by: removedBy },\n });\n }\n\n /**\n * 检查地址是否在白名单\n */\n isWhitelisted(address: string): boolean {\n return this.whitelist.has(address.toLowerCase());\n }\n\n /**\n * 获取待处理转账列表\n */\n getPendingTransfers(): PendingTransfer[] {\n return Array.from(this.pendingTransfers.values())\n .filter(p => p.status === 'pending');\n }\n\n /**\n * 获取当前限额配置\n */\n getLimits(): SecurityLimits {\n return { ...this.limits };\n }\n\n /**\n * 获取今日已用额度\n */\n getDailyUsed(): number {\n this.updateDailyTotal();\n return this.dailyTotal;\n }\n\n /**\n * 更新日限额计数器\n */\n private updateDailyTotal(): void {\n const today = new Date().toISOString().slice(0, 10);\n if (this.dailyDate !== today) {\n this.dailyDate = today;\n this.dailyTotal = 0;\n }\n }\n}\n","/**\n * PermitPayment - EIP-2612 无 Gas 预授权\n * \n * 让用户通过签名授权,服务方代付 Gas 执行 transferFrom\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n ChainName,\n ChainConfig,\n PermitRequest,\n PermitSignature,\n PermitExecuteResult,\n EIP712TypedData,\n} from '../types/index.js';\n\nexport interface PermitConfig {\n chain?: ChainName;\n privateKey?: string;\n spenderAddress?: string;\n rpcUrl?: string;\n}\n\nexport class PermitPayment {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly spenderAddress: string;\n \n private provider: ethers.JsonRpcProvider;\n private wallet?: ethers.Wallet;\n private usdcContract: ethers.Contract;\n\n constructor(config: PermitConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n this.spenderAddress = config.spenderAddress || process.env.PAYMENT_AGENT_WALLET || '';\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n\n const privateKey = config.privateKey || process.env.PAYMENT_AGENT_PRIVATE_KEY;\n if (privateKey) {\n this.wallet = new ethers.Wallet(privateKey, this.provider);\n this.spenderAddress = this.wallet.address;\n }\n\n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.wallet || this.provider\n );\n }\n\n /**\n * 获取用户当前 nonce\n */\n async getNonce(owner: string): Promise<number> {\n return Number(await this.usdcContract.nonces(owner));\n }\n\n /**\n * 生成 EIP-712 签名请求(发给前端/用户钱包)\n */\n async createPermitRequest(\n owner: string,\n amount: number,\n orderId: string,\n deadlineMinutes: number = 30\n ): Promise<PermitRequest> {\n const nonce = await this.getNonce(owner);\n const deadline = Math.floor(Date.now() / 1000) + deadlineMinutes * 60;\n const value = BigInt(Math.floor(amount * 1e6)).toString();\n\n // USDC 的 EIP-712 domain(不同链可能不同)\n const domain = {\n name: 'USD Coin',\n version: '2',\n chainId: this.chainConfig.chainId,\n verifyingContract: this.chainConfig.usdc,\n };\n\n const types = {\n EIP712Domain: [\n { name: 'name', type: 'string' },\n { name: 'version', type: 'string' },\n { name: 'chainId', type: 'uint256' },\n { name: 'verifyingContract', type: 'address' },\n ],\n Permit: [\n { name: 'owner', type: 'address' },\n { name: 'spender', type: 'address' },\n { name: 'value', type: 'uint256' },\n { name: 'nonce', type: 'uint256' },\n { name: 'deadline', type: 'uint256' },\n ],\n };\n\n const message = {\n owner,\n spender: this.spenderAddress,\n value,\n nonce,\n deadline,\n };\n\n const typedData: EIP712TypedData = {\n types,\n primaryType: 'Permit',\n domain,\n message,\n };\n\n return {\n type: 'permit_request',\n version: '1.0',\n order_id: orderId,\n typed_data: typedData,\n };\n }\n\n /**\n * 执行 permit + transferFrom\n * \n * @param owner 用户地址\n * @param amount 金额\n * @param signature 用户签名 {v, r, s, deadline}\n */\n async executePermitAndTransfer(\n owner: string,\n amount: number,\n signature: PermitSignature\n ): Promise<PermitExecuteResult> {\n if (!this.wallet) {\n return { success: false, error: 'Wallet not configured. Private key required.' };\n }\n\n try {\n const value = BigInt(Math.floor(amount * 1e6));\n\n // 1. 调用 permit\n const permitTx = await this.usdcContract.permit(\n owner,\n this.spenderAddress,\n value,\n signature.deadline,\n signature.v,\n signature.r,\n signature.s\n );\n await permitTx.wait();\n\n // 2. 调用 transferFrom\n const transferTx = await this.usdcContract.transferFrom(owner, this.spenderAddress, value);\n const receipt = await transferTx.wait();\n\n return {\n success: receipt.status === 1,\n tx_hash: transferTx.hash,\n };\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 仅执行 permit(不 transfer)\n */\n async executePermit(\n owner: string,\n amount: number,\n signature: PermitSignature\n ): Promise<PermitExecuteResult> {\n if (!this.wallet) {\n return { success: false, error: 'Wallet not configured. Private key required.' };\n }\n\n try {\n const value = BigInt(Math.floor(amount * 1e6));\n\n const tx = await this.usdcContract.permit(\n owner,\n this.spenderAddress,\n value,\n signature.deadline,\n signature.v,\n signature.r,\n signature.s\n );\n const receipt = await tx.wait();\n\n return {\n success: receipt.status === 1,\n tx_hash: tx.hash,\n };\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 格式化 Permit 请求为用户消息\n */\n formatPermitMessage(request: PermitRequest): string {\n const { typed_data } = request;\n const { message } = typed_data;\n\n return `🔐 **签名授权请求**\n\n授权 \\`${(Number(message.value) / 1e6).toFixed(2)} USDC\\` 给服务方\n\n**签名信息:**\n- Owner: \\`${message.owner}\\`\n- Spender: \\`${message.spender}\\`\n- Amount: ${(Number(message.value) / 1e6).toFixed(2)} USDC\n- Deadline: ${new Date(message.deadline * 1000).toISOString()}\n\n请在钱包中签名此请求(不消耗 Gas)。\n\n\\`\\`\\`json\n${JSON.stringify(typed_data, null, 2)}\n\\`\\`\\``;\n }\n}\n","/**\n * 链上支付验证模块\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, getChainById, type ChainConfig, type ChainName } from '../chains';\n\n// ERC20 Transfer 事件签名\nconst TRANSFER_EVENT_TOPIC = ethers.id('Transfer(address,address,uint256)');\n\nexport interface VerifyPaymentParams {\n txHash: string;\n expectedAmount: number;\n expectedTo?: string;\n chain?: string | number;\n}\n\nexport interface VerifyPaymentResult {\n verified: boolean;\n amount?: number;\n from?: string;\n to?: string;\n txHash?: string;\n blockNumber?: number;\n error?: string;\n}\n\n/**\n * 验证链上支付\n */\nexport async function verifyPayment(params: VerifyPaymentParams): Promise<VerifyPaymentResult> {\n const { txHash, expectedAmount, expectedTo } = params;\n \n // 获取链配置\n let chain: ChainConfig | undefined;\n try {\n if (typeof params.chain === 'number') {\n chain = getChainById(params.chain);\n } else {\n chain = getChain((params.chain || 'base') as ChainName);\n }\n if (!chain) {\n return { verified: false, error: `不支持的链: ${params.chain}` };\n }\n } catch (e) {\n return { verified: false, error: `不支持的链: ${params.chain}` };\n }\n\n try {\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n \n // 获取交易回执\n const receipt = await provider.getTransactionReceipt(txHash);\n \n if (!receipt) {\n return { verified: false, error: '交易未找到或未确认' };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, error: '交易失败' };\n }\n\n // 解析 Transfer 事件\n const usdcAddress = chain.usdc?.toLowerCase();\n if (!usdcAddress) {\n return { verified: false, error: `链 ${chain.name} 未配置USDC地址` };\n }\n\n for (const log of receipt.logs) {\n // 检查是否是 USDC 合约\n if (log.address.toLowerCase() !== usdcAddress) {\n continue;\n }\n\n // 检查是否是 Transfer 事件\n if (log.topics.length < 3 || log.topics[0] !== TRANSFER_EVENT_TOPIC) {\n continue;\n }\n\n // 解析 Transfer 事件参数\n const from = '0x' + log.topics[1].slice(-40);\n const to = '0x' + log.topics[2].slice(-40);\n const amountRaw = BigInt(log.data);\n const amount = Number(amountRaw) / 1e6; // USDC 6位小数\n\n // 验证收款地址\n if (expectedTo && to.toLowerCase() !== expectedTo.toLowerCase()) {\n continue;\n }\n\n // 验证金额\n if (amount < expectedAmount) {\n return {\n verified: false,\n error: `金额不足: 收到 ${amount} USDC, 需要 ${expectedAmount} USDC`,\n amount,\n from,\n to,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n }\n\n // 验证成功\n return {\n verified: true,\n amount,\n from,\n to,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n }\n\n return { verified: false, error: '未找到USDC转账记录' };\n\n } catch (e: any) {\n return { verified: false, error: e.message || String(e) };\n }\n}\n\n/**\n * 获取交易状态\n */\nexport async function getTransactionStatus(\n txHash: string,\n chain: string | number = 'base'\n): Promise<{\n status: 'pending' | 'confirmed' | 'failed' | 'not_found';\n blockNumber?: number;\n confirmations?: number;\n}> {\n let chainConfig: ChainConfig | undefined;\n try {\n chainConfig = typeof chain === 'number' ? getChainById(chain) : getChain(chain as ChainName);\n if (!chainConfig) return { status: 'not_found' };\n } catch {\n return { status: 'not_found' };\n }\n\n try {\n const provider = new ethers.JsonRpcProvider(chainConfig.rpc);\n const receipt = await provider.getTransactionReceipt(txHash);\n\n if (!receipt) {\n // 检查是否在 pending 池中\n const tx = await provider.getTransaction(txHash);\n if (tx) {\n return { status: 'pending' };\n }\n return { status: 'not_found' };\n }\n\n const currentBlock = await provider.getBlockNumber();\n const confirmations = currentBlock - receipt.blockNumber;\n\n if (receipt.status === 1) {\n return {\n status: 'confirmed',\n blockNumber: receipt.blockNumber,\n confirmations,\n };\n } else {\n return {\n status: 'failed',\n blockNumber: receipt.blockNumber,\n };\n }\n } catch {\n return { status: 'not_found' };\n }\n}\n\n/**\n * 等待交易确认\n */\nexport async function waitForTransaction(\n txHash: string,\n chain: string | number = 'base',\n confirmations = 1,\n timeoutMs = 60000\n): Promise<VerifyPaymentResult & { confirmed: boolean }> {\n let chainConfig: ChainConfig | undefined;\n try {\n chainConfig = typeof chain === 'number' ? getChainById(chain) : getChain(chain as ChainName);\n if (!chainConfig) {\n return { verified: false, confirmed: false, error: `不支持的链: ${chain}` };\n }\n } catch (e) {\n return { verified: false, confirmed: false, error: `不支持的链: ${chain}` };\n }\n\n const provider = new ethers.JsonRpcProvider(chainConfig.rpc);\n \n try {\n const receipt = await provider.waitForTransaction(txHash, confirmations, timeoutMs);\n \n if (!receipt) {\n return { verified: false, confirmed: false, error: '等待超时' };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, confirmed: true, error: '交易失败' };\n }\n\n return {\n verified: true,\n confirmed: true,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n } catch (e: any) {\n return { verified: false, confirmed: false, error: e.message || String(e) };\n }\n}\n"],"mappings":"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","names":["import_ethers","import_ethers","import_ethers"]}
|
|
1
|
+
{"version":3,"sources":["../src/cli.ts","../src/agent/PaymentAgent.ts","../src/chains/index.ts","../src/wallet/Wallet.ts","../src/audit/AuditLog.ts","../src/wallet/SecureWallet.ts","../src/wallet/createWallet.ts","../src/wallet/PermitWallet.ts","../src/permit/Permit.ts","../src/verify/index.ts"],"sourcesContent":["#!/usr/bin/env node\n/**\n * Payment Agent CLI\n */\n\nimport { Command } from 'commander';\nimport { PaymentAgent, Wallet, SecureWallet, CHAINS, listChains } from '../src/index.js';\n\nconst program = new Command();\n\nprogram\n .name('payment-agent')\n .description('Blockchain payment infrastructure for AI Agents')\n .version('0.1.0');\n\n// ============ balance ============\nprogram\n .command('balance')\n .description('Get wallet balance')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('-a, --address <address>', 'Wallet address')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: options.address || process.env.PAYMENT_AGENT_WALLET,\n });\n \n const balance = await agent.getBalance();\n console.log(JSON.stringify(balance, null, 2));\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ invoice ============\nprogram\n .command('invoice')\n .description('Generate payment invoice')\n .requiredOption('-o, --order <orderId>', 'Order ID')\n .requiredOption('-a, --amount <amount>', 'Amount in USDC')\n .option('-s, --service <service>', 'Service type', 'payment')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('--json', 'Output raw JSON')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: process.env.PAYMENT_AGENT_WALLET,\n });\n \n const invoice = agent.createInvoice({\n orderId: options.order,\n amount: parseFloat(options.amount),\n service: options.service,\n });\n\n if (options.json) {\n console.log(JSON.stringify(invoice, null, 2));\n } else {\n console.log(agent.formatInvoiceMessage(invoice));\n }\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ verify ============\nprogram\n .command('verify')\n .description('Verify a payment transaction')\n .requiredOption('-t, --tx <txHash>', 'Transaction hash')\n .option('-a, --amount <amount>', 'Expected amount')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .action(async (options) => {\n try {\n const agent = new PaymentAgent({\n chain: options.chain,\n walletAddress: process.env.PAYMENT_AGENT_WALLET,\n });\n \n const result = await agent.verifyPayment(options.tx, {\n expectedAmount: options.amount ? parseFloat(options.amount) : undefined,\n });\n\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.verified ? 0 : 1);\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ transfer ============\nprogram\n .command('transfer')\n .description('Transfer USDC (requires private key)')\n .requiredOption('--to <address>', 'Recipient address')\n .requiredOption('-a, --amount <amount>', 'Amount in USDC')\n .option('-r, --reason <reason>', 'Transfer reason')\n .option('-c, --chain <chain>', 'Chain name', 'base_sepolia')\n .option('--secure', 'Use secure wallet with limits')\n .action(async (options) => {\n try {\n if (options.secure) {\n const wallet = new SecureWallet({ chain: options.chain });\n const result = await wallet.transfer({\n to: options.to,\n amount: parseFloat(options.amount),\n reason: options.reason,\n });\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.success ? 0 : 1);\n } else {\n const wallet = new Wallet({ chain: options.chain });\n const result = await wallet.transfer(options.to, parseFloat(options.amount));\n console.log(JSON.stringify(result, null, 2));\n process.exit(result.success ? 0 : 1);\n }\n } catch (error) {\n console.error('Error:', (error as Error).message);\n process.exit(1);\n }\n });\n\n// ============ chains ============\nprogram\n .command('chains')\n .description('List supported chains')\n .action(() => {\n const chains = listChains();\n console.log('Supported chains:');\n for (const name of chains) {\n const config = CHAINS[name];\n console.log(` ${name}: ${config.name} (chainId: ${config.chainId})`);\n }\n });\n\nprogram.parse();\n","/**\n * PaymentAgent - 核心支付代理\n * \n * 功能:\n * - 生成 Invoice(支付请求)\n * - 验证链上支付\n * - 生成钱包深度链接\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n PaymentAgentConfig,\n ChainName,\n ChainConfig,\n Invoice,\n CreateInvoiceParams,\n VerifyResult,\n VerifyOptions,\n WalletBalance,\n} from '../types/index.js';\n\nexport class PaymentAgent {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly walletAddress: string;\n \n private provider: ethers.JsonRpcProvider;\n private usdcContract: ethers.Contract;\n \n static readonly PROTOCOL_VERSION = '1.0';\n\n constructor(config: PaymentAgentConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n this.walletAddress = config.walletAddress || process.env.PAYMENT_AGENT_WALLET || '';\n \n if (!this.walletAddress) {\n throw new Error('walletAddress is required. Set via config or PAYMENT_AGENT_WALLET env var.');\n }\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.provider\n );\n }\n\n /**\n * 生成支付请求(Invoice)\n */\n createInvoice(params: CreateInvoiceParams): Invoice {\n const expiresMinutes = params.expiresMinutes || 30;\n const expiresAt = new Date(Date.now() + expiresMinutes * 60 * 1000).toISOString();\n \n const invoice: Invoice = {\n type: 'payment_request',\n version: PaymentAgent.PROTOCOL_VERSION,\n order_id: params.orderId,\n service: params.service,\n description: params.description || `${params.service} service`,\n amount: params.amount.toFixed(2),\n token: 'USDC',\n chain: this.chain,\n chain_id: this.chainConfig.chainId,\n recipient: this.walletAddress,\n memo: params.orderId,\n expires_at: expiresAt,\n deep_link: this.generateDeepLink(params.amount, params.orderId),\n explorer_url: `${this.chainConfig.explorer}${this.walletAddress}`,\n };\n\n if (params.metadata) {\n invoice.metadata = params.metadata;\n }\n\n return invoice;\n }\n\n /**\n * 生成钱包深度链接(支持 MetaMask 等)\n */\n generateDeepLink(amount: number, memo: string): string {\n const amountWei = Math.floor(amount * 1e6); // USDC 6位小数\n return `https://metamask.app.link/send/${this.chainConfig.usdc}@${this.chainConfig.chainId}/transfer?address=${this.walletAddress}&uint256=${amountWei}`;\n }\n\n /**\n * 验证链上支付\n */\n async verifyPayment(txHash: string, options: VerifyOptions = {}): Promise<VerifyResult> {\n try {\n // 确保 txHash 格式正确\n if (!txHash.startsWith('0x')) {\n txHash = '0x' + txHash;\n }\n\n const receipt = await this.provider.getTransactionReceipt(txHash);\n \n if (!receipt) {\n return { verified: false, error: 'Transaction not found', pending: true };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, error: 'Transaction failed' };\n }\n\n // Transfer 事件签名\n const transferTopic = ethers.id('Transfer(address,address,uint256)');\n const usdcAddress = this.chainConfig.usdc.toLowerCase();\n\n for (const log of receipt.logs) {\n // 检查是否是 USDC 合约的 Transfer 事件\n if (\n log.address.toLowerCase() === usdcAddress &&\n log.topics.length >= 3 &&\n log.topics[0] === transferTopic\n ) {\n // 解析 from, to, amount\n const from = ethers.getAddress('0x' + log.topics[1].slice(-40));\n const to = ethers.getAddress('0x' + log.topics[2].slice(-40));\n const amountWei = BigInt(log.data);\n const amount = Number(amountWei) / 1e6;\n\n // 检查接收地址\n if (to.toLowerCase() !== this.walletAddress.toLowerCase()) {\n continue;\n }\n\n // 检查金额(允许误差)\n const tolerance = options.tolerance ?? 0.01;\n if (options.expectedAmount) {\n const diff = Math.abs(amount - options.expectedAmount);\n if (diff > options.expectedAmount * tolerance) {\n return {\n verified: false,\n error: `Amount mismatch: expected ${options.expectedAmount}, got ${amount}`,\n };\n }\n }\n\n const currentBlock = await this.provider.getBlockNumber();\n \n return {\n verified: true,\n tx_hash: txHash,\n amount: amount.toFixed(2),\n token: 'USDC',\n from,\n to,\n block_number: receipt.blockNumber,\n confirmations: currentBlock - receipt.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${txHash}`,\n };\n }\n }\n\n return { verified: false, error: 'No USDC transfer to recipient found in transaction' };\n } catch (error) {\n return { verified: false, error: (error as Error).message };\n }\n }\n\n /**\n * 扫描最近转账(按金额匹配)\n */\n async scanRecentTransfers(expectedAmount: number, timeoutMinutes: number = 30): Promise<VerifyResult> {\n try {\n const currentBlock = await this.provider.getBlockNumber();\n const blocksPerMinute = Math.ceil(60 / this.chainConfig.avgBlockTime);\n const fromBlock = currentBlock - (timeoutMinutes * blocksPerMinute);\n\n // 使用 getLogs 扫描 Transfer 事件\n const transferTopic = ethers.id('Transfer(address,address,uint256)');\n const recipientTopic = ethers.zeroPadValue(this.walletAddress, 32);\n\n const logs = await this.provider.getLogs({\n address: this.chainConfig.usdc,\n topics: [transferTopic, null, recipientTopic],\n fromBlock,\n toBlock: 'latest',\n });\n\n for (const log of logs) {\n const amountWei = BigInt(log.data);\n const amount = Number(amountWei) / 1e6;\n\n // 按金额匹配\n if (Math.abs(amount - expectedAmount) < 0.01) {\n const from = ethers.getAddress('0x' + log.topics[1].slice(-40));\n \n return {\n verified: true,\n tx_hash: log.transactionHash,\n amount: amount.toFixed(2),\n token: 'USDC',\n from,\n to: this.walletAddress,\n block_number: log.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${log.transactionHash}`,\n };\n }\n }\n\n return { verified: false, error: 'No matching payment found' };\n } catch (error) {\n return { verified: false, error: (error as Error).message };\n }\n }\n\n /**\n * 获取钱包余额\n */\n async getBalance(address?: string): Promise<WalletBalance> {\n const addr = address || this.walletAddress;\n \n const [ethBalance, usdcBalance] = await Promise.all([\n this.provider.getBalance(addr),\n this.usdcContract.balanceOf(addr),\n ]);\n\n return {\n address: addr,\n eth: ethers.formatEther(ethBalance),\n usdc: (Number(usdcBalance) / 1e6).toFixed(2),\n chain: this.chain,\n };\n }\n\n /**\n * 格式化 Invoice 为人类可读消息\n */\n formatInvoiceMessage(invoice: Invoice, includeJson: boolean = true): string {\n let msg = `🎬 **Payment Request**\n\n**Service:** ${invoice.service}\n**Price:** ${invoice.amount} USDC (${this.chainConfig.name})\n\n**💳 Payment Options:**\n\n1️⃣ **Direct Transfer:**\n Send exactly \\`${invoice.amount} USDC\\` to:\n \\`${invoice.recipient}\\`\n (Network: ${this.chainConfig.name})\n\n2️⃣ **One-Click Pay (MetaMask):**\n ${invoice.deep_link}\n\n⏱️ Expires: ${invoice.expires_at}`;\n\n if (includeJson) {\n msg += `\n\n3️⃣ **For AI Agents:**\n\\`\\`\\`json\n${JSON.stringify(invoice, null, 2)}\n\\`\\`\\``;\n }\n\n msg += `\n\nAfter payment, reply with your tx hash:\n\\`paid: 0x...\\``;\n\n return msg;\n }\n}\n","/**\n * 区块链配置\n */\n\nimport type { ChainConfig, ChainName } from '../types/index.js';\n\nexport const CHAINS: Record<ChainName, ChainConfig> = {\n // ============ 主网 ============\n base: {\n name: 'Base',\n chainId: 8453,\n rpc: 'https://mainnet.base.org',\n usdc: '0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913',\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-rpc.com',\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 usdc: '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48',\n explorer: 'https://etherscan.io/address/',\n explorerTx: 'https://etherscan.io/tx/',\n avgBlockTime: 12,\n },\n\n // ============ 测试网 ============\n base_sepolia: {\n name: 'Base Sepolia',\n chainId: 84532,\n rpc: 'https://sepolia.base.org',\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 usdc: '0x1c7D4B196Cb0C7B01d743Fbc6116a902379C7238',\n explorer: 'https://sepolia.etherscan.io/address/',\n explorerTx: 'https://sepolia.etherscan.io/tx/',\n avgBlockTime: 12,\n },\n};\n\n/**\n * 获取链配置\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 * 列出所有支持的链\n */\nexport function listChains(): ChainName[] {\n return Object.keys(CHAINS) as ChainName[];\n}\n\n/**\n * 根据 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(最小化,仅包含需要的方法)\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 };\n","/**\n * Wallet - 基础托管钱包\n * \n * 功能:\n * - 查询余额\n * - 发送 USDC 转账\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n ChainName,\n ChainConfig,\n WalletBalance,\n TransferResult,\n} from '../types/index.js';\n\nexport interface WalletConfig {\n chain?: ChainName;\n privateKey?: string;\n rpcUrl?: string;\n}\n\nexport class Wallet {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly address: string;\n \n private wallet: ethers.Wallet;\n private provider: ethers.JsonRpcProvider;\n private usdcContract: ethers.Contract;\n\n constructor(config: WalletConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n \n const privateKey = config.privateKey || process.env.PAYMENT_AGENT_PRIVATE_KEY;\n if (!privateKey) {\n throw new Error('privateKey is required. Set via config or PAYMENT_AGENT_PRIVATE_KEY env var.');\n }\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n this.wallet = new ethers.Wallet(privateKey, this.provider);\n this.address = this.wallet.address;\n \n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.wallet\n );\n }\n\n /**\n * 获取钱包余额\n */\n async getBalance(): Promise<WalletBalance> {\n const [ethBalance, usdcBalance] = await Promise.all([\n this.provider.getBalance(this.address),\n this.usdcContract.balanceOf(this.address),\n ]);\n\n return {\n address: this.address,\n eth: ethers.formatEther(ethBalance),\n usdc: (Number(usdcBalance) / 1e6).toFixed(2),\n chain: this.chain,\n };\n }\n\n /**\n * 发送 USDC 转账\n */\n async transfer(to: string, amount: number): Promise<TransferResult> {\n try {\n // 验证地址\n to = ethers.getAddress(to);\n \n // 转换金额(USDC 6位小数)\n const amountWei = BigInt(Math.floor(amount * 1e6));\n\n // 检查余额\n const balance = await this.usdcContract.balanceOf(this.address);\n if (BigInt(balance) < amountWei) {\n return {\n success: false,\n error: `Insufficient USDC balance: ${Number(balance) / 1e6} < ${amount}`,\n };\n }\n\n // 发送交易\n const tx = await this.usdcContract.transfer(to, amountWei);\n const receipt = await tx.wait();\n\n if (receipt.status === 1) {\n return {\n success: true,\n tx_hash: tx.hash,\n from: this.address,\n to,\n amount,\n gas_used: Number(receipt.gasUsed),\n block_number: receipt.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${tx.hash}`,\n };\n } else {\n return {\n success: false,\n tx_hash: tx.hash,\n error: 'Transaction reverted',\n };\n }\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 获取 ETH 余额\n */\n async getEthBalance(): Promise<string> {\n const balance = await this.provider.getBalance(this.address);\n return ethers.formatEther(balance);\n }\n\n /**\n * 获取 USDC 余额\n */\n async getUsdcBalance(): Promise<string> {\n const balance = await this.usdcContract.balanceOf(this.address);\n return (Number(balance) / 1e6).toFixed(2);\n }\n}\n","/**\n * AuditLog - 不可篡改审计日志\n * \n * 特点:\n * - 链式哈希,任何修改都会破坏链条\n * - 按日期分文件存储\n * - JSONL 格式便于追加和解析\n */\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as crypto from 'crypto';\nimport type { AuditEntry, AuditAction } from '../types/index.js';\n\nexport interface LogParams {\n action: AuditAction;\n request_id: string;\n from?: string;\n to?: string;\n amount?: number;\n tx_hash?: string;\n reason?: string;\n requester?: string;\n metadata?: Record<string, unknown>;\n}\n\nexport class AuditLog {\n private basePath: string;\n private lastHash: string = '0000000000000000';\n\n constructor(basePath?: string) {\n this.basePath = basePath || path.join(process.cwd(), 'data', 'audit');\n this.ensureDir();\n this.loadLastHash();\n }\n\n /**\n * 记录审计日志\n */\n async log(params: LogParams): Promise<AuditEntry> {\n const now = new Date();\n \n const entry: AuditEntry = {\n timestamp: now.getTime() / 1000,\n datetime: now.toISOString(),\n action: params.action,\n request_id: params.request_id,\n from: params.from,\n to: params.to,\n amount: params.amount,\n tx_hash: params.tx_hash,\n reason: params.reason,\n requester: params.requester,\n prev_hash: this.lastHash,\n hash: '', // 计算后填充\n metadata: params.metadata,\n };\n\n // 计算哈希(不包含 hash 字段本身)\n entry.hash = this.calculateHash(entry);\n this.lastHash = entry.hash;\n\n // 写入文件\n const filePath = this.getFilePath(now);\n const line = JSON.stringify(entry) + '\\n';\n fs.appendFileSync(filePath, line, 'utf-8');\n\n return entry;\n }\n\n /**\n * 读取指定日期的日志\n */\n read(date?: Date): AuditEntry[] {\n const filePath = this.getFilePath(date || new Date());\n \n if (!fs.existsSync(filePath)) {\n return [];\n }\n\n const content = fs.readFileSync(filePath, 'utf-8');\n const lines = content.trim().split('\\n').filter(Boolean);\n \n return lines.map(line => JSON.parse(line) as AuditEntry);\n }\n\n /**\n * 验证日志完整性\n */\n verify(date?: Date): { valid: boolean; errors: string[] } {\n const entries = this.read(date);\n const errors: string[] = [];\n\n for (let i = 0; i < entries.length; i++) {\n const entry = entries[i];\n \n // 验证哈希\n const expectedHash = this.calculateHash(entry);\n if (entry.hash !== expectedHash) {\n errors.push(`Entry ${i}: hash mismatch (expected ${expectedHash}, got ${entry.hash})`);\n }\n\n // 验证链接\n if (i > 0 && entry.prev_hash !== entries[i - 1].hash) {\n errors.push(`Entry ${i}: prev_hash mismatch`);\n }\n }\n\n return { valid: errors.length === 0, errors };\n }\n\n /**\n * 搜索日志\n */\n search(filter: Partial<{\n action: AuditAction;\n request_id: string;\n from: string;\n to: string;\n startDate: Date;\n endDate: Date;\n }>): AuditEntry[] {\n const results: AuditEntry[] = [];\n const startDate = filter.startDate || new Date(Date.now() - 30 * 24 * 60 * 60 * 1000);\n const endDate = filter.endDate || new Date();\n\n // 遍历日期范围\n const current = new Date(startDate);\n while (current <= endDate) {\n const entries = this.read(current);\n \n for (const entry of entries) {\n let match = true;\n\n if (filter.action && entry.action !== filter.action) match = false;\n if (filter.request_id && entry.request_id !== filter.request_id) match = false;\n if (filter.from && entry.from?.toLowerCase() !== filter.from.toLowerCase()) match = false;\n if (filter.to && entry.to?.toLowerCase() !== filter.to.toLowerCase()) match = false;\n\n if (match) {\n results.push(entry);\n }\n }\n\n current.setDate(current.getDate() + 1);\n }\n\n return results;\n }\n\n /**\n * 获取日志文件路径\n */\n private getFilePath(date: Date): string {\n const dateStr = date.toISOString().slice(0, 10);\n return path.join(this.basePath, `audit_${dateStr}.jsonl`);\n }\n\n /**\n * 计算条目哈希\n */\n private calculateHash(entry: AuditEntry): string {\n const data = {\n timestamp: entry.timestamp,\n action: entry.action,\n request_id: entry.request_id,\n from: entry.from,\n to: entry.to,\n amount: entry.amount,\n tx_hash: entry.tx_hash,\n prev_hash: entry.prev_hash,\n };\n \n const str = JSON.stringify(data);\n return crypto.createHash('sha256').update(str).digest('hex').slice(0, 16);\n }\n\n /**\n * 加载最后一条日志的哈希\n */\n private loadLastHash(): void {\n const today = new Date();\n \n // 检查今天和昨天的日志\n for (let i = 0; i < 2; i++) {\n const date = new Date(today);\n date.setDate(date.getDate() - i);\n \n const entries = this.read(date);\n if (entries.length > 0) {\n this.lastHash = entries[entries.length - 1].hash;\n return;\n }\n }\n }\n\n /**\n * 确保目录存在\n */\n private ensureDir(): void {\n if (!fs.existsSync(this.basePath)) {\n fs.mkdirSync(this.basePath, { recursive: true });\n }\n }\n}\n","/**\n * SecureWallet - 安全托管钱包\n * \n * 在基础 Wallet 之上增加:\n * - 单笔限额控制\n * - 日限额控制\n * - 白名单机制\n * - 审计日志\n * - 超限审批队列\n */\n\nimport { Wallet, type WalletConfig } from './Wallet.js';\nimport { AuditLog } from '../audit/AuditLog.js';\nimport type {\n SecurityLimits,\n SecureWalletConfig,\n TransferResult,\n TransferParams,\n PendingTransfer,\n} from '../types/index.js';\n\nconst DEFAULT_LIMITS: SecurityLimits = {\n singleMax: 100, // 单笔最大 $100\n dailyMax: 1000, // 日最大 $1000\n requireWhitelist: true,\n};\n\nexport class SecureWallet {\n private wallet: Wallet;\n private limits: SecurityLimits;\n private whitelist: Set<string>;\n private auditLog: AuditLog;\n private dailyTotal: number = 0;\n private dailyDate: string = '';\n private pendingTransfers: Map<string, PendingTransfer> = new Map();\n\n constructor(config: SecureWalletConfig = {}) {\n this.wallet = new Wallet({\n chain: config.chain,\n privateKey: config.privateKey,\n });\n\n this.limits = { ...DEFAULT_LIMITS, ...config.limits };\n this.whitelist = new Set((config.whitelist || []).map(a => a.toLowerCase()));\n this.auditLog = new AuditLog(config.auditPath);\n }\n\n /**\n * 获取钱包地址\n */\n get address(): string {\n return this.wallet.address;\n }\n\n /**\n * 获取余额\n */\n async getBalance() {\n return this.wallet.getBalance();\n }\n\n /**\n * 安全转账(带限额和白名单检查)\n * \n * 支持两种调用方式:\n * - transfer({ to, amount, reason?, requester? })\n * - transfer(to, amount)\n */\n async transfer(paramsOrTo: TransferParams | string, amountArg?: number | string): Promise<TransferResult> {\n // 支持两种调用方式\n let params: TransferParams;\n if (typeof paramsOrTo === 'string') {\n params = { \n to: paramsOrTo, \n amount: typeof amountArg === 'string' ? parseFloat(amountArg) : (amountArg || 0)\n };\n } else {\n params = paramsOrTo;\n }\n \n const { to, amount, reason, requester } = params;\n const toAddress = to.toLowerCase();\n const requestId = `tr_${Date.now().toString(36)}${Math.random().toString(36).slice(2, 8)}`;\n\n // 记录请求\n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n from: this.wallet.address,\n to,\n amount,\n reason,\n requester,\n });\n\n // 1. 白名单检查\n if (this.limits.requireWhitelist && !this.whitelist.has(toAddress)) {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: 'Address not in whitelist' },\n });\n return { success: false, error: `Address not in whitelist: ${to}` };\n }\n\n // 2. 单笔限额检查\n if (amount > this.limits.singleMax) {\n // 加入审批队列\n const pending: PendingTransfer = {\n id: requestId,\n to,\n amount,\n reason,\n requester,\n created_at: new Date().toISOString(),\n status: 'pending',\n };\n this.pendingTransfers.set(requestId, pending);\n \n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n metadata: { pending: true, reason: 'Exceeds single limit' },\n });\n \n return {\n success: false,\n error: `Amount ${amount} exceeds single limit ${this.limits.singleMax}. Pending approval: ${requestId}`,\n };\n }\n\n // 3. 日限额检查\n this.updateDailyTotal();\n if (this.dailyTotal + amount > this.limits.dailyMax) {\n const pending: PendingTransfer = {\n id: requestId,\n to,\n amount,\n reason,\n requester,\n created_at: new Date().toISOString(),\n status: 'pending',\n };\n this.pendingTransfers.set(requestId, pending);\n \n await this.auditLog.log({\n action: 'transfer_request',\n request_id: requestId,\n metadata: { pending: true, reason: 'Exceeds daily limit' },\n });\n \n return {\n success: false,\n error: `Daily limit would be exceeded (${this.dailyTotal} + ${amount} > ${this.limits.dailyMax}). Pending approval: ${requestId}`,\n };\n }\n\n // 4. 执行转账\n const result = await this.wallet.transfer(to, amount);\n\n // 5. 记录结果\n if (result.success) {\n this.dailyTotal += amount;\n await this.auditLog.log({\n action: 'transfer_executed',\n request_id: requestId,\n from: this.wallet.address,\n to,\n amount,\n tx_hash: result.tx_hash,\n reason,\n requester,\n });\n } else {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: result.error },\n });\n }\n\n return result;\n }\n\n /**\n * 审批待处理转账\n */\n async approve(requestId: string, approver: string): Promise<TransferResult> {\n const pending = this.pendingTransfers.get(requestId);\n if (!pending) {\n return { success: false, error: `Pending transfer not found: ${requestId}` };\n }\n\n if (pending.status !== 'pending') {\n return { success: false, error: `Transfer already ${pending.status}` };\n }\n\n await this.auditLog.log({\n action: 'transfer_approved',\n request_id: requestId,\n metadata: { approver },\n });\n\n pending.status = 'approved';\n\n // 执行转账(跳过限额检查)\n const result = await this.wallet.transfer(pending.to, pending.amount);\n\n if (result.success) {\n pending.status = 'executed';\n this.dailyTotal += pending.amount;\n \n await this.auditLog.log({\n action: 'transfer_executed',\n request_id: requestId,\n from: this.wallet.address,\n to: pending.to,\n amount: pending.amount,\n tx_hash: result.tx_hash,\n reason: pending.reason,\n requester: pending.requester,\n metadata: { approved_by: approver },\n });\n } else {\n await this.auditLog.log({\n action: 'transfer_failed',\n request_id: requestId,\n metadata: { error: result.error },\n });\n }\n\n return result;\n }\n\n /**\n * 拒绝待处理转账\n */\n async reject(requestId: string, rejecter: string, reason?: string): Promise<void> {\n const pending = this.pendingTransfers.get(requestId);\n if (!pending) {\n throw new Error(`Pending transfer not found: ${requestId}`);\n }\n\n pending.status = 'rejected';\n\n await this.auditLog.log({\n action: 'transfer_rejected',\n request_id: requestId,\n metadata: { rejected_by: rejecter, reason },\n });\n }\n\n /**\n * 添加白名单地址\n */\n async addToWhitelist(address: string, addedBy: string): Promise<void> {\n const addr = address.toLowerCase();\n this.whitelist.add(addr);\n\n await this.auditLog.log({\n action: 'whitelist_add',\n request_id: `wl_${Date.now()}`,\n to: address,\n metadata: { added_by: addedBy },\n });\n }\n\n /**\n * 移除白名单地址\n */\n async removeFromWhitelist(address: string, removedBy: string): Promise<void> {\n const addr = address.toLowerCase();\n this.whitelist.delete(addr);\n\n await this.auditLog.log({\n action: 'whitelist_remove',\n request_id: `wl_${Date.now()}`,\n to: address,\n metadata: { removed_by: removedBy },\n });\n }\n\n /**\n * 检查地址是否在白名单\n */\n isWhitelisted(address: string): boolean {\n return this.whitelist.has(address.toLowerCase());\n }\n\n /**\n * 获取待处理转账列表\n */\n getPendingTransfers(): PendingTransfer[] {\n return Array.from(this.pendingTransfers.values())\n .filter(p => p.status === 'pending');\n }\n\n /**\n * 获取当前限额配置\n */\n getLimits(): SecurityLimits {\n return { ...this.limits };\n }\n\n /**\n * 获取今日已用额度\n */\n getDailyUsed(): number {\n this.updateDailyTotal();\n return this.dailyTotal;\n }\n\n /**\n * 更新日限额计数器\n */\n private updateDailyTotal(): void {\n const today = new Date().toISOString().slice(0, 10);\n if (this.dailyDate !== today) {\n this.dailyDate = today;\n this.dailyTotal = 0;\n }\n }\n}\n","/**\n * createWallet - Create a new wallet for Agent\n * \n * Features:\n * - Generate new Ethereum wallet\n * - Securely store private key (encrypted or plaintext, depending on config)\n * - Return wallet address (never return private key)\n */\n\nimport { ethers } from 'ethers';\nimport { writeFileSync, readFileSync, existsSync, mkdirSync } from 'fs';\nimport { join, dirname } from 'path';\nimport { createCipheriv, createDecipheriv, randomBytes, scryptSync } from 'crypto';\n\nexport interface CreateWalletOptions {\n /** Storage path, default ~/.moltspay/wallet.json */\n storagePath?: string;\n /** Encryption password (optional, plaintext if not provided) */\n password?: string;\n /** Wallet label/name */\n label?: string;\n /** Overwrite if wallet exists */\n overwrite?: boolean;\n}\n\nexport interface WalletData {\n address: string;\n label?: string;\n createdAt: string;\n encrypted: boolean;\n /** Encrypted or plaintext private key */\n privateKey: string;\n /** Encryption IV */\n iv?: string;\n /** Encryption salt */\n salt?: string;\n}\n\nexport interface CreateWalletResult {\n success: boolean;\n address?: string;\n storagePath?: string;\n error?: string;\n /** Whether newly created (false means loaded existing) */\n isNew?: boolean;\n}\n\nconst DEFAULT_STORAGE_DIR = join(process.env.HOME || '~', '.moltspay');\nconst DEFAULT_STORAGE_FILE = 'wallet.json';\n\n/**\n * Encrypt private key\n */\nfunction encryptPrivateKey(privateKey: string, password: string): { encrypted: string; iv: string; salt: string } {\n const salt = randomBytes(16);\n const key = scryptSync(password, salt, 32);\n const iv = randomBytes(16);\n \n const cipher = createCipheriv('aes-256-cbc', key, iv);\n let encrypted = cipher.update(privateKey, 'utf8', 'hex');\n encrypted += cipher.final('hex');\n \n return {\n encrypted,\n iv: iv.toString('hex'),\n salt: salt.toString('hex'),\n };\n}\n\n/**\n * Decrypt private key\n */\nfunction decryptPrivateKey(encrypted: string, password: string, iv: string, salt: string): string {\n const key = scryptSync(password, Buffer.from(salt, 'hex'), 32);\n const decipher = createDecipheriv('aes-256-cbc', key, Buffer.from(iv, 'hex'));\n \n let decrypted = decipher.update(encrypted, 'hex', 'utf8');\n decrypted += decipher.final('utf8');\n \n return decrypted;\n}\n\n/**\n * Create new wallet\n * \n * @example\n * ```typescript\n * // Create unencrypted wallet\n * const result = await createWallet();\n * console.log('Wallet address:', result.address);\n * \n * // Create encrypted wallet\n * const result = await createWallet({ password: 'mySecurePassword' });\n * \n * // Specify storage path\n * const result = await createWallet({ storagePath: './my-wallet.json' });\n * ```\n */\nexport function createWallet(options: CreateWalletOptions = {}): CreateWalletResult {\n const storagePath = options.storagePath || join(DEFAULT_STORAGE_DIR, DEFAULT_STORAGE_FILE);\n \n // Check if exists\n if (existsSync(storagePath) && !options.overwrite) {\n // Load existing wallet\n try {\n const existing = JSON.parse(readFileSync(storagePath, 'utf8')) as WalletData;\n return {\n success: true,\n address: existing.address,\n storagePath,\n isNew: false,\n };\n } catch (error) {\n return {\n success: false,\n error: `Failed to load existing wallet: ${(error as Error).message}`,\n };\n }\n }\n\n try {\n // Create new wallet\n const wallet = ethers.Wallet.createRandom();\n \n // Prepare storage data\n const walletData: WalletData = {\n address: wallet.address,\n label: options.label,\n createdAt: new Date().toISOString(),\n encrypted: !!options.password,\n privateKey: '',\n };\n\n if (options.password) {\n // Encrypted storage\n const { encrypted, iv, salt } = encryptPrivateKey(wallet.privateKey, options.password);\n walletData.privateKey = encrypted;\n walletData.iv = iv;\n walletData.salt = salt;\n } else {\n // Plaintext storage (not recommended, but convenient for testing/dev)\n walletData.privateKey = wallet.privateKey;\n }\n\n // Ensure directory exists\n const dir = dirname(storagePath);\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true });\n }\n\n // Write file\n writeFileSync(storagePath, JSON.stringify(walletData, null, 2), { mode: 0o600 });\n\n return {\n success: true,\n address: wallet.address,\n storagePath,\n isNew: true,\n };\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n}\n\n/**\n * Load existing wallet\n */\nexport function loadWallet(options: { storagePath?: string; password?: string } = {}): {\n success: boolean;\n address?: string;\n privateKey?: string;\n error?: string;\n} {\n const storagePath = options.storagePath || join(DEFAULT_STORAGE_DIR, DEFAULT_STORAGE_FILE);\n\n if (!existsSync(storagePath)) {\n return { success: false, error: 'Wallet not found. Run createWallet() first.' };\n }\n\n try {\n const data = JSON.parse(readFileSync(storagePath, 'utf8')) as WalletData;\n\n if (data.encrypted) {\n if (!options.password) {\n return { success: false, error: 'Wallet is encrypted. Password required.' };\n }\n const privateKey = decryptPrivateKey(data.privateKey, options.password, data.iv!, data.salt!);\n return { success: true, address: data.address, privateKey };\n } else {\n return { success: true, address: data.address, privateKey: data.privateKey };\n }\n } catch (error) {\n return { success: false, error: (error as Error).message };\n }\n}\n\n/**\n * Get wallet address (no password required)\n */\nexport function getWalletAddress(storagePath?: string): string | null {\n const path = storagePath || join(DEFAULT_STORAGE_DIR, DEFAULT_STORAGE_FILE);\n \n if (!existsSync(path)) {\n return null;\n }\n\n try {\n const data = JSON.parse(readFileSync(path, 'utf8')) as WalletData;\n return data.address;\n } catch {\n return null;\n }\n}\n\n/**\n * Check if wallet exists\n */\nexport function walletExists(storagePath?: string): boolean {\n const path = storagePath || join(DEFAULT_STORAGE_DIR, DEFAULT_STORAGE_FILE);\n return existsSync(path);\n}\n","/**\n * PermitWallet - Pay using Boss's Permit authorization\n * \n * Scenario:\n * - Agent doesn't have USDC, but Boss gave a Permit authorization\n * - Agent uses Permit signature + own wallet to execute transferFrom\n * - Agent only needs small amount of ETH for gas, USDC is deducted from Boss's wallet\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport { loadWallet } from './createWallet.js';\nimport type {\n ChainName,\n ChainConfig,\n TransferResult,\n PermitSignature,\n} from '../types/index.js';\n\nexport interface PermitData {\n /** Boss's wallet address (USDC holder) */\n owner: string;\n /** Agent's wallet address (authorized spender) */\n spender: string;\n /** Authorized amount (USDC, raw 6 decimal value) */\n value: string;\n /** Expiration timestamp */\n deadline: number;\n /** Signature v */\n v: number;\n /** Signature r */\n r: string;\n /** Signature s */\n s: string;\n}\n\nexport interface PermitWalletConfig {\n chain?: ChainName;\n /** Agent's private key (for executing transactions) */\n privateKey?: string;\n /** Load private key from file */\n walletPath?: string;\n /** Decryption password */\n walletPassword?: string;\n rpcUrl?: string;\n}\n\nexport interface TransferWithPermitParams {\n /** Recipient address */\n to: string;\n /** Amount (USDC) */\n amount: number;\n /** Boss-signed Permit data */\n permit: PermitData;\n}\n\nexport interface TransferWithPermitResult extends TransferResult {\n /** Permit transaction hash */\n permitTxHash?: string;\n /** Transfer transaction hash */\n transferTxHash?: string;\n}\n\n// Extended ABI to support permit and transferFrom\nconst PERMIT_ABI = [\n ...ERC20_ABI,\n 'function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s)',\n 'function transferFrom(address from, address to, uint256 amount) returns (bool)',\n 'function allowance(address owner, address spender) view returns (uint256)',\n];\n\nexport class PermitWallet {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly address: string;\n \n private wallet: ethers.Wallet;\n private provider: ethers.JsonRpcProvider;\n private usdcContract: ethers.Contract;\n\n constructor(config: PermitWalletConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n \n // Get private key\n let privateKey = config.privateKey || process.env.PAYMENT_AGENT_PRIVATE_KEY;\n \n // Or load from file\n if (!privateKey && config.walletPath) {\n const loaded = loadWallet({ \n storagePath: config.walletPath, \n password: config.walletPassword \n });\n if (!loaded.success || !loaded.privateKey) {\n throw new Error(loaded.error || 'Failed to load wallet');\n }\n privateKey = loaded.privateKey;\n }\n \n if (!privateKey) {\n throw new Error('privateKey is required. Set via config, env var, or walletPath.');\n }\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n this.wallet = new ethers.Wallet(privateKey, this.provider);\n this.address = this.wallet.address;\n \n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n PERMIT_ABI,\n this.wallet\n );\n }\n\n /**\n * Check if Permit is valid (current allowance)\n */\n async checkPermitAllowance(owner: string): Promise<string> {\n const allowance = await this.usdcContract.allowance(owner, this.address);\n return (Number(allowance) / 1e6).toFixed(2);\n }\n\n /**\n * Pay using Permit authorization\n * \n * Flow:\n * 1. Call permit() to record Boss's authorization in the contract\n * 2. Call transferFrom() to transfer from Boss's wallet to recipient\n * \n * @example\n * ```typescript\n * const wallet = new PermitWallet({ chain: 'base' });\n * \n * // Boss-signed permit data\n * const permit = {\n * owner: '0xBOSS...',\n * spender: wallet.address,\n * value: '10000000', // 10 USDC\n * deadline: 1234567890,\n * v: 27,\n * r: '0x...',\n * s: '0x...'\n * };\n * \n * const result = await wallet.transferWithPermit({\n * to: '0xSELLER...',\n * amount: 3.99,\n * permit\n * });\n * ```\n */\n async transferWithPermit(params: TransferWithPermitParams): Promise<TransferWithPermitResult> {\n const { to, amount, permit } = params;\n\n try {\n // Validate addresses\n const toAddress = ethers.getAddress(to);\n const ownerAddress = ethers.getAddress(permit.owner);\n \n // Verify spender is this wallet\n if (ethers.getAddress(permit.spender).toLowerCase() !== this.address.toLowerCase()) {\n return {\n success: false,\n error: `Permit spender (${permit.spender}) doesn't match wallet address (${this.address})`,\n };\n }\n\n // Check deadline\n const now = Math.floor(Date.now() / 1000);\n if (permit.deadline < now) {\n return {\n success: false,\n error: `Permit expired at ${new Date(permit.deadline * 1000).toISOString()}`,\n };\n }\n\n // Convert amount\n const amountWei = BigInt(Math.floor(amount * 1e6));\n const permitValue = BigInt(permit.value);\n \n // Check if authorized amount is sufficient\n if (amountWei > permitValue) {\n return {\n success: false,\n error: `Permit value (${Number(permitValue) / 1e6} USDC) < transfer amount (${amount} USDC)`,\n };\n }\n\n // Check existing allowance\n const currentAllowance = await this.usdcContract.allowance(ownerAddress, this.address);\n \n let permitTxHash: string | undefined;\n \n // If allowance insufficient, execute permit first\n if (BigInt(currentAllowance) < amountWei) {\n console.log('Executing permit...');\n const permitTx = await this.usdcContract.permit(\n ownerAddress,\n this.address,\n permitValue,\n permit.deadline,\n permit.v,\n permit.r,\n permit.s\n );\n const permitReceipt = await permitTx.wait();\n \n if (permitReceipt.status !== 1) {\n return {\n success: false,\n error: 'Permit transaction failed',\n permitTxHash: permitTx.hash,\n };\n }\n permitTxHash = permitTx.hash;\n console.log('Permit executed:', permitTxHash);\n }\n\n // Execute transferFrom\n console.log('Executing transferFrom...');\n const transferTx = await this.usdcContract.transferFrom(\n ownerAddress,\n toAddress,\n amountWei\n );\n const transferReceipt = await transferTx.wait();\n\n if (transferReceipt.status === 1) {\n return {\n success: true,\n tx_hash: transferTx.hash,\n permitTxHash,\n transferTxHash: transferTx.hash,\n from: ownerAddress,\n to: toAddress,\n amount,\n gas_used: Number(transferReceipt.gasUsed),\n block_number: transferReceipt.blockNumber,\n explorer_url: `${this.chainConfig.explorerTx}${transferTx.hash}`,\n };\n } else {\n return {\n success: false,\n error: 'TransferFrom transaction failed',\n tx_hash: transferTx.hash,\n permitTxHash,\n };\n }\n } catch (error) {\n const message = (error as Error).message;\n \n // Parse common errors\n if (message.includes('ERC20InsufficientAllowance')) {\n return {\n success: false,\n error: 'Insufficient allowance. Permit may have been used or expired.',\n };\n }\n if (message.includes('ERC20InsufficientBalance')) {\n return {\n success: false,\n error: 'Boss wallet has insufficient USDC balance.',\n };\n }\n if (message.includes('InvalidSignature') || message.includes('invalid signature')) {\n return {\n success: false,\n error: 'Invalid permit signature. Ask Boss to re-sign.',\n };\n }\n \n return {\n success: false,\n error: message,\n };\n }\n }\n\n /**\n * Get ETH balance (for gas)\n */\n async getGasBalance(): Promise<string> {\n const balance = await this.provider.getBalance(this.address);\n return ethers.formatEther(balance);\n }\n\n /**\n * Check if there's enough gas\n */\n async hasEnoughGas(minEth: number = 0.001): Promise<boolean> {\n const balance = await this.getGasBalance();\n return parseFloat(balance) >= minEth;\n }\n}\n\n/**\n * Format Permit request message (to send to Boss)\n */\nexport function formatPermitRequest(params: {\n agentAddress: string;\n amount: number;\n deadlineHours?: number;\n chain?: ChainName;\n reason?: string;\n}): string {\n const { agentAddress, amount, deadlineHours = 24, chain = 'base', reason } = params;\n const chainConfig = getChain(chain);\n const deadline = Math.floor(Date.now() / 1000) + deadlineHours * 3600;\n const value = BigInt(Math.floor(amount * 1e6)).toString();\n\n return `🔐 **USDC Spending Allowance Request**\n\n${reason ? `**Purpose:** ${reason}\\n` : ''}\n**Authorization Details:**\n- Authorized address (Agent): \\`${agentAddress}\\`\n- Amount: ${amount} USDC\n- Valid for: ${deadlineHours} hours\n- Chain: ${chainConfig.name}\n\n**Please sign the following EIP-2612 Permit with your wallet:**\n\n\\`\\`\\`json\n{\n \"types\": {\n \"Permit\": [\n { \"name\": \"owner\", \"type\": \"address\" },\n { \"name\": \"spender\", \"type\": \"address\" },\n { \"name\": \"value\", \"type\": \"uint256\" },\n { \"name\": \"nonce\", \"type\": \"uint256\" },\n { \"name\": \"deadline\", \"type\": \"uint256\" }\n ]\n },\n \"primaryType\": \"Permit\",\n \"domain\": {\n \"name\": \"USD Coin\",\n \"version\": \"2\",\n \"chainId\": ${chainConfig.chainId},\n \"verifyingContract\": \"${chainConfig.usdc}\"\n },\n \"message\": {\n \"owner\": \"<YOUR_WALLET_ADDRESS>\",\n \"spender\": \"${agentAddress}\",\n \"value\": \"${value}\",\n \"nonce\": \"<GET_FROM_CONTRACT>\",\n \"deadline\": ${deadline}\n }\n}\n\\`\\`\\`\n\nAfter signing, send { v, r, s, deadline } to the Agent.\n\n⚠️ Note: This authorization only allows the Agent to spend up to ${amount} USDC from your wallet. Your private key is never exposed.`;\n}\n","/**\n * PermitPayment - EIP-2612 无 Gas 预授权\n * \n * 让用户通过签名授权,服务方代付 Gas 执行 transferFrom\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, ERC20_ABI } from '../chains/index.js';\nimport type {\n ChainName,\n ChainConfig,\n PermitRequest,\n PermitSignature,\n PermitExecuteResult,\n EIP712TypedData,\n} from '../types/index.js';\n\nexport interface PermitConfig {\n chain?: ChainName;\n privateKey?: string;\n spenderAddress?: string;\n rpcUrl?: string;\n}\n\nexport class PermitPayment {\n readonly chain: ChainName;\n readonly chainConfig: ChainConfig;\n readonly spenderAddress: string;\n \n private provider: ethers.JsonRpcProvider;\n private wallet?: ethers.Wallet;\n private usdcContract: ethers.Contract;\n\n constructor(config: PermitConfig = {}) {\n this.chain = config.chain || 'base_sepolia';\n this.chainConfig = getChain(this.chain);\n this.spenderAddress = config.spenderAddress || process.env.PAYMENT_AGENT_WALLET || '';\n\n const rpcUrl = config.rpcUrl || this.chainConfig.rpc;\n this.provider = new ethers.JsonRpcProvider(rpcUrl);\n\n const privateKey = config.privateKey || process.env.PAYMENT_AGENT_PRIVATE_KEY;\n if (privateKey) {\n this.wallet = new ethers.Wallet(privateKey, this.provider);\n this.spenderAddress = this.wallet.address;\n }\n\n this.usdcContract = new ethers.Contract(\n this.chainConfig.usdc,\n ERC20_ABI,\n this.wallet || this.provider\n );\n }\n\n /**\n * 获取用户当前 nonce\n */\n async getNonce(owner: string): Promise<number> {\n return Number(await this.usdcContract.nonces(owner));\n }\n\n /**\n * 生成 EIP-712 签名请求(发给前端/用户钱包)\n */\n async createPermitRequest(\n owner: string,\n amount: number,\n orderId: string,\n deadlineMinutes: number = 30\n ): Promise<PermitRequest> {\n const nonce = await this.getNonce(owner);\n const deadline = Math.floor(Date.now() / 1000) + deadlineMinutes * 60;\n const value = BigInt(Math.floor(amount * 1e6)).toString();\n\n // USDC 的 EIP-712 domain(不同链可能不同)\n const domain = {\n name: 'USD Coin',\n version: '2',\n chainId: this.chainConfig.chainId,\n verifyingContract: this.chainConfig.usdc,\n };\n\n const types = {\n EIP712Domain: [\n { name: 'name', type: 'string' },\n { name: 'version', type: 'string' },\n { name: 'chainId', type: 'uint256' },\n { name: 'verifyingContract', type: 'address' },\n ],\n Permit: [\n { name: 'owner', type: 'address' },\n { name: 'spender', type: 'address' },\n { name: 'value', type: 'uint256' },\n { name: 'nonce', type: 'uint256' },\n { name: 'deadline', type: 'uint256' },\n ],\n };\n\n const message = {\n owner,\n spender: this.spenderAddress,\n value,\n nonce,\n deadline,\n };\n\n const typedData: EIP712TypedData = {\n types,\n primaryType: 'Permit',\n domain,\n message,\n };\n\n return {\n type: 'permit_request',\n version: '1.0',\n order_id: orderId,\n typed_data: typedData,\n };\n }\n\n /**\n * 执行 permit + transferFrom\n * \n * @param owner 用户地址\n * @param amount 金额\n * @param signature 用户签名 {v, r, s, deadline}\n */\n async executePermitAndTransfer(\n owner: string,\n amount: number,\n signature: PermitSignature\n ): Promise<PermitExecuteResult> {\n if (!this.wallet) {\n return { success: false, error: 'Wallet not configured. Private key required.' };\n }\n\n try {\n const value = BigInt(Math.floor(amount * 1e6));\n\n // 1. 调用 permit\n const permitTx = await this.usdcContract.permit(\n owner,\n this.spenderAddress,\n value,\n signature.deadline,\n signature.v,\n signature.r,\n signature.s\n );\n await permitTx.wait();\n\n // 2. 调用 transferFrom\n const transferTx = await this.usdcContract.transferFrom(owner, this.spenderAddress, value);\n const receipt = await transferTx.wait();\n\n return {\n success: receipt.status === 1,\n tx_hash: transferTx.hash,\n };\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 仅执行 permit(不 transfer)\n */\n async executePermit(\n owner: string,\n amount: number,\n signature: PermitSignature\n ): Promise<PermitExecuteResult> {\n if (!this.wallet) {\n return { success: false, error: 'Wallet not configured. Private key required.' };\n }\n\n try {\n const value = BigInt(Math.floor(amount * 1e6));\n\n const tx = await this.usdcContract.permit(\n owner,\n this.spenderAddress,\n value,\n signature.deadline,\n signature.v,\n signature.r,\n signature.s\n );\n const receipt = await tx.wait();\n\n return {\n success: receipt.status === 1,\n tx_hash: tx.hash,\n };\n } catch (error) {\n return {\n success: false,\n error: (error as Error).message,\n };\n }\n }\n\n /**\n * 格式化 Permit 请求为用户消息\n */\n formatPermitMessage(request: PermitRequest): string {\n const { typed_data } = request;\n const { message } = typed_data;\n\n return `🔐 **签名授权请求**\n\n授权 \\`${(Number(message.value) / 1e6).toFixed(2)} USDC\\` 给服务方\n\n**签名信息:**\n- Owner: \\`${message.owner}\\`\n- Spender: \\`${message.spender}\\`\n- Amount: ${(Number(message.value) / 1e6).toFixed(2)} USDC\n- Deadline: ${new Date(message.deadline * 1000).toISOString()}\n\n请在钱包中签名此请求(不消耗 Gas)。\n\n\\`\\`\\`json\n${JSON.stringify(typed_data, null, 2)}\n\\`\\`\\``;\n }\n}\n","/**\n * 链上支付验证模块\n */\n\nimport { ethers } from 'ethers';\nimport { getChain, getChainById, type ChainConfig, type ChainName } from '../chains';\n\n// ERC20 Transfer 事件签名\nconst TRANSFER_EVENT_TOPIC = ethers.id('Transfer(address,address,uint256)');\n\nexport interface VerifyPaymentParams {\n txHash: string;\n expectedAmount: number;\n expectedTo?: string;\n chain?: string | number;\n}\n\nexport interface VerifyPaymentResult {\n verified: boolean;\n amount?: number;\n from?: string;\n to?: string;\n txHash?: string;\n blockNumber?: number;\n error?: string;\n}\n\n/**\n * 验证链上支付\n */\nexport async function verifyPayment(params: VerifyPaymentParams): Promise<VerifyPaymentResult> {\n const { txHash, expectedAmount, expectedTo } = params;\n \n // 获取链配置\n let chain: ChainConfig | undefined;\n try {\n if (typeof params.chain === 'number') {\n chain = getChainById(params.chain);\n } else {\n chain = getChain((params.chain || 'base') as ChainName);\n }\n if (!chain) {\n return { verified: false, error: `不支持的链: ${params.chain}` };\n }\n } catch (e) {\n return { verified: false, error: `不支持的链: ${params.chain}` };\n }\n\n try {\n const provider = new ethers.JsonRpcProvider(chain.rpc);\n \n // 获取交易回执\n const receipt = await provider.getTransactionReceipt(txHash);\n \n if (!receipt) {\n return { verified: false, error: '交易未找到或未确认' };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, error: '交易失败' };\n }\n\n // 解析 Transfer 事件\n const usdcAddress = chain.usdc?.toLowerCase();\n if (!usdcAddress) {\n return { verified: false, error: `链 ${chain.name} 未配置USDC地址` };\n }\n\n for (const log of receipt.logs) {\n // 检查是否是 USDC 合约\n if (log.address.toLowerCase() !== usdcAddress) {\n continue;\n }\n\n // 检查是否是 Transfer 事件\n if (log.topics.length < 3 || log.topics[0] !== TRANSFER_EVENT_TOPIC) {\n continue;\n }\n\n // 解析 Transfer 事件参数\n const from = '0x' + log.topics[1].slice(-40);\n const to = '0x' + log.topics[2].slice(-40);\n const amountRaw = BigInt(log.data);\n const amount = Number(amountRaw) / 1e6; // USDC 6位小数\n\n // 验证收款地址\n if (expectedTo && to.toLowerCase() !== expectedTo.toLowerCase()) {\n continue;\n }\n\n // 验证金额\n if (amount < expectedAmount) {\n return {\n verified: false,\n error: `金额不足: 收到 ${amount} USDC, 需要 ${expectedAmount} USDC`,\n amount,\n from,\n to,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n }\n\n // 验证成功\n return {\n verified: true,\n amount,\n from,\n to,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n }\n\n return { verified: false, error: '未找到USDC转账记录' };\n\n } catch (e: any) {\n return { verified: false, error: e.message || String(e) };\n }\n}\n\n/**\n * 获取交易状态\n */\nexport async function getTransactionStatus(\n txHash: string,\n chain: string | number = 'base'\n): Promise<{\n status: 'pending' | 'confirmed' | 'failed' | 'not_found';\n blockNumber?: number;\n confirmations?: number;\n}> {\n let chainConfig: ChainConfig | undefined;\n try {\n chainConfig = typeof chain === 'number' ? getChainById(chain) : getChain(chain as ChainName);\n if (!chainConfig) return { status: 'not_found' };\n } catch {\n return { status: 'not_found' };\n }\n\n try {\n const provider = new ethers.JsonRpcProvider(chainConfig.rpc);\n const receipt = await provider.getTransactionReceipt(txHash);\n\n if (!receipt) {\n // 检查是否在 pending 池中\n const tx = await provider.getTransaction(txHash);\n if (tx) {\n return { status: 'pending' };\n }\n return { status: 'not_found' };\n }\n\n const currentBlock = await provider.getBlockNumber();\n const confirmations = currentBlock - receipt.blockNumber;\n\n if (receipt.status === 1) {\n return {\n status: 'confirmed',\n blockNumber: receipt.blockNumber,\n confirmations,\n };\n } else {\n return {\n status: 'failed',\n blockNumber: receipt.blockNumber,\n };\n }\n } catch {\n return { status: 'not_found' };\n }\n}\n\n/**\n * 等待交易确认\n */\nexport async function waitForTransaction(\n txHash: string,\n chain: string | number = 'base',\n confirmations = 1,\n timeoutMs = 60000\n): Promise<VerifyPaymentResult & { confirmed: boolean }> {\n let chainConfig: ChainConfig | undefined;\n try {\n chainConfig = typeof chain === 'number' ? getChainById(chain) : getChain(chain as ChainName);\n if (!chainConfig) {\n return { verified: false, confirmed: false, error: `不支持的链: ${chain}` };\n }\n } catch (e) {\n return { verified: false, confirmed: false, error: `不支持的链: ${chain}` };\n }\n\n const provider = new ethers.JsonRpcProvider(chainConfig.rpc);\n \n try {\n const receipt = await provider.waitForTransaction(txHash, confirmations, timeoutMs);\n \n if (!receipt) {\n return { verified: false, confirmed: false, error: '等待超时' };\n }\n\n if (receipt.status !== 1) {\n return { verified: false, confirmed: true, error: '交易失败' };\n }\n\n return {\n verified: true,\n confirmed: true,\n txHash,\n blockNumber: receipt.blockNumber,\n };\n } catch (e: any) {\n return { verified: false, confirmed: false, error: e.message || String(e) };\n }\n}\n"],"mappings":"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","names":["import_ethers","import_ethers","import_ethers","import_ethers","import_ethers"]}
|
package/dist/cli.mjs
CHANGED
|
@@ -781,12 +781,26 @@ var SecureWallet = class {
|
|
|
781
781
|
}
|
|
782
782
|
};
|
|
783
783
|
|
|
784
|
-
// src/
|
|
784
|
+
// src/wallet/createWallet.ts
|
|
785
785
|
import { ethers as ethers3 } from "ethers";
|
|
786
|
+
import { join as join2, dirname } from "path";
|
|
787
|
+
var DEFAULT_STORAGE_DIR = join2(process.env.HOME || "~", ".moltspay");
|
|
786
788
|
|
|
787
|
-
// src/
|
|
789
|
+
// src/wallet/PermitWallet.ts
|
|
788
790
|
import { ethers as ethers4 } from "ethers";
|
|
789
|
-
var
|
|
791
|
+
var PERMIT_ABI = [
|
|
792
|
+
...ERC20_ABI,
|
|
793
|
+
"function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s)",
|
|
794
|
+
"function transferFrom(address from, address to, uint256 amount) returns (bool)",
|
|
795
|
+
"function allowance(address owner, address spender) view returns (uint256)"
|
|
796
|
+
];
|
|
797
|
+
|
|
798
|
+
// src/permit/Permit.ts
|
|
799
|
+
import { ethers as ethers5 } from "ethers";
|
|
800
|
+
|
|
801
|
+
// src/verify/index.ts
|
|
802
|
+
import { ethers as ethers6 } from "ethers";
|
|
803
|
+
var TRANSFER_EVENT_TOPIC = ethers6.id("Transfer(address,address,uint256)");
|
|
790
804
|
|
|
791
805
|
// src/cli.ts
|
|
792
806
|
var program = new Command();
|