@agirails/sdk 2.0.4 → 2.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 +536 -87
- package/dist/ACTPClient.d.ts +200 -0
- package/dist/ACTPClient.d.ts.map +1 -1
- package/dist/ACTPClient.js +266 -2
- package/dist/ACTPClient.js.map +1 -1
- package/dist/abi/ACTPKernel.json +16 -0
- package/dist/adapters/AdapterRegistry.d.ts +140 -0
- package/dist/adapters/AdapterRegistry.d.ts.map +1 -0
- package/dist/adapters/AdapterRegistry.js +166 -0
- package/dist/adapters/AdapterRegistry.js.map +1 -0
- package/dist/adapters/AdapterRouter.d.ts +165 -0
- package/dist/adapters/AdapterRouter.d.ts.map +1 -0
- package/dist/adapters/AdapterRouter.js +350 -0
- package/dist/adapters/AdapterRouter.js.map +1 -0
- package/dist/adapters/BaseAdapter.d.ts +17 -0
- package/dist/adapters/BaseAdapter.d.ts.map +1 -1
- package/dist/adapters/BaseAdapter.js +21 -0
- package/dist/adapters/BaseAdapter.js.map +1 -1
- package/dist/adapters/BasicAdapter.d.ts +72 -3
- package/dist/adapters/BasicAdapter.d.ts.map +1 -1
- package/dist/adapters/BasicAdapter.js +178 -2
- package/dist/adapters/BasicAdapter.js.map +1 -1
- package/dist/adapters/IAdapter.d.ts +230 -0
- package/dist/adapters/IAdapter.d.ts.map +1 -0
- package/dist/adapters/IAdapter.js +44 -0
- package/dist/adapters/IAdapter.js.map +1 -0
- package/dist/adapters/StandardAdapter.d.ts +80 -6
- package/dist/adapters/StandardAdapter.d.ts.map +1 -1
- package/dist/adapters/StandardAdapter.js +203 -6
- package/dist/adapters/StandardAdapter.js.map +1 -1
- package/dist/adapters/X402Adapter.d.ts +208 -0
- package/dist/adapters/X402Adapter.d.ts.map +1 -0
- package/dist/adapters/X402Adapter.js +423 -0
- package/dist/adapters/X402Adapter.js.map +1 -0
- package/dist/adapters/index.d.ts +8 -0
- package/dist/adapters/index.d.ts.map +1 -1
- package/dist/adapters/index.js +19 -1
- package/dist/adapters/index.js.map +1 -1
- package/dist/cli/commands/init.d.ts +4 -0
- package/dist/cli/commands/init.d.ts.map +1 -1
- package/dist/cli/commands/init.js +146 -4
- package/dist/cli/commands/init.js.map +1 -1
- package/dist/config/networks.d.ts +9 -0
- package/dist/config/networks.d.ts.map +1 -1
- package/dist/config/networks.js +27 -12
- package/dist/config/networks.js.map +1 -1
- package/dist/erc8004/ERC8004Bridge.d.ts +155 -0
- package/dist/erc8004/ERC8004Bridge.d.ts.map +1 -0
- package/dist/erc8004/ERC8004Bridge.js +325 -0
- package/dist/erc8004/ERC8004Bridge.js.map +1 -0
- package/dist/erc8004/ReputationReporter.d.ts +223 -0
- package/dist/erc8004/ReputationReporter.d.ts.map +1 -0
- package/dist/erc8004/ReputationReporter.js +266 -0
- package/dist/erc8004/ReputationReporter.js.map +1 -0
- package/dist/erc8004/index.d.ts +36 -0
- package/dist/erc8004/index.d.ts.map +1 -0
- package/dist/erc8004/index.js +46 -0
- package/dist/erc8004/index.js.map +1 -0
- package/dist/index.d.ts +11 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +51 -2
- package/dist/index.js.map +1 -1
- package/dist/level0/provide.d.ts.map +1 -1
- package/dist/level0/provide.js +2 -1
- package/dist/level0/provide.js.map +1 -1
- package/dist/level1/Agent.d.ts.map +1 -1
- package/dist/level1/Agent.js +11 -3
- package/dist/level1/Agent.js.map +1 -1
- package/dist/protocol/ACTPKernel.d.ts +1 -1
- package/dist/protocol/ACTPKernel.d.ts.map +1 -1
- package/dist/protocol/ACTPKernel.js +23 -12
- package/dist/protocol/ACTPKernel.js.map +1 -1
- package/dist/protocol/DIDResolver.js +1 -1
- package/dist/protocol/DIDResolver.js.map +1 -1
- package/dist/protocol/EASHelper.d.ts.map +1 -1
- package/dist/protocol/EASHelper.js +2 -3
- package/dist/protocol/EASHelper.js.map +1 -1
- package/dist/protocol/MessageSigner.d.ts.map +1 -1
- package/dist/protocol/MessageSigner.js +8 -8
- package/dist/protocol/MessageSigner.js.map +1 -1
- package/dist/runtime/BlockchainRuntime.d.ts +7 -0
- package/dist/runtime/BlockchainRuntime.d.ts.map +1 -1
- package/dist/runtime/BlockchainRuntime.js +40 -22
- package/dist/runtime/BlockchainRuntime.js.map +1 -1
- package/dist/runtime/IACTPRuntime.d.ts +21 -0
- package/dist/runtime/IACTPRuntime.d.ts.map +1 -1
- package/dist/runtime/MockRuntime.d.ts +19 -0
- package/dist/runtime/MockRuntime.d.ts.map +1 -1
- package/dist/runtime/MockRuntime.js +56 -4
- package/dist/runtime/MockRuntime.js.map +1 -1
- package/dist/runtime/types/MockState.d.ts +11 -2
- package/dist/runtime/types/MockState.d.ts.map +1 -1
- package/dist/runtime/types/MockState.js.map +1 -1
- package/dist/storage/ArchiveBundleBuilder.d.ts +150 -0
- package/dist/storage/ArchiveBundleBuilder.d.ts.map +1 -0
- package/dist/storage/ArchiveBundleBuilder.js +468 -0
- package/dist/storage/ArchiveBundleBuilder.js.map +1 -0
- package/dist/storage/ArweaveClient.d.ts +271 -0
- package/dist/storage/ArweaveClient.d.ts.map +1 -0
- package/dist/storage/ArweaveClient.js +761 -0
- package/dist/storage/ArweaveClient.js.map +1 -0
- package/dist/storage/FilebaseClient.d.ts +193 -0
- package/dist/storage/FilebaseClient.d.ts.map +1 -0
- package/dist/storage/FilebaseClient.js +643 -0
- package/dist/storage/FilebaseClient.js.map +1 -0
- package/dist/storage/index.d.ts +47 -0
- package/dist/storage/index.d.ts.map +1 -0
- package/dist/storage/index.js +64 -0
- package/dist/storage/index.js.map +1 -0
- package/dist/storage/types.d.ts +291 -0
- package/dist/storage/types.d.ts.map +1 -0
- package/dist/storage/types.js +18 -0
- package/dist/storage/types.js.map +1 -0
- package/dist/types/adapter.d.ts +359 -0
- package/dist/types/adapter.d.ts.map +1 -0
- package/dist/types/adapter.js +115 -0
- package/dist/types/adapter.js.map +1 -0
- package/dist/types/erc8004.d.ts +184 -0
- package/dist/types/erc8004.d.ts.map +1 -0
- package/dist/types/erc8004.js +132 -0
- package/dist/types/erc8004.js.map +1 -0
- package/dist/types/index.d.ts +3 -0
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/index.js +3 -0
- package/dist/types/index.js.map +1 -1
- package/dist/types/state.d.ts +5 -4
- package/dist/types/state.d.ts.map +1 -1
- package/dist/types/state.js +10 -9
- package/dist/types/state.js.map +1 -1
- package/dist/types/transaction.d.ts +12 -0
- package/dist/types/transaction.d.ts.map +1 -1
- package/dist/types/x402.d.ts +162 -0
- package/dist/types/x402.d.ts.map +1 -0
- package/dist/types/x402.js +162 -0
- package/dist/types/x402.js.map +1 -0
- package/dist/utils/IPFSClient.d.ts.map +1 -1
- package/dist/utils/IPFSClient.js +5 -2
- package/dist/utils/IPFSClient.js.map +1 -1
- package/dist/utils/NonceManager.d.ts.map +1 -1
- package/dist/utils/NonceManager.js +3 -2
- package/dist/utils/NonceManager.js.map +1 -1
- package/dist/utils/UsedAttestationTracker.d.ts.map +1 -1
- package/dist/utils/UsedAttestationTracker.js +3 -3
- package/dist/utils/UsedAttestationTracker.js.map +1 -1
- package/dist/utils/circuitBreaker.d.ts +136 -0
- package/dist/utils/circuitBreaker.d.ts.map +1 -0
- package/dist/utils/circuitBreaker.js +253 -0
- package/dist/utils/circuitBreaker.js.map +1 -0
- package/dist/utils/retry.d.ts +120 -0
- package/dist/utils/retry.d.ts.map +1 -0
- package/dist/utils/retry.js +260 -0
- package/dist/utils/retry.js.map +1 -0
- package/dist/utils/validation.d.ts +100 -0
- package/dist/utils/validation.d.ts.map +1 -1
- package/dist/utils/validation.js +248 -1
- package/dist/utils/validation.js.map +1 -1
- package/package.json +16 -3
- package/src/ACTPClient.ts +318 -2
- package/src/abi/ACTPKernel.json +16 -0
- package/src/adapters/AdapterRegistry.ts +173 -0
- package/src/adapters/AdapterRouter.ts +417 -0
- package/src/adapters/BaseAdapter.ts +25 -0
- package/src/adapters/BasicAdapter.ts +210 -3
- package/src/adapters/IAdapter.ts +292 -0
- package/src/adapters/StandardAdapter.ts +246 -7
- package/src/adapters/X402Adapter.ts +653 -0
- package/src/adapters/index.ts +27 -0
- package/src/cli/commands/init.ts +166 -3
- package/src/config/networks.ts +36 -12
- package/src/erc8004/ERC8004Bridge.ts +461 -0
- package/src/erc8004/ReputationReporter.ts +472 -0
- package/src/erc8004/index.ts +61 -0
- package/src/index.ts +97 -0
- package/src/level0/provide.ts +2 -1
- package/src/level1/Agent.ts +13 -3
- package/src/protocol/ACTPKernel.ts +33 -12
- package/src/protocol/DIDResolver.ts +1 -1
- package/src/protocol/EASHelper.ts +2 -5
- package/src/protocol/MessageSigner.ts +8 -14
- package/src/runtime/BlockchainRuntime.ts +41 -45
- package/src/runtime/IACTPRuntime.ts +22 -0
- package/src/runtime/MockRuntime.ts +58 -4
- package/src/runtime/types/MockState.ts +12 -2
- package/src/storage/ArchiveBundleBuilder.ts +563 -0
- package/src/storage/ArweaveClient.ts +945 -0
- package/src/storage/FilebaseClient.ts +790 -0
- package/src/storage/index.ts +96 -0
- package/src/storage/types.ts +348 -0
- package/src/types/adapter.ts +296 -0
- package/src/types/erc8004.ts +293 -0
- package/src/types/index.ts +3 -0
- package/src/types/state.ts +10 -9
- package/src/types/transaction.ts +12 -0
- package/src/types/x402.ts +219 -0
- package/src/utils/IPFSClient.ts +5 -4
- package/src/utils/NonceManager.ts +3 -2
- package/src/utils/UsedAttestationTracker.ts +3 -5
- package/src/utils/circuitBreaker.ts +324 -0
- package/src/utils/fsSafe.ts +5 -0
- package/src/utils/retry.ts +365 -0
- package/src/utils/validation.ts +295 -1
|
@@ -0,0 +1,461 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* ERC-8004 Identity Bridge
|
|
3
|
+
*
|
|
4
|
+
* READ-ONLY access to ERC-8004 Identity Registry.
|
|
5
|
+
* Resolves agent IDs to wallet addresses for ACTP payments.
|
|
6
|
+
*
|
|
7
|
+
* SECURITY NOTES:
|
|
8
|
+
* - All operations are view functions (no gas costs)
|
|
9
|
+
* - Safe to call without signer
|
|
10
|
+
* - Caches results to minimize RPC calls
|
|
11
|
+
* - Handles network errors gracefully
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* const bridge = new ERC8004Bridge({ network: 'base-sepolia' });
|
|
16
|
+
*
|
|
17
|
+
* // Check if agent exists
|
|
18
|
+
* const exists = await bridge.verifyAgent('12345');
|
|
19
|
+
*
|
|
20
|
+
* // Get wallet for payment
|
|
21
|
+
* const wallet = await bridge.getAgentWallet('12345');
|
|
22
|
+
*
|
|
23
|
+
* // Get full agent info
|
|
24
|
+
* const agent = await bridge.resolveAgent('12345');
|
|
25
|
+
* ```
|
|
26
|
+
*
|
|
27
|
+
* @module erc8004/ERC8004Bridge
|
|
28
|
+
*/
|
|
29
|
+
|
|
30
|
+
import { ethers } from 'ethers';
|
|
31
|
+
import {
|
|
32
|
+
ERC8004Agent,
|
|
33
|
+
ERC8004AgentMetadata,
|
|
34
|
+
ERC8004Network,
|
|
35
|
+
ERC8004Error,
|
|
36
|
+
ERC8004ErrorCode,
|
|
37
|
+
ERC8004_IDENTITY_REGISTRY,
|
|
38
|
+
ERC8004_IDENTITY_ABI,
|
|
39
|
+
ERC8004_DEFAULT_RPC,
|
|
40
|
+
} from '../types/erc8004';
|
|
41
|
+
|
|
42
|
+
// ============================================================================
|
|
43
|
+
// Types
|
|
44
|
+
// ============================================================================
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Interface for ERC-8004 Identity Registry contract.
|
|
48
|
+
* Used for testing with mock implementations.
|
|
49
|
+
*/
|
|
50
|
+
export interface IERC8004IdentityRegistry {
|
|
51
|
+
ownerOf(agentId: string): Promise<string>;
|
|
52
|
+
getAgentURI(agentId: string): Promise<string>;
|
|
53
|
+
balanceOf(owner: string): Promise<bigint>;
|
|
54
|
+
tokenOfOwnerByIndex(owner: string, index: number): Promise<bigint>;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Configuration for ERC8004Bridge.
|
|
59
|
+
*/
|
|
60
|
+
export interface ERC8004BridgeConfig {
|
|
61
|
+
/** Target network */
|
|
62
|
+
network: ERC8004Network;
|
|
63
|
+
|
|
64
|
+
/** Custom RPC URL (optional, uses default if not provided) */
|
|
65
|
+
rpcUrl?: string;
|
|
66
|
+
|
|
67
|
+
/** Override registry address (optional, for testing) */
|
|
68
|
+
registryAddress?: string;
|
|
69
|
+
|
|
70
|
+
/** Custom fetch function (optional, for testing) */
|
|
71
|
+
fetchFn?: typeof fetch;
|
|
72
|
+
|
|
73
|
+
/** Cache TTL in milliseconds (default: 60000 = 1 minute) */
|
|
74
|
+
cacheTimeMs?: number;
|
|
75
|
+
|
|
76
|
+
/** Metadata fetch timeout in milliseconds (default: 10000 = 10 seconds) */
|
|
77
|
+
metadataTimeoutMs?: number;
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Injected contract instance (optional, for testing).
|
|
81
|
+
* If provided, skips creating a real ethers Contract.
|
|
82
|
+
* @internal
|
|
83
|
+
*/
|
|
84
|
+
_testContract?: IERC8004IdentityRegistry;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
/**
|
|
88
|
+
* Cached agent entry.
|
|
89
|
+
*/
|
|
90
|
+
interface CachedAgent {
|
|
91
|
+
agent: ERC8004Agent;
|
|
92
|
+
expiresAt: number;
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// ============================================================================
|
|
96
|
+
// ERC8004Bridge
|
|
97
|
+
// ============================================================================
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Bridge for reading from ERC-8004 Identity Registry.
|
|
101
|
+
*
|
|
102
|
+
* Provides methods to:
|
|
103
|
+
* - Verify agent existence
|
|
104
|
+
* - Get agent wallet address for payments
|
|
105
|
+
* - Resolve full agent info including metadata
|
|
106
|
+
* - List agents owned by an address
|
|
107
|
+
*/
|
|
108
|
+
export class ERC8004Bridge {
|
|
109
|
+
private readonly provider: ethers.JsonRpcProvider | null;
|
|
110
|
+
private readonly registry: IERC8004IdentityRegistry;
|
|
111
|
+
private readonly cache: Map<string, CachedAgent>;
|
|
112
|
+
private readonly fetchFn: typeof fetch;
|
|
113
|
+
private readonly cacheTimeMs: number;
|
|
114
|
+
private readonly metadataTimeoutMs: number;
|
|
115
|
+
private readonly network: ERC8004Network;
|
|
116
|
+
|
|
117
|
+
constructor(config: ERC8004BridgeConfig) {
|
|
118
|
+
this.network = config.network;
|
|
119
|
+
this.fetchFn = config.fetchFn ?? fetch;
|
|
120
|
+
this.cacheTimeMs = config.cacheTimeMs ?? 60000;
|
|
121
|
+
this.metadataTimeoutMs = config.metadataTimeoutMs ?? 10000;
|
|
122
|
+
this.cache = new Map();
|
|
123
|
+
|
|
124
|
+
// Use injected contract for testing
|
|
125
|
+
if (config._testContract) {
|
|
126
|
+
this.provider = null;
|
|
127
|
+
this.registry = config._testContract;
|
|
128
|
+
return;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
// Setup provider
|
|
132
|
+
const rpcUrl = config.rpcUrl ?? ERC8004_DEFAULT_RPC[config.network];
|
|
133
|
+
this.provider = new ethers.JsonRpcProvider(rpcUrl);
|
|
134
|
+
|
|
135
|
+
// Setup registry contract
|
|
136
|
+
const registryAddress =
|
|
137
|
+
config.registryAddress ?? ERC8004_IDENTITY_REGISTRY[config.network];
|
|
138
|
+
|
|
139
|
+
if (registryAddress === ethers.ZeroAddress) {
|
|
140
|
+
console.warn(
|
|
141
|
+
`[ERC8004] Registry not deployed on ${config.network}. Using zero address.`
|
|
142
|
+
);
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
this.registry = new ethers.Contract(
|
|
146
|
+
registryAddress,
|
|
147
|
+
ERC8004_IDENTITY_ABI,
|
|
148
|
+
this.provider
|
|
149
|
+
) as unknown as IERC8004IdentityRegistry;
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
// ==========================================================================
|
|
153
|
+
// Public Methods
|
|
154
|
+
// ==========================================================================
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Verify agent exists in ERC-8004 Identity Registry.
|
|
158
|
+
*
|
|
159
|
+
* Safe to call frequently - uses cache and is a view function.
|
|
160
|
+
*
|
|
161
|
+
* @param agentId - ERC-8004 agent ID (uint256 as string)
|
|
162
|
+
* @returns true if agent exists, false otherwise
|
|
163
|
+
*/
|
|
164
|
+
async verifyAgent(agentId: string): Promise<boolean> {
|
|
165
|
+
// Validate format first
|
|
166
|
+
if (!this.isValidAgentId(agentId)) {
|
|
167
|
+
return false;
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// Check cache
|
|
171
|
+
const cached = this.cache.get(agentId);
|
|
172
|
+
if (cached && cached.expiresAt > Date.now()) {
|
|
173
|
+
return true;
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
try {
|
|
177
|
+
const owner = await this.registry.ownerOf(agentId);
|
|
178
|
+
return owner !== ethers.ZeroAddress;
|
|
179
|
+
} catch {
|
|
180
|
+
// ownerOf reverts for non-existent tokens (ERC-721 standard)
|
|
181
|
+
return false;
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Get wallet address for receiving payments.
|
|
187
|
+
*
|
|
188
|
+
* Wallet priority:
|
|
189
|
+
* 1. metadata.paymentAddress (explicit payment destination)
|
|
190
|
+
* 2. metadata.wallet (alternative field name)
|
|
191
|
+
* 3. owner address (fallback)
|
|
192
|
+
*
|
|
193
|
+
* @param agentId - ERC-8004 agent ID
|
|
194
|
+
* @returns Checksummed wallet address
|
|
195
|
+
* @throws ERC8004Error if agent not found
|
|
196
|
+
*/
|
|
197
|
+
async getAgentWallet(agentId: string): Promise<string> {
|
|
198
|
+
const agent = await this.resolveAgent(agentId);
|
|
199
|
+
return agent.wallet;
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
/**
|
|
203
|
+
* Resolve full agent info from ERC-8004.
|
|
204
|
+
*
|
|
205
|
+
* Fetches on-chain data (owner, agentURI) and off-chain metadata.
|
|
206
|
+
* Results are cached for cacheTimeMs.
|
|
207
|
+
*
|
|
208
|
+
* @param agentId - ERC-8004 agent ID
|
|
209
|
+
* @returns Full agent info including metadata
|
|
210
|
+
* @throws ERC8004Error if agent not found or invalid ID
|
|
211
|
+
*/
|
|
212
|
+
async resolveAgent(agentId: string): Promise<ERC8004Agent> {
|
|
213
|
+
// Check cache first
|
|
214
|
+
const cached = this.cache.get(agentId);
|
|
215
|
+
if (cached && cached.expiresAt > Date.now()) {
|
|
216
|
+
return cached.agent;
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
// Validate format
|
|
220
|
+
if (!this.isValidAgentId(agentId)) {
|
|
221
|
+
throw new ERC8004Error(
|
|
222
|
+
`Invalid agent ID format: "${agentId}"`,
|
|
223
|
+
ERC8004ErrorCode.INVALID_AGENT_ID,
|
|
224
|
+
agentId
|
|
225
|
+
);
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
// Fetch on-chain data
|
|
229
|
+
let owner: string;
|
|
230
|
+
let agentURI: string;
|
|
231
|
+
|
|
232
|
+
try {
|
|
233
|
+
[owner, agentURI] = await Promise.all([
|
|
234
|
+
this.registry.ownerOf(agentId),
|
|
235
|
+
this.registry.getAgentURI(agentId),
|
|
236
|
+
]);
|
|
237
|
+
} catch (error) {
|
|
238
|
+
// Check if it's a "nonexistent token" error
|
|
239
|
+
const isNotFound =
|
|
240
|
+
error instanceof Error &&
|
|
241
|
+
(error.message.includes('nonexistent') ||
|
|
242
|
+
error.message.includes('ERC721') ||
|
|
243
|
+
error.message.includes('invalid token'));
|
|
244
|
+
|
|
245
|
+
if (isNotFound) {
|
|
246
|
+
throw new ERC8004Error(
|
|
247
|
+
`Agent ${agentId} not found in ERC-8004 registry`,
|
|
248
|
+
ERC8004ErrorCode.AGENT_NOT_FOUND,
|
|
249
|
+
agentId
|
|
250
|
+
);
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
throw new ERC8004Error(
|
|
254
|
+
`Failed to fetch agent ${agentId}: ${error instanceof Error ? error.message : 'Unknown error'}`,
|
|
255
|
+
ERC8004ErrorCode.NETWORK_ERROR,
|
|
256
|
+
agentId,
|
|
257
|
+
error instanceof Error ? error : undefined
|
|
258
|
+
);
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
// Check owner is valid
|
|
262
|
+
if (owner === ethers.ZeroAddress) {
|
|
263
|
+
throw new ERC8004Error(
|
|
264
|
+
`Agent ${agentId} not found in ERC-8004 registry`,
|
|
265
|
+
ERC8004ErrorCode.AGENT_NOT_FOUND,
|
|
266
|
+
agentId
|
|
267
|
+
);
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
// Fetch metadata (may return undefined if fetch fails)
|
|
271
|
+
const metadata = await this.fetchMetadata(agentURI, agentId);
|
|
272
|
+
|
|
273
|
+
// Determine wallet address
|
|
274
|
+
// Priority: paymentAddress > wallet > owner
|
|
275
|
+
let walletAddress = owner;
|
|
276
|
+
|
|
277
|
+
if (metadata?.paymentAddress && this.isValidAddress(metadata.paymentAddress)) {
|
|
278
|
+
walletAddress = metadata.paymentAddress;
|
|
279
|
+
} else if (metadata?.wallet && this.isValidAddress(metadata.wallet)) {
|
|
280
|
+
walletAddress = metadata.wallet;
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
// Build agent object
|
|
284
|
+
const agent: ERC8004Agent = {
|
|
285
|
+
agentId,
|
|
286
|
+
owner: ethers.getAddress(owner), // Checksummed
|
|
287
|
+
wallet: ethers.getAddress(walletAddress), // Checksummed
|
|
288
|
+
agentURI,
|
|
289
|
+
metadata,
|
|
290
|
+
network: this.network,
|
|
291
|
+
};
|
|
292
|
+
|
|
293
|
+
// Cache result
|
|
294
|
+
this.cache.set(agentId, {
|
|
295
|
+
agent,
|
|
296
|
+
expiresAt: Date.now() + this.cacheTimeMs,
|
|
297
|
+
});
|
|
298
|
+
|
|
299
|
+
return agent;
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
/**
|
|
303
|
+
* Get all agent IDs owned by an address.
|
|
304
|
+
*
|
|
305
|
+
* @param owner - Owner address to query
|
|
306
|
+
* @returns Array of agent IDs (may be empty)
|
|
307
|
+
*/
|
|
308
|
+
async getAgentsByOwner(owner: string): Promise<string[]> {
|
|
309
|
+
if (!this.isValidAddress(owner)) {
|
|
310
|
+
return [];
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
try {
|
|
314
|
+
const balance = await this.registry.balanceOf(owner);
|
|
315
|
+
const balanceNum = Number(balance);
|
|
316
|
+
|
|
317
|
+
if (balanceNum === 0) {
|
|
318
|
+
return [];
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
const agentIds: string[] = [];
|
|
322
|
+
|
|
323
|
+
for (let i = 0; i < balanceNum; i++) {
|
|
324
|
+
const agentId = await this.registry.tokenOfOwnerByIndex(owner, i);
|
|
325
|
+
agentIds.push(agentId.toString());
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
return agentIds;
|
|
329
|
+
} catch (error) {
|
|
330
|
+
console.warn(`[ERC8004] Failed to get agents for owner ${owner}:`, error);
|
|
331
|
+
return [];
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
/**
|
|
336
|
+
* Clear all cached entries.
|
|
337
|
+
* Useful for testing or forcing fresh data.
|
|
338
|
+
*/
|
|
339
|
+
clearCache(): void {
|
|
340
|
+
this.cache.clear();
|
|
341
|
+
}
|
|
342
|
+
|
|
343
|
+
/**
|
|
344
|
+
* Get cache statistics (for debugging).
|
|
345
|
+
*/
|
|
346
|
+
getCacheStats(): { size: number; network: ERC8004Network } {
|
|
347
|
+
return {
|
|
348
|
+
size: this.cache.size,
|
|
349
|
+
network: this.network,
|
|
350
|
+
};
|
|
351
|
+
}
|
|
352
|
+
|
|
353
|
+
// ==========================================================================
|
|
354
|
+
// Private Methods
|
|
355
|
+
// ==========================================================================
|
|
356
|
+
|
|
357
|
+
/**
|
|
358
|
+
* Validate agent ID format.
|
|
359
|
+
* Agent IDs are uint256 values (0 to 2^256-1).
|
|
360
|
+
*/
|
|
361
|
+
private isValidAgentId(agentId: string): boolean {
|
|
362
|
+
if (!agentId || typeof agentId !== 'string') {
|
|
363
|
+
return false;
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
// Must not look like an Ethereum address or URL
|
|
367
|
+
if (agentId.startsWith('0x') || agentId.includes('://')) {
|
|
368
|
+
return false;
|
|
369
|
+
}
|
|
370
|
+
|
|
371
|
+
try {
|
|
372
|
+
const bn = BigInt(agentId);
|
|
373
|
+
return bn >= 0n && bn < 2n ** 256n;
|
|
374
|
+
} catch {
|
|
375
|
+
return false;
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
/**
|
|
380
|
+
* Validate Ethereum address format.
|
|
381
|
+
*/
|
|
382
|
+
private isValidAddress(address: string): boolean {
|
|
383
|
+
try {
|
|
384
|
+
ethers.getAddress(address);
|
|
385
|
+
return true;
|
|
386
|
+
} catch {
|
|
387
|
+
return false;
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
/**
|
|
392
|
+
* Fetch and parse metadata from agentURI.
|
|
393
|
+
*
|
|
394
|
+
* Handles:
|
|
395
|
+
* - IPFS URIs (ipfs://...)
|
|
396
|
+
* - HTTPS URIs
|
|
397
|
+
* - Timeout protection
|
|
398
|
+
* - Parse errors
|
|
399
|
+
*
|
|
400
|
+
* Returns undefined on any failure (never throws).
|
|
401
|
+
*/
|
|
402
|
+
private async fetchMetadata(
|
|
403
|
+
agentURI: string,
|
|
404
|
+
agentId: string
|
|
405
|
+
): Promise<ERC8004AgentMetadata | undefined> {
|
|
406
|
+
if (!agentURI) {
|
|
407
|
+
return undefined;
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
try {
|
|
411
|
+
// Convert IPFS URI to HTTP gateway
|
|
412
|
+
let url = agentURI;
|
|
413
|
+
if (url.startsWith('ipfs://')) {
|
|
414
|
+
const cid = url.slice(7);
|
|
415
|
+
url = `https://ipfs.io/ipfs/${cid}`;
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
// Validate URL
|
|
419
|
+
if (!url.startsWith('http://') && !url.startsWith('https://')) {
|
|
420
|
+
console.warn(`[ERC8004] Invalid agentURI scheme for ${agentId}: ${url}`);
|
|
421
|
+
return undefined;
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
// Fetch with timeout
|
|
425
|
+
const controller = new AbortController();
|
|
426
|
+
const timeoutId = setTimeout(() => controller.abort(), this.metadataTimeoutMs);
|
|
427
|
+
|
|
428
|
+
try {
|
|
429
|
+
const response = await this.fetchFn(url, {
|
|
430
|
+
headers: { Accept: 'application/json' },
|
|
431
|
+
signal: controller.signal,
|
|
432
|
+
});
|
|
433
|
+
|
|
434
|
+
clearTimeout(timeoutId);
|
|
435
|
+
|
|
436
|
+
if (!response.ok) {
|
|
437
|
+
console.warn(
|
|
438
|
+
`[ERC8004] Metadata fetch failed for ${agentId}: HTTP ${response.status}`
|
|
439
|
+
);
|
|
440
|
+
return undefined;
|
|
441
|
+
}
|
|
442
|
+
|
|
443
|
+
const data = await response.json();
|
|
444
|
+
return data as ERC8004AgentMetadata;
|
|
445
|
+
} finally {
|
|
446
|
+
clearTimeout(timeoutId);
|
|
447
|
+
}
|
|
448
|
+
} catch (error) {
|
|
449
|
+
// Log but don't throw - metadata is optional
|
|
450
|
+
const errorMessage =
|
|
451
|
+
error instanceof Error
|
|
452
|
+
? error.name === 'AbortError'
|
|
453
|
+
? 'timeout'
|
|
454
|
+
: error.message
|
|
455
|
+
: 'unknown error';
|
|
456
|
+
|
|
457
|
+
console.warn(`[ERC8004] Metadata fetch failed for ${agentId}: ${errorMessage}`);
|
|
458
|
+
return undefined;
|
|
459
|
+
}
|
|
460
|
+
}
|
|
461
|
+
}
|