@agirails/sdk 2.2.3 → 2.3.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 +65 -31
- package/dist/ACTPClient.d.ts +42 -1
- package/dist/ACTPClient.d.ts.map +1 -1
- package/dist/ACTPClient.js +207 -22
- package/dist/ACTPClient.js.map +1 -1
- package/dist/abi/AgentRegistry.json +133 -0
- package/dist/adapters/AdapterRouter.d.ts.map +1 -1
- package/dist/adapters/AdapterRouter.js.map +1 -1
- package/dist/adapters/BasicAdapter.d.ts +10 -1
- package/dist/adapters/BasicAdapter.d.ts.map +1 -1
- package/dist/adapters/BasicAdapter.js +36 -1
- package/dist/adapters/BasicAdapter.js.map +1 -1
- package/dist/adapters/X402Adapter.d.ts +34 -7
- package/dist/adapters/X402Adapter.d.ts.map +1 -1
- package/dist/adapters/X402Adapter.js +36 -8
- package/dist/adapters/X402Adapter.js.map +1 -1
- package/dist/adapters/index.d.ts +1 -1
- package/dist/adapters/index.d.ts.map +1 -1
- package/dist/adapters/index.js.map +1 -1
- package/dist/cli/commands/diff.d.ts +11 -0
- package/dist/cli/commands/diff.d.ts.map +1 -0
- package/dist/cli/commands/diff.js +115 -0
- package/dist/cli/commands/diff.js.map +1 -0
- package/dist/cli/commands/init.d.ts +1 -0
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +260 -19
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/cli/commands/publish.d.ts +11 -0
- package/dist/cli/commands/publish.d.ts.map +1 -0
- package/dist/cli/commands/publish.js +170 -0
- package/dist/cli/commands/publish.js.map +1 -0
- package/dist/cli/commands/pull.d.ts +12 -0
- package/dist/cli/commands/pull.d.ts.map +1 -0
- package/dist/cli/commands/pull.js +99 -0
- package/dist/cli/commands/pull.js.map +1 -0
- package/dist/cli/commands/register.d.ts +16 -0
- package/dist/cli/commands/register.d.ts.map +1 -0
- package/dist/cli/commands/register.js +211 -0
- package/dist/cli/commands/register.js.map +1 -0
- package/dist/cli/index.js +10 -0
- package/dist/cli/index.js.map +1 -1
- package/dist/cli/utils/config.d.ts +6 -0
- package/dist/cli/utils/config.d.ts.map +1 -1
- package/dist/cli/utils/config.js.map +1 -1
- package/dist/config/agirailsmd.d.ts +94 -0
- package/dist/config/agirailsmd.d.ts.map +1 -0
- package/dist/config/agirailsmd.js +209 -0
- package/dist/config/agirailsmd.js.map +1 -0
- package/dist/config/networks.d.ts +22 -4
- package/dist/config/networks.d.ts.map +1 -1
- package/dist/config/networks.js +64 -26
- package/dist/config/networks.js.map +1 -1
- package/dist/config/publishPipeline.d.ts +75 -0
- package/dist/config/publishPipeline.d.ts.map +1 -0
- package/dist/config/publishPipeline.js +193 -0
- package/dist/config/publishPipeline.js.map +1 -0
- package/dist/config/syncOperations.d.ts +67 -0
- package/dist/config/syncOperations.d.ts.map +1 -0
- package/dist/config/syncOperations.js +208 -0
- package/dist/config/syncOperations.js.map +1 -0
- package/dist/erc8004/ERC8004Bridge.d.ts.map +1 -1
- package/dist/erc8004/ERC8004Bridge.js +6 -5
- package/dist/erc8004/ERC8004Bridge.js.map +1 -1
- package/dist/erc8004/ReputationReporter.d.ts.map +1 -1
- package/dist/erc8004/ReputationReporter.js +9 -12
- package/dist/erc8004/ReputationReporter.js.map +1 -1
- package/dist/index.d.ts +5 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +11 -3
- package/dist/index.js.map +1 -1
- package/dist/level0/request.d.ts.map +1 -1
- package/dist/level0/request.js +23 -86
- package/dist/level0/request.js.map +1 -1
- package/dist/level1/Agent.d.ts +0 -11
- package/dist/level1/Agent.d.ts.map +1 -1
- package/dist/level1/Agent.js +19 -36
- package/dist/level1/Agent.js.map +1 -1
- package/dist/protocol/ACTPKernel.d.ts +7 -1
- package/dist/protocol/ACTPKernel.d.ts.map +1 -1
- package/dist/protocol/ACTPKernel.js +13 -10
- package/dist/protocol/ACTPKernel.js.map +1 -1
- package/dist/protocol/EventMonitor.d.ts +14 -0
- package/dist/protocol/EventMonitor.d.ts.map +1 -1
- package/dist/protocol/EventMonitor.js +14 -0
- package/dist/protocol/EventMonitor.js.map +1 -1
- package/dist/registry/AgentRegistryClient.d.ts +75 -0
- package/dist/registry/AgentRegistryClient.d.ts.map +1 -0
- package/dist/registry/AgentRegistryClient.js +160 -0
- package/dist/registry/AgentRegistryClient.js.map +1 -0
- package/dist/runtime/BlockchainRuntime.d.ts +5 -0
- package/dist/runtime/BlockchainRuntime.d.ts.map +1 -1
- package/dist/runtime/BlockchainRuntime.js +1 -1
- package/dist/runtime/BlockchainRuntime.js.map +1 -1
- package/dist/storage/ArchiveBundleBuilder.d.ts.map +1 -1
- package/dist/storage/ArchiveBundleBuilder.js.map +1 -1
- package/dist/storage/ArweaveClient.d.ts.map +1 -1
- package/dist/storage/ArweaveClient.js +2 -0
- package/dist/storage/ArweaveClient.js.map +1 -1
- package/dist/storage/FilebaseClient.d.ts.map +1 -1
- package/dist/storage/FilebaseClient.js +2 -0
- package/dist/storage/FilebaseClient.js.map +1 -1
- package/dist/types/adapter.d.ts +39 -0
- package/dist/types/adapter.d.ts.map +1 -1
- package/dist/types/adapter.js +7 -0
- package/dist/types/adapter.js.map +1 -1
- package/dist/types/x402.d.ts +23 -0
- package/dist/types/x402.d.ts.map +1 -1
- package/dist/types/x402.js.map +1 -1
- package/dist/utils/ErrorRecoveryGuide.d.ts.map +1 -1
- package/dist/utils/ErrorRecoveryGuide.js +3 -2
- package/dist/utils/ErrorRecoveryGuide.js.map +1 -1
- package/dist/utils/IPFSClient.d.ts +3 -2
- package/dist/utils/IPFSClient.d.ts.map +1 -1
- package/dist/utils/IPFSClient.js +7 -5
- package/dist/utils/IPFSClient.js.map +1 -1
- package/dist/utils/computeTypeHash.js +1 -3
- package/dist/utils/computeTypeHash.js.map +1 -1
- package/dist/utils/retry.d.ts.map +1 -1
- package/dist/utils/retry.js +0 -1
- package/dist/utils/retry.js.map +1 -1
- package/dist/utils/validation.d.ts +2 -2
- package/dist/utils/validation.d.ts.map +1 -1
- package/dist/utils/validation.js +2 -2
- package/dist/utils/validation.js.map +1 -1
- package/dist/wallet/AutoWalletProvider.d.ts +77 -0
- package/dist/wallet/AutoWalletProvider.d.ts.map +1 -0
- package/dist/wallet/AutoWalletProvider.js +197 -0
- package/dist/wallet/AutoWalletProvider.js.map +1 -0
- package/dist/wallet/EOAWalletProvider.d.ts +21 -0
- package/dist/wallet/EOAWalletProvider.d.ts.map +1 -0
- package/dist/wallet/EOAWalletProvider.js +57 -0
- package/dist/wallet/EOAWalletProvider.js.map +1 -0
- package/dist/wallet/IWalletProvider.d.ts +115 -0
- package/dist/wallet/IWalletProvider.d.ts.map +1 -0
- package/dist/wallet/IWalletProvider.js +12 -0
- package/dist/wallet/IWalletProvider.js.map +1 -0
- package/dist/wallet/aa/BundlerClient.d.ts +70 -0
- package/dist/wallet/aa/BundlerClient.d.ts.map +1 -0
- package/dist/wallet/aa/BundlerClient.js +183 -0
- package/dist/wallet/aa/BundlerClient.js.map +1 -0
- package/dist/wallet/aa/DualNonceManager.d.ts +55 -0
- package/dist/wallet/aa/DualNonceManager.d.ts.map +1 -0
- package/dist/wallet/aa/DualNonceManager.js +131 -0
- package/dist/wallet/aa/DualNonceManager.js.map +1 -0
- package/dist/wallet/aa/PaymasterClient.d.ts +52 -0
- package/dist/wallet/aa/PaymasterClient.d.ts.map +1 -0
- package/dist/wallet/aa/PaymasterClient.js +115 -0
- package/dist/wallet/aa/PaymasterClient.js.map +1 -0
- package/dist/wallet/aa/TransactionBatcher.d.ts +87 -0
- package/dist/wallet/aa/TransactionBatcher.d.ts.map +1 -0
- package/dist/wallet/aa/TransactionBatcher.js +148 -0
- package/dist/wallet/aa/TransactionBatcher.js.map +1 -0
- package/dist/wallet/aa/UserOpBuilder.d.ts +71 -0
- package/dist/wallet/aa/UserOpBuilder.d.ts.map +1 -0
- package/dist/wallet/aa/UserOpBuilder.js +196 -0
- package/dist/wallet/aa/UserOpBuilder.js.map +1 -0
- package/dist/wallet/aa/constants.d.ts +54 -0
- package/dist/wallet/aa/constants.d.ts.map +1 -0
- package/dist/wallet/aa/constants.js +18 -0
- package/dist/wallet/aa/constants.js.map +1 -0
- package/dist/wallet/keystore.d.ts +16 -0
- package/dist/wallet/keystore.d.ts.map +1 -0
- package/dist/wallet/keystore.js +132 -0
- package/dist/wallet/keystore.js.map +1 -0
- package/package.json +5 -2
- package/src/ACTPClient.ts +275 -27
- package/src/abi/AgentRegistry.json +133 -0
- package/src/adapters/AdapterRouter.ts +0 -1
- package/src/adapters/BasicAdapter.ts +41 -1
- package/src/adapters/X402Adapter.ts +94 -16
- package/src/adapters/index.ts +9 -1
- package/src/cli/commands/diff.ts +141 -0
- package/src/cli/commands/init.ts +311 -22
- package/src/cli/commands/publish.ts +208 -0
- package/src/cli/commands/pull.ts +124 -0
- package/src/cli/commands/register.ts +233 -0
- package/src/cli/index.ts +12 -0
- package/src/cli/utils/config.ts +9 -0
- package/src/config/agirailsmd.ts +262 -0
- package/src/config/networks.ts +89 -26
- package/src/config/publishPipeline.ts +276 -0
- package/src/config/syncOperations.ts +279 -0
- package/src/erc8004/ERC8004Bridge.ts +6 -5
- package/src/erc8004/ReputationReporter.ts +14 -18
- package/src/index.ts +15 -0
- package/src/level0/request.ts +27 -88
- package/src/level1/Agent.ts +21 -37
- package/src/protocol/ACTPKernel.ts +20 -10
- package/src/protocol/EventMonitor.ts +14 -0
- package/src/registry/AgentRegistryClient.ts +202 -0
- package/src/runtime/BlockchainRuntime.ts +7 -1
- package/src/storage/ArchiveBundleBuilder.ts +0 -2
- package/src/storage/ArweaveClient.ts +2 -1
- package/src/storage/FilebaseClient.ts +3 -3
- package/src/types/adapter.ts +14 -0
- package/src/types/x402.ts +32 -0
- package/src/utils/ErrorRecoveryGuide.ts +4 -2
- package/src/utils/IPFSClient.ts +9 -7
- package/src/utils/computeTypeHash.ts +1 -3
- package/src/utils/retry.ts +0 -1
- package/src/utils/validation.ts +2 -2
- package/src/wallet/AutoWalletProvider.ts +294 -0
- package/src/wallet/EOAWalletProvider.ts +69 -0
- package/src/wallet/IWalletProvider.ts +133 -0
- package/src/wallet/aa/BundlerClient.ts +273 -0
- package/src/wallet/aa/DualNonceManager.ts +163 -0
- package/src/wallet/aa/PaymasterClient.ts +173 -0
- package/src/wallet/aa/TransactionBatcher.ts +240 -0
- package/src/wallet/aa/UserOpBuilder.ts +246 -0
- package/src/wallet/aa/constants.ts +60 -0
- package/src/wallet/keystore.ts +119 -0
|
@@ -0,0 +1,294 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* AutoWalletProvider — Tier 1 (Auto) wallet implementation.
|
|
3
|
+
*
|
|
4
|
+
* Creates a CoinbaseSmartWallet with gas-sponsored transactions:
|
|
5
|
+
* 1. Load/generate local encrypted key
|
|
6
|
+
* 2. Derive Smart Wallet address (counterfactual CREATE2)
|
|
7
|
+
* 3. Check AgentRegistry registration
|
|
8
|
+
* 4. Build UserOp with executeBatch
|
|
9
|
+
* 5. Get paymaster sponsorship (Coinbase → Pimlico fallback)
|
|
10
|
+
* 6. Submit via bundler
|
|
11
|
+
*
|
|
12
|
+
* If agent is NOT registered, falls back to EOA behavior with a warning.
|
|
13
|
+
*
|
|
14
|
+
* @module wallet/AutoWalletProvider
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
import { ethers } from 'ethers';
|
|
18
|
+
import {
|
|
19
|
+
IWalletProvider,
|
|
20
|
+
TransactionRequest,
|
|
21
|
+
TransactionReceipt,
|
|
22
|
+
WalletInfo,
|
|
23
|
+
BatchedPayParams,
|
|
24
|
+
BatchedPayResult,
|
|
25
|
+
} from './IWalletProvider';
|
|
26
|
+
import { computeSmartWalletAddress } from './aa/UserOpBuilder';
|
|
27
|
+
import {
|
|
28
|
+
buildUserOp,
|
|
29
|
+
signUserOp,
|
|
30
|
+
} from './aa/UserOpBuilder';
|
|
31
|
+
import { SmartWalletCall } from './aa/constants';
|
|
32
|
+
import { BundlerClient } from './aa/BundlerClient';
|
|
33
|
+
import { PaymasterClient } from './aa/PaymasterClient';
|
|
34
|
+
import { DualNonceManager } from './aa/DualNonceManager';
|
|
35
|
+
import { buildACTPPayBatch } from './aa/TransactionBatcher';
|
|
36
|
+
import { sdkLogger } from '../utils/Logger';
|
|
37
|
+
|
|
38
|
+
// ============================================================================
|
|
39
|
+
// Types
|
|
40
|
+
// ============================================================================
|
|
41
|
+
|
|
42
|
+
export interface AutoWalletConfig {
|
|
43
|
+
/** Ethers signer (the EOA that owns the Smart Wallet) */
|
|
44
|
+
signer: ethers.Wallet;
|
|
45
|
+
/** Ethers provider */
|
|
46
|
+
provider: ethers.JsonRpcProvider;
|
|
47
|
+
/** Chain ID */
|
|
48
|
+
chainId: number;
|
|
49
|
+
/** ACTPKernel contract address (for ACTP nonce reads) */
|
|
50
|
+
actpKernelAddress: string;
|
|
51
|
+
/** Bundler configuration */
|
|
52
|
+
bundler: {
|
|
53
|
+
primaryUrl: string;
|
|
54
|
+
backupUrl?: string;
|
|
55
|
+
};
|
|
56
|
+
/** Paymaster configuration */
|
|
57
|
+
paymaster: {
|
|
58
|
+
primaryUrl: string;
|
|
59
|
+
backupUrl?: string;
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// ============================================================================
|
|
64
|
+
// AutoWalletProvider
|
|
65
|
+
// ============================================================================
|
|
66
|
+
|
|
67
|
+
export class AutoWalletProvider implements IWalletProvider {
|
|
68
|
+
private readonly signer: ethers.Wallet;
|
|
69
|
+
private readonly provider: ethers.JsonRpcProvider;
|
|
70
|
+
private readonly chainId: number;
|
|
71
|
+
private readonly bundler: BundlerClient;
|
|
72
|
+
private readonly paymaster: PaymasterClient;
|
|
73
|
+
private readonly nonceManager: DualNonceManager;
|
|
74
|
+
private smartWalletAddress: string = '';
|
|
75
|
+
private isDeployed: boolean = false;
|
|
76
|
+
|
|
77
|
+
private constructor(
|
|
78
|
+
config: AutoWalletConfig,
|
|
79
|
+
smartWalletAddress: string,
|
|
80
|
+
isDeployed: boolean
|
|
81
|
+
) {
|
|
82
|
+
this.signer = config.signer;
|
|
83
|
+
this.provider = config.provider;
|
|
84
|
+
this.chainId = config.chainId;
|
|
85
|
+
this.smartWalletAddress = smartWalletAddress;
|
|
86
|
+
this.isDeployed = isDeployed;
|
|
87
|
+
|
|
88
|
+
this.bundler = new BundlerClient({
|
|
89
|
+
primaryUrl: config.bundler.primaryUrl,
|
|
90
|
+
backupUrl: config.bundler.backupUrl,
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
this.paymaster = new PaymasterClient({
|
|
94
|
+
primaryUrl: config.paymaster.primaryUrl,
|
|
95
|
+
backupUrl: config.paymaster.backupUrl,
|
|
96
|
+
chainId: config.chainId,
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
this.nonceManager = new DualNonceManager(
|
|
100
|
+
config.provider,
|
|
101
|
+
smartWalletAddress,
|
|
102
|
+
config.actpKernelAddress
|
|
103
|
+
);
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Factory method — computes counterfactual address and checks deployment.
|
|
108
|
+
*/
|
|
109
|
+
static async create(config: AutoWalletConfig): Promise<AutoWalletProvider> {
|
|
110
|
+
const signerAddress = config.signer.address;
|
|
111
|
+
const smartWalletAddress = await computeSmartWalletAddress(
|
|
112
|
+
signerAddress,
|
|
113
|
+
config.provider
|
|
114
|
+
);
|
|
115
|
+
|
|
116
|
+
// Check if wallet is already deployed
|
|
117
|
+
const code = await config.provider.getCode(smartWalletAddress);
|
|
118
|
+
const isDeployed = code !== '0x';
|
|
119
|
+
|
|
120
|
+
sdkLogger.info('AutoWalletProvider initialized', {
|
|
121
|
+
signer: signerAddress,
|
|
122
|
+
smartWallet: smartWalletAddress,
|
|
123
|
+
deployed: isDeployed,
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
return new AutoWalletProvider(config, smartWalletAddress, isDeployed);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
getAddress(): string {
|
|
130
|
+
return this.smartWalletAddress;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
async sendTransaction(tx: TransactionRequest): Promise<TransactionReceipt> {
|
|
134
|
+
return this.sendBatchTransaction([tx]);
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
async sendBatchTransaction(
|
|
138
|
+
txs: TransactionRequest[]
|
|
139
|
+
): Promise<TransactionReceipt> {
|
|
140
|
+
if (txs.length === 0) {
|
|
141
|
+
throw new Error('sendBatchTransaction requires at least one transaction');
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// Convert to SmartWalletCall[]
|
|
145
|
+
const calls: SmartWalletCall[] = txs.map((tx) => ({
|
|
146
|
+
target: tx.to,
|
|
147
|
+
value: tx.value ? BigInt(tx.value) : 0n,
|
|
148
|
+
data: tx.data,
|
|
149
|
+
}));
|
|
150
|
+
|
|
151
|
+
// Use nonce manager for sequential ordering
|
|
152
|
+
// Note: We don't know if this specific batch increments ACTP nonce
|
|
153
|
+
// (that's handled at the ACTPClient level), so pass false here.
|
|
154
|
+
// The caller (ACTPClient.payBatched) manages ACTP nonce tracking.
|
|
155
|
+
return this.nonceManager.enqueue(
|
|
156
|
+
async ({ entryPointNonce }) => {
|
|
157
|
+
const receipt = await this.submitUserOp(calls, entryPointNonce);
|
|
158
|
+
return { result: receipt, success: receipt.success };
|
|
159
|
+
},
|
|
160
|
+
false // ACTP nonce management is done at ACTPClient level
|
|
161
|
+
);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
getWalletInfo(): WalletInfo {
|
|
165
|
+
return {
|
|
166
|
+
address: this.smartWalletAddress,
|
|
167
|
+
tier: 'auto',
|
|
168
|
+
supportsBatching: true,
|
|
169
|
+
gasSponsored: true,
|
|
170
|
+
chainId: this.chainId,
|
|
171
|
+
};
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
/**
|
|
175
|
+
* Get the DualNonceManager (for ACTPClient to use for ACTP-aware batching).
|
|
176
|
+
*/
|
|
177
|
+
getNonceManager(): DualNonceManager {
|
|
178
|
+
return this.nonceManager;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* Check if the Smart Wallet is deployed on-chain.
|
|
183
|
+
*/
|
|
184
|
+
getIsDeployed(): boolean {
|
|
185
|
+
return this.isDeployed;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
/**
|
|
189
|
+
* Execute a batched ACTP payment atomically.
|
|
190
|
+
*
|
|
191
|
+
* Builds approve + createTransaction + linkEscrow as a single UserOp.
|
|
192
|
+
* Manages ACTP nonce inside the mutex queue for concurrent safety.
|
|
193
|
+
*/
|
|
194
|
+
async payACTPBatched(params: BatchedPayParams): Promise<BatchedPayResult> {
|
|
195
|
+
return this.nonceManager.enqueue(
|
|
196
|
+
async ({ entryPointNonce, actpNonce }) => {
|
|
197
|
+
const batch = buildACTPPayBatch({
|
|
198
|
+
...params,
|
|
199
|
+
actpNonce,
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
const receipt = await this.submitUserOp(batch.calls, entryPointNonce);
|
|
203
|
+
|
|
204
|
+
return {
|
|
205
|
+
result: {
|
|
206
|
+
txId: batch.txId,
|
|
207
|
+
hash: receipt.hash,
|
|
208
|
+
success: receipt.success,
|
|
209
|
+
},
|
|
210
|
+
success: receipt.success,
|
|
211
|
+
};
|
|
212
|
+
},
|
|
213
|
+
true // Increment ACTP nonce on success
|
|
214
|
+
);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
// ==========================================================================
|
|
218
|
+
// Internal
|
|
219
|
+
// ==========================================================================
|
|
220
|
+
|
|
221
|
+
/**
|
|
222
|
+
* Build, sponsor, sign, and submit a UserOp.
|
|
223
|
+
*/
|
|
224
|
+
private async submitUserOp(
|
|
225
|
+
calls: SmartWalletCall[],
|
|
226
|
+
entryPointNonce: bigint
|
|
227
|
+
): Promise<TransactionReceipt> {
|
|
228
|
+
const signerAddress = this.signer.address;
|
|
229
|
+
|
|
230
|
+
// 1. Build unsigned UserOp
|
|
231
|
+
const userOp = buildUserOp({
|
|
232
|
+
sender: this.smartWalletAddress,
|
|
233
|
+
nonce: entryPointNonce,
|
|
234
|
+
calls,
|
|
235
|
+
isFirstDeploy: !this.isDeployed,
|
|
236
|
+
signerAddress,
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
// 2. Get fee data
|
|
240
|
+
const feeData = await this.provider.getFeeData();
|
|
241
|
+
userOp.maxFeePerGas = feeData.maxFeePerGas ?? 2_000_000_000n;
|
|
242
|
+
userOp.maxPriorityFeePerGas = feeData.maxPriorityFeePerGas ?? 1_000_000_000n;
|
|
243
|
+
|
|
244
|
+
// 3. Get stub paymaster data (for gas estimation)
|
|
245
|
+
const stubData = await this.paymaster.getPaymasterStubData(userOp);
|
|
246
|
+
userOp.paymasterAndData = stubData.paymasterAndData;
|
|
247
|
+
|
|
248
|
+
// 4. Dummy signature for gas estimation
|
|
249
|
+
userOp.signature = dummySignature();
|
|
250
|
+
|
|
251
|
+
// 5. Estimate gas
|
|
252
|
+
const gasEstimate = await this.bundler.estimateUserOperationGas(userOp);
|
|
253
|
+
userOp.callGasLimit = gasEstimate.callGasLimit;
|
|
254
|
+
userOp.verificationGasLimit = gasEstimate.verificationGasLimit;
|
|
255
|
+
userOp.preVerificationGas = gasEstimate.preVerificationGas;
|
|
256
|
+
|
|
257
|
+
// 6. Get final paymaster data (with real gas values)
|
|
258
|
+
const finalPaymaster = await this.paymaster.getPaymasterData(userOp);
|
|
259
|
+
userOp.paymasterAndData = finalPaymaster.paymasterAndData;
|
|
260
|
+
|
|
261
|
+
// 7. Sign
|
|
262
|
+
userOp.signature = await signUserOp(userOp, this.signer, this.chainId);
|
|
263
|
+
|
|
264
|
+
// 8. Submit
|
|
265
|
+
const userOpHash = await this.bundler.sendUserOperation(userOp);
|
|
266
|
+
sdkLogger.info('UserOp submitted', { userOpHash });
|
|
267
|
+
|
|
268
|
+
// 9. Wait for receipt
|
|
269
|
+
const receipt = await this.bundler.waitForReceipt(userOpHash);
|
|
270
|
+
|
|
271
|
+
// Mark as deployed after first successful UserOp
|
|
272
|
+
if (!this.isDeployed && receipt.success) {
|
|
273
|
+
this.isDeployed = true;
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
return {
|
|
277
|
+
hash: receipt.transactionHash,
|
|
278
|
+
success: receipt.success,
|
|
279
|
+
};
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
/**
|
|
284
|
+
* Dummy signature for gas estimation.
|
|
285
|
+
* CoinbaseSmartWallet expects abi.encode(uint256 ownerIndex, bytes sig)
|
|
286
|
+
* where sig is 65 bytes (r,s,v).
|
|
287
|
+
*/
|
|
288
|
+
function dummySignature(): string {
|
|
289
|
+
const dummySig = '0x' + 'ff'.repeat(65);
|
|
290
|
+
return ethers.AbiCoder.defaultAbiCoder().encode(
|
|
291
|
+
['uint256', 'bytes'],
|
|
292
|
+
[0, dummySig]
|
|
293
|
+
);
|
|
294
|
+
}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* EOAWalletProvider — Tier 2 (BYOW) wallet implementation.
|
|
3
|
+
*
|
|
4
|
+
* Wraps an ethers.Wallet for traditional EOA signing.
|
|
5
|
+
* sendBatchTransaction() executes calls sequentially (no atomic batching).
|
|
6
|
+
* This is the backward-compatible path for agents that don't register.
|
|
7
|
+
*
|
|
8
|
+
* @module wallet/EOAWalletProvider
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { ethers } from 'ethers';
|
|
12
|
+
import {
|
|
13
|
+
IWalletProvider,
|
|
14
|
+
TransactionRequest,
|
|
15
|
+
TransactionReceipt,
|
|
16
|
+
WalletInfo,
|
|
17
|
+
} from './IWalletProvider';
|
|
18
|
+
|
|
19
|
+
export class EOAWalletProvider implements IWalletProvider {
|
|
20
|
+
private readonly wallet: ethers.Wallet;
|
|
21
|
+
private readonly chainId: number;
|
|
22
|
+
|
|
23
|
+
constructor(wallet: ethers.Wallet, chainId: number) {
|
|
24
|
+
this.wallet = wallet;
|
|
25
|
+
this.chainId = chainId;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
getAddress(): string {
|
|
29
|
+
return this.wallet.address;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
async sendTransaction(tx: TransactionRequest): Promise<TransactionReceipt> {
|
|
33
|
+
const response = await this.wallet.sendTransaction({
|
|
34
|
+
to: tx.to,
|
|
35
|
+
data: tx.data,
|
|
36
|
+
value: tx.value ? BigInt(tx.value) : 0n,
|
|
37
|
+
});
|
|
38
|
+
const receipt = await response.wait();
|
|
39
|
+
return {
|
|
40
|
+
hash: receipt!.hash,
|
|
41
|
+
success: receipt!.status === 1,
|
|
42
|
+
};
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
async sendBatchTransaction(txs: TransactionRequest[]): Promise<TransactionReceipt> {
|
|
46
|
+
if (txs.length === 0) {
|
|
47
|
+
throw new Error('sendBatchTransaction requires at least one transaction');
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
let lastReceipt: TransactionReceipt | undefined;
|
|
51
|
+
for (const tx of txs) {
|
|
52
|
+
lastReceipt = await this.sendTransaction(tx);
|
|
53
|
+
if (!lastReceipt.success) {
|
|
54
|
+
return lastReceipt; // Fail fast
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
return lastReceipt!;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
getWalletInfo(): WalletInfo {
|
|
61
|
+
return {
|
|
62
|
+
address: this.wallet.address,
|
|
63
|
+
tier: 'eoa',
|
|
64
|
+
supportsBatching: false,
|
|
65
|
+
gasSponsored: false,
|
|
66
|
+
chainId: this.chainId,
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* IWalletProvider — Wallet abstraction for ACTP SDK.
|
|
3
|
+
*
|
|
4
|
+
* Enables pluggable wallet implementations:
|
|
5
|
+
* - EOAWalletProvider (Tier 2): Direct ethers.Wallet signing
|
|
6
|
+
* - AutoWalletProvider (Tier 1): CoinbaseSmartWallet + Paymaster (gasless)
|
|
7
|
+
*
|
|
8
|
+
* @module wallet/IWalletProvider
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
// ============================================================================
|
|
12
|
+
// Types
|
|
13
|
+
// ============================================================================
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Low-level transaction request.
|
|
17
|
+
* Plain strings only — no ethers or viem types at the interface boundary.
|
|
18
|
+
*/
|
|
19
|
+
export interface TransactionRequest {
|
|
20
|
+
/** Target contract address (0x-prefixed) */
|
|
21
|
+
to: string;
|
|
22
|
+
/** Calldata (0x-prefixed hex) */
|
|
23
|
+
data: string;
|
|
24
|
+
/** ETH value in wei (decimal string). Defaults to "0". */
|
|
25
|
+
value?: string;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Transaction receipt returned after submission.
|
|
30
|
+
*/
|
|
31
|
+
export interface TransactionReceipt {
|
|
32
|
+
/** Transaction hash (EOA) or UserOp hash (AA) */
|
|
33
|
+
hash: string;
|
|
34
|
+
/** Whether the transaction succeeded */
|
|
35
|
+
success: boolean;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Wallet tier — determines gas behavior.
|
|
40
|
+
*/
|
|
41
|
+
export type WalletTier = 'auto' | 'eoa';
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Information about the wallet provider.
|
|
45
|
+
*/
|
|
46
|
+
export interface WalletInfo {
|
|
47
|
+
/** The account address used for ACTP transactions */
|
|
48
|
+
address: string;
|
|
49
|
+
/** Wallet tier */
|
|
50
|
+
tier: WalletTier;
|
|
51
|
+
/** Whether this wallet supports atomic batching (AA = true, EOA = false) */
|
|
52
|
+
supportsBatching: boolean;
|
|
53
|
+
/** Whether gas is sponsored (AA with paymaster = true) */
|
|
54
|
+
gasSponsored: boolean;
|
|
55
|
+
/** Chain ID */
|
|
56
|
+
chainId: number;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
/**
|
|
60
|
+
* Result of a batched ACTP payment via AA wallet.
|
|
61
|
+
*/
|
|
62
|
+
export interface BatchedPayResult {
|
|
63
|
+
/** Pre-computed ACTP transaction ID (bytes32) */
|
|
64
|
+
txId: string;
|
|
65
|
+
/** Transaction hash from the UserOp */
|
|
66
|
+
hash: string;
|
|
67
|
+
/** Whether the UserOp succeeded */
|
|
68
|
+
success: boolean;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
/**
|
|
72
|
+
* Parameters for batched ACTP payment.
|
|
73
|
+
*/
|
|
74
|
+
export interface BatchedPayParams {
|
|
75
|
+
provider: string;
|
|
76
|
+
requester: string;
|
|
77
|
+
amount: string;
|
|
78
|
+
deadline: number;
|
|
79
|
+
disputeWindow: number;
|
|
80
|
+
serviceHash: string;
|
|
81
|
+
agentId: string;
|
|
82
|
+
contracts: {
|
|
83
|
+
usdc: string;
|
|
84
|
+
actpKernel: string;
|
|
85
|
+
escrowVault: string;
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// ============================================================================
|
|
90
|
+
// Interface
|
|
91
|
+
// ============================================================================
|
|
92
|
+
|
|
93
|
+
/**
|
|
94
|
+
* Wallet provider interface.
|
|
95
|
+
*
|
|
96
|
+
* All methods use plain strings (no ethers/viem types) to keep the
|
|
97
|
+
* interface decoupled from any specific library.
|
|
98
|
+
*/
|
|
99
|
+
export interface IWalletProvider {
|
|
100
|
+
/**
|
|
101
|
+
* Get the account address used as requester in ACTP transactions.
|
|
102
|
+
* For AA wallets this is the Smart Wallet address (not the signer EOA).
|
|
103
|
+
*/
|
|
104
|
+
getAddress(): string;
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Send a single transaction.
|
|
108
|
+
*/
|
|
109
|
+
sendTransaction(tx: TransactionRequest): Promise<TransactionReceipt>;
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Send multiple transactions atomically (AA) or sequentially (EOA).
|
|
113
|
+
*
|
|
114
|
+
* AA wallets batch all calls into a single UserOp.
|
|
115
|
+
* EOA wallets execute them one by one and return the last receipt.
|
|
116
|
+
*/
|
|
117
|
+
sendBatchTransaction(txs: TransactionRequest[]): Promise<TransactionReceipt>;
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Get wallet metadata.
|
|
121
|
+
*/
|
|
122
|
+
getWalletInfo(): WalletInfo;
|
|
123
|
+
|
|
124
|
+
/**
|
|
125
|
+
* Execute a batched ACTP payment atomically (approve + createTx + linkEscrow).
|
|
126
|
+
*
|
|
127
|
+
* Only available on AA wallets (supportsBatching: true).
|
|
128
|
+
* Handles ACTP nonce management internally via the DualNonceManager mutex.
|
|
129
|
+
*
|
|
130
|
+
* Returns undefined if batched payments are not supported.
|
|
131
|
+
*/
|
|
132
|
+
payACTPBatched?(params: BatchedPayParams): Promise<BatchedPayResult>;
|
|
133
|
+
}
|