web-agent-bridge 2.4.0 → 2.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.ar.md +18 -0
- package/README.md +18 -0
- package/package.json +1 -1
- package/sdk/index.d.ts +170 -0
- package/sdk/index.js +246 -1
- package/sdk/package.json +1 -1
- package/server/control-plane/index.js +301 -0
- package/server/data-plane/index.js +354 -0
- package/server/index.js +2 -0
- package/server/llm/index.js +404 -0
- package/server/observability/index.js +394 -0
- package/server/protocol/capabilities.js +223 -0
- package/server/protocol/index.js +243 -0
- package/server/protocol/schema.js +584 -0
- package/server/registry/index.js +326 -0
- package/server/routes/runtime.js +725 -0
- package/server/runtime/event-bus.js +210 -0
- package/server/runtime/index.js +233 -0
- package/server/runtime/sandbox.js +266 -0
- package/server/runtime/scheduler.js +395 -0
- package/server/runtime/state-manager.js +188 -0
- package/server/security/index.js +355 -0
|
@@ -0,0 +1,584 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* WAB Protocol (WABP) - Schema Registry
|
|
5
|
+
*
|
|
6
|
+
* Formal JSON Schema definitions for all WAB Protocol commands.
|
|
7
|
+
* Every command in the system must be registered here with:
|
|
8
|
+
* - input/output types (JSON Schema)
|
|
9
|
+
* - required capabilities
|
|
10
|
+
* - permissions
|
|
11
|
+
* - versioning
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
const PROTOCOL_VERSION = '1.0.0';
|
|
15
|
+
|
|
16
|
+
// ─── Primitive Type Schemas ─────────────────────────────────────────────────
|
|
17
|
+
|
|
18
|
+
const Types = {
|
|
19
|
+
AgentId: { type: 'string', pattern: '^agent_[a-zA-Z0-9_-]{4,64}$', description: 'Unique agent identifier' },
|
|
20
|
+
SiteId: { type: 'string', pattern: '^site_[a-zA-Z0-9_-]{4,64}$', description: 'Unique site identifier' },
|
|
21
|
+
TaskId: { type: 'string', pattern: '^task_[a-zA-Z0-9_-]{4,64}$', description: 'Unique task identifier' },
|
|
22
|
+
TraceId: { type: 'string', pattern: '^trace_[a-f0-9]{32}$', description: 'Distributed trace identifier' },
|
|
23
|
+
SpanId: { type: 'string', pattern: '^span_[a-f0-9]{16}$', description: 'Trace span identifier' },
|
|
24
|
+
Timestamp: { type: 'number', description: 'Unix epoch milliseconds' },
|
|
25
|
+
Url: { type: 'string', format: 'uri', description: 'Valid URL' },
|
|
26
|
+
Selector: { type: 'string', minLength: 1, maxLength: 1000, description: 'DOM selector or semantic action reference' },
|
|
27
|
+
Version: { type: 'string', pattern: '^\\d+\\.\\d+\\.\\d+$', description: 'Semver version string' },
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
// ─── Capability Definitions ─────────────────────────────────────────────────
|
|
31
|
+
|
|
32
|
+
const Capabilities = {
|
|
33
|
+
// Browser capabilities
|
|
34
|
+
'browser.read': { description: 'Read page content and DOM', risk: 'low' },
|
|
35
|
+
'browser.click': { description: 'Click elements', risk: 'medium' },
|
|
36
|
+
'browser.fill': { description: 'Fill form fields', risk: 'medium' },
|
|
37
|
+
'browser.navigate': { description: 'Navigate to URLs', risk: 'medium' },
|
|
38
|
+
'browser.scroll': { description: 'Scroll page', risk: 'low' },
|
|
39
|
+
'browser.screenshot': { description: 'Capture screenshots', risk: 'low' },
|
|
40
|
+
'browser.execute': { description: 'Execute registered actions', risk: 'high' },
|
|
41
|
+
|
|
42
|
+
// Data capabilities
|
|
43
|
+
'data.extract': { description: 'Extract structured data', risk: 'low' },
|
|
44
|
+
'data.compare': { description: 'Compare data across sources', risk: 'low' },
|
|
45
|
+
'data.store': { description: 'Store data persistently', risk: 'medium' },
|
|
46
|
+
|
|
47
|
+
// Agent capabilities
|
|
48
|
+
'agent.spawn': { description: 'Create child agents', risk: 'high' },
|
|
49
|
+
'agent.communicate': { description: 'Send messages to other agents', risk: 'medium' },
|
|
50
|
+
'agent.delegate': { description: 'Delegate tasks to other agents', risk: 'high' },
|
|
51
|
+
|
|
52
|
+
// System capabilities
|
|
53
|
+
'system.api': { description: 'Make external API calls', risk: 'high' },
|
|
54
|
+
'system.webhook': { description: 'Trigger webhooks', risk: 'high' },
|
|
55
|
+
'system.schedule': { description: 'Schedule future tasks', risk: 'medium' },
|
|
56
|
+
|
|
57
|
+
// Commerce capabilities
|
|
58
|
+
'commerce.price': { description: 'Access pricing data', risk: 'low' },
|
|
59
|
+
'commerce.negotiate': { description: 'Negotiate prices', risk: 'high' },
|
|
60
|
+
'commerce.purchase': { description: 'Execute purchases', risk: 'critical' },
|
|
61
|
+
|
|
62
|
+
// AI capabilities
|
|
63
|
+
'ai.infer': { description: 'Run LLM inference', risk: 'medium' },
|
|
64
|
+
'ai.vision': { description: 'Visual analysis', risk: 'low' },
|
|
65
|
+
'ai.embed': { description: 'Generate embeddings', risk: 'low' },
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
// ─── Permission Levels ──────────────────────────────────────────────────────
|
|
69
|
+
|
|
70
|
+
const PermissionLevels = {
|
|
71
|
+
none: 0,
|
|
72
|
+
read: 1,
|
|
73
|
+
write: 2,
|
|
74
|
+
execute: 3,
|
|
75
|
+
admin: 4,
|
|
76
|
+
owner: 5,
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
// ─── Command Schema Registry ────────────────────────────────────────────────
|
|
80
|
+
|
|
81
|
+
const _commands = new Map();
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Command definition structure:
|
|
85
|
+
* {
|
|
86
|
+
* name: string,
|
|
87
|
+
* version: string,
|
|
88
|
+
* description: string,
|
|
89
|
+
* category: string,
|
|
90
|
+
* capabilities: string[],
|
|
91
|
+
* permission: string,
|
|
92
|
+
* input: JSONSchema,
|
|
93
|
+
* output: JSONSchema,
|
|
94
|
+
* errors: { code: number, message: string }[],
|
|
95
|
+
* idempotent: boolean,
|
|
96
|
+
* timeout: number, // ms, 0 = no timeout
|
|
97
|
+
* }
|
|
98
|
+
*/
|
|
99
|
+
|
|
100
|
+
function registerCommand(def) {
|
|
101
|
+
if (!def.name || !def.version) throw new Error('Command requires name and version');
|
|
102
|
+
if (!def.input || !def.output) throw new Error('Command requires input/output schemas');
|
|
103
|
+
const key = `${def.name}@${def.version}`;
|
|
104
|
+
if (_commands.has(key)) throw new Error(`Command ${key} already registered`);
|
|
105
|
+
|
|
106
|
+
const command = {
|
|
107
|
+
name: def.name,
|
|
108
|
+
version: def.version || '1.0.0',
|
|
109
|
+
description: def.description || '',
|
|
110
|
+
category: def.category || 'general',
|
|
111
|
+
capabilities: def.capabilities || [],
|
|
112
|
+
permission: def.permission || 'execute',
|
|
113
|
+
input: def.input,
|
|
114
|
+
output: def.output,
|
|
115
|
+
errors: def.errors || [],
|
|
116
|
+
idempotent: def.idempotent !== false,
|
|
117
|
+
timeout: def.timeout || 30000,
|
|
118
|
+
deprecated: def.deprecated || false,
|
|
119
|
+
since: def.since || PROTOCOL_VERSION,
|
|
120
|
+
};
|
|
121
|
+
|
|
122
|
+
_commands.set(key, command);
|
|
123
|
+
// Also register as latest
|
|
124
|
+
_commands.set(def.name, command);
|
|
125
|
+
return command;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
function getCommand(name, version) {
|
|
129
|
+
const key = version ? `${name}@${version}` : name;
|
|
130
|
+
return _commands.get(key) || null;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
function listCommands(category) {
|
|
134
|
+
const all = [];
|
|
135
|
+
for (const [key, cmd] of _commands) {
|
|
136
|
+
if (key.includes('@')) continue; // skip versioned duplicates
|
|
137
|
+
if (!category || cmd.category === category) all.push(cmd);
|
|
138
|
+
}
|
|
139
|
+
return all;
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
function validateInput(commandName, data, version) {
|
|
143
|
+
const cmd = getCommand(commandName, version);
|
|
144
|
+
if (!cmd) return { valid: false, errors: [{ path: '', message: `Unknown command: ${commandName}` }] };
|
|
145
|
+
return _validateSchema(data, cmd.input);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
function validateOutput(commandName, data, version) {
|
|
149
|
+
const cmd = getCommand(commandName, version);
|
|
150
|
+
if (!cmd) return { valid: false, errors: [{ path: '', message: `Unknown command: ${commandName}` }] };
|
|
151
|
+
return _validateSchema(data, cmd.output);
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// ─── Lightweight JSON Schema Validator ──────────────────────────────────────
|
|
155
|
+
|
|
156
|
+
function _validateSchema(data, schema, path = '') {
|
|
157
|
+
const errors = [];
|
|
158
|
+
|
|
159
|
+
if (schema.type) {
|
|
160
|
+
const actualType = Array.isArray(data) ? 'array' : (data === null ? 'null' : typeof data);
|
|
161
|
+
if (schema.type !== actualType) {
|
|
162
|
+
errors.push({ path, message: `Expected ${schema.type}, got ${actualType}` });
|
|
163
|
+
return { valid: false, errors };
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (schema.type === 'object' && schema.properties) {
|
|
168
|
+
if (schema.required) {
|
|
169
|
+
for (const req of schema.required) {
|
|
170
|
+
if (data[req] === undefined) {
|
|
171
|
+
errors.push({ path: `${path}.${req}`, message: `Required field missing: ${req}` });
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
if (schema.additionalProperties === false) {
|
|
176
|
+
const allowed = new Set(Object.keys(schema.properties));
|
|
177
|
+
for (const key of Object.keys(data)) {
|
|
178
|
+
if (!allowed.has(key)) {
|
|
179
|
+
errors.push({ path: `${path}.${key}`, message: `Unexpected field: ${key}` });
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
for (const [key, propSchema] of Object.entries(schema.properties)) {
|
|
184
|
+
if (data[key] !== undefined) {
|
|
185
|
+
const sub = _validateSchema(data[key], propSchema, `${path}.${key}`);
|
|
186
|
+
errors.push(...sub.errors);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
if (schema.type === 'array' && schema.items && Array.isArray(data)) {
|
|
192
|
+
for (let i = 0; i < data.length; i++) {
|
|
193
|
+
const sub = _validateSchema(data[i], schema.items, `${path}[${i}]`);
|
|
194
|
+
errors.push(...sub.errors);
|
|
195
|
+
}
|
|
196
|
+
if (schema.minItems && data.length < schema.minItems) {
|
|
197
|
+
errors.push({ path, message: `Array must have at least ${schema.minItems} items` });
|
|
198
|
+
}
|
|
199
|
+
if (schema.maxItems && data.length > schema.maxItems) {
|
|
200
|
+
errors.push({ path, message: `Array must have at most ${schema.maxItems} items` });
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
if (schema.type === 'string' && typeof data === 'string') {
|
|
205
|
+
if (schema.minLength && data.length < schema.minLength) {
|
|
206
|
+
errors.push({ path, message: `String must be at least ${schema.minLength} characters` });
|
|
207
|
+
}
|
|
208
|
+
if (schema.maxLength && data.length > schema.maxLength) {
|
|
209
|
+
errors.push({ path, message: `String must be at most ${schema.maxLength} characters` });
|
|
210
|
+
}
|
|
211
|
+
if (schema.pattern && !new RegExp(schema.pattern).test(data)) {
|
|
212
|
+
errors.push({ path, message: `String does not match pattern: ${schema.pattern}` });
|
|
213
|
+
}
|
|
214
|
+
if (schema.enum && !schema.enum.includes(data)) {
|
|
215
|
+
errors.push({ path, message: `Value must be one of: ${schema.enum.join(', ')}` });
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
if (schema.type === 'number' && typeof data === 'number') {
|
|
220
|
+
if (schema.minimum !== undefined && data < schema.minimum) {
|
|
221
|
+
errors.push({ path, message: `Number must be >= ${schema.minimum}` });
|
|
222
|
+
}
|
|
223
|
+
if (schema.maximum !== undefined && data > schema.maximum) {
|
|
224
|
+
errors.push({ path, message: `Number must be <= ${schema.maximum}` });
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
return { valid: errors.length === 0, errors };
|
|
229
|
+
}
|
|
230
|
+
|
|
231
|
+
// ─── Register Core Commands ─────────────────────────────────────────────────
|
|
232
|
+
|
|
233
|
+
// Discovery
|
|
234
|
+
registerCommand({
|
|
235
|
+
name: 'wab.discover',
|
|
236
|
+
version: '1.0.0',
|
|
237
|
+
description: 'Discover available actions on a site',
|
|
238
|
+
category: 'discovery',
|
|
239
|
+
capabilities: ['browser.read'],
|
|
240
|
+
permission: 'read',
|
|
241
|
+
input: {
|
|
242
|
+
type: 'object',
|
|
243
|
+
properties: {
|
|
244
|
+
url: Types.Url,
|
|
245
|
+
category: { type: 'string', enum: ['navigation', 'commerce', 'form', 'content', 'all'] },
|
|
246
|
+
depth: { type: 'number', minimum: 1, maximum: 5 },
|
|
247
|
+
},
|
|
248
|
+
required: ['url'],
|
|
249
|
+
},
|
|
250
|
+
output: {
|
|
251
|
+
type: 'object',
|
|
252
|
+
properties: {
|
|
253
|
+
actions: { type: 'array', items: { type: 'object', properties: {
|
|
254
|
+
name: { type: 'string' },
|
|
255
|
+
category: { type: 'string' },
|
|
256
|
+
selector: Types.Selector,
|
|
257
|
+
params: { type: 'object' },
|
|
258
|
+
capabilities: { type: 'array', items: { type: 'string' } },
|
|
259
|
+
}}},
|
|
260
|
+
meta: { type: 'object', properties: {
|
|
261
|
+
protocol: Types.Version,
|
|
262
|
+
site: { type: 'string' },
|
|
263
|
+
tier: { type: 'string' },
|
|
264
|
+
timestamp: Types.Timestamp,
|
|
265
|
+
}},
|
|
266
|
+
},
|
|
267
|
+
},
|
|
268
|
+
idempotent: true,
|
|
269
|
+
timeout: 15000,
|
|
270
|
+
});
|
|
271
|
+
|
|
272
|
+
// Execute action
|
|
273
|
+
registerCommand({
|
|
274
|
+
name: 'wab.execute',
|
|
275
|
+
version: '1.0.0',
|
|
276
|
+
description: 'Execute a registered action on a site',
|
|
277
|
+
category: 'execution',
|
|
278
|
+
capabilities: ['browser.execute'],
|
|
279
|
+
permission: 'execute',
|
|
280
|
+
input: {
|
|
281
|
+
type: 'object',
|
|
282
|
+
properties: {
|
|
283
|
+
action: { type: 'string', minLength: 1 },
|
|
284
|
+
params: { type: 'object' },
|
|
285
|
+
traceId: Types.TraceId,
|
|
286
|
+
timeout: { type: 'number', minimum: 100, maximum: 300000 },
|
|
287
|
+
},
|
|
288
|
+
required: ['action'],
|
|
289
|
+
},
|
|
290
|
+
output: {
|
|
291
|
+
type: 'object',
|
|
292
|
+
properties: {
|
|
293
|
+
success: { type: 'boolean' },
|
|
294
|
+
result: {},
|
|
295
|
+
duration: { type: 'number' },
|
|
296
|
+
traceId: Types.TraceId,
|
|
297
|
+
},
|
|
298
|
+
},
|
|
299
|
+
idempotent: false,
|
|
300
|
+
timeout: 30000,
|
|
301
|
+
});
|
|
302
|
+
|
|
303
|
+
// Semantic execute (checkout.addItem instead of click(selector))
|
|
304
|
+
registerCommand({
|
|
305
|
+
name: 'wab.semantic.execute',
|
|
306
|
+
version: '1.0.0',
|
|
307
|
+
description: 'Execute a semantic action (e.g., checkout.addItem) without raw DOM selectors',
|
|
308
|
+
category: 'execution',
|
|
309
|
+
capabilities: ['browser.execute'],
|
|
310
|
+
permission: 'execute',
|
|
311
|
+
input: {
|
|
312
|
+
type: 'object',
|
|
313
|
+
properties: {
|
|
314
|
+
domain: { type: 'string', description: 'Semantic domain (checkout, search, auth, etc.)' },
|
|
315
|
+
action: { type: 'string', description: 'Action within domain (addItem, submitForm, etc.)' },
|
|
316
|
+
params: { type: 'object' },
|
|
317
|
+
traceId: Types.TraceId,
|
|
318
|
+
},
|
|
319
|
+
required: ['domain', 'action'],
|
|
320
|
+
},
|
|
321
|
+
output: {
|
|
322
|
+
type: 'object',
|
|
323
|
+
properties: {
|
|
324
|
+
success: { type: 'boolean' },
|
|
325
|
+
result: {},
|
|
326
|
+
resolvedSelector: Types.Selector,
|
|
327
|
+
confidence: { type: 'number', minimum: 0, maximum: 1 },
|
|
328
|
+
duration: { type: 'number' },
|
|
329
|
+
},
|
|
330
|
+
},
|
|
331
|
+
idempotent: false,
|
|
332
|
+
timeout: 30000,
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
// Agent registration
|
|
336
|
+
registerCommand({
|
|
337
|
+
name: 'wab.agent.register',
|
|
338
|
+
version: '1.0.0',
|
|
339
|
+
description: 'Register an agent with the runtime',
|
|
340
|
+
category: 'lifecycle',
|
|
341
|
+
capabilities: [],
|
|
342
|
+
permission: 'write',
|
|
343
|
+
input: {
|
|
344
|
+
type: 'object',
|
|
345
|
+
properties: {
|
|
346
|
+
name: { type: 'string', minLength: 1, maxLength: 128 },
|
|
347
|
+
type: { type: 'string', enum: ['browser', 'server', 'hybrid', 'orchestrator'] },
|
|
348
|
+
capabilities: { type: 'array', items: { type: 'string' } },
|
|
349
|
+
publicKey: { type: 'string', description: 'Ed25519 public key (base64)' },
|
|
350
|
+
metadata: { type: 'object' },
|
|
351
|
+
},
|
|
352
|
+
required: ['name', 'type', 'capabilities'],
|
|
353
|
+
},
|
|
354
|
+
output: {
|
|
355
|
+
type: 'object',
|
|
356
|
+
properties: {
|
|
357
|
+
agentId: Types.AgentId,
|
|
358
|
+
token: { type: 'string' },
|
|
359
|
+
grantedCapabilities: { type: 'array', items: { type: 'string' } },
|
|
360
|
+
expiresAt: Types.Timestamp,
|
|
361
|
+
},
|
|
362
|
+
},
|
|
363
|
+
idempotent: false,
|
|
364
|
+
});
|
|
365
|
+
|
|
366
|
+
// Task submission
|
|
367
|
+
registerCommand({
|
|
368
|
+
name: 'wab.task.submit',
|
|
369
|
+
version: '1.0.0',
|
|
370
|
+
description: 'Submit a task to the runtime scheduler',
|
|
371
|
+
category: 'runtime',
|
|
372
|
+
capabilities: ['system.schedule'],
|
|
373
|
+
permission: 'execute',
|
|
374
|
+
input: {
|
|
375
|
+
type: 'object',
|
|
376
|
+
properties: {
|
|
377
|
+
type: { type: 'string', enum: ['browser', 'api', 'extraction', 'comparison', 'workflow', 'composite'] },
|
|
378
|
+
objective: { type: 'string', minLength: 1 },
|
|
379
|
+
steps: { type: 'array', items: { type: 'object', properties: {
|
|
380
|
+
command: { type: 'string' },
|
|
381
|
+
params: { type: 'object' },
|
|
382
|
+
dependsOn: { type: 'array', items: { type: 'string' } },
|
|
383
|
+
retries: { type: 'number', minimum: 0, maximum: 10 },
|
|
384
|
+
timeout: { type: 'number' },
|
|
385
|
+
}, required: ['command'] }},
|
|
386
|
+
priority: { type: 'number', minimum: 0, maximum: 100 },
|
|
387
|
+
deadline: Types.Timestamp,
|
|
388
|
+
agentId: Types.AgentId,
|
|
389
|
+
},
|
|
390
|
+
required: ['type', 'objective'],
|
|
391
|
+
},
|
|
392
|
+
output: {
|
|
393
|
+
type: 'object',
|
|
394
|
+
properties: {
|
|
395
|
+
taskId: Types.TaskId,
|
|
396
|
+
status: { type: 'string', enum: ['queued', 'scheduled', 'running'] },
|
|
397
|
+
estimatedStart: Types.Timestamp,
|
|
398
|
+
},
|
|
399
|
+
},
|
|
400
|
+
idempotent: false,
|
|
401
|
+
});
|
|
402
|
+
|
|
403
|
+
// Task status
|
|
404
|
+
registerCommand({
|
|
405
|
+
name: 'wab.task.status',
|
|
406
|
+
version: '1.0.0',
|
|
407
|
+
description: 'Get task execution status',
|
|
408
|
+
category: 'runtime',
|
|
409
|
+
capabilities: [],
|
|
410
|
+
permission: 'read',
|
|
411
|
+
input: {
|
|
412
|
+
type: 'object',
|
|
413
|
+
properties: {
|
|
414
|
+
taskId: Types.TaskId,
|
|
415
|
+
},
|
|
416
|
+
required: ['taskId'],
|
|
417
|
+
},
|
|
418
|
+
output: {
|
|
419
|
+
type: 'object',
|
|
420
|
+
properties: {
|
|
421
|
+
taskId: Types.TaskId,
|
|
422
|
+
status: { type: 'string', enum: ['queued', 'scheduled', 'running', 'paused', 'completed', 'failed', 'cancelled'] },
|
|
423
|
+
progress: { type: 'number', minimum: 0, maximum: 100 },
|
|
424
|
+
currentStep: { type: 'number' },
|
|
425
|
+
totalSteps: { type: 'number' },
|
|
426
|
+
result: {},
|
|
427
|
+
error: { type: 'object', properties: { code: { type: 'string' }, message: { type: 'string' } } },
|
|
428
|
+
startedAt: Types.Timestamp,
|
|
429
|
+
completedAt: Types.Timestamp,
|
|
430
|
+
checkpoints: { type: 'array', items: { type: 'object' } },
|
|
431
|
+
},
|
|
432
|
+
},
|
|
433
|
+
idempotent: true,
|
|
434
|
+
});
|
|
435
|
+
|
|
436
|
+
// Price comparison (semantic)
|
|
437
|
+
registerCommand({
|
|
438
|
+
name: 'wab.commerce.compare',
|
|
439
|
+
version: '1.0.0',
|
|
440
|
+
description: 'Compare prices across sources with fairness scoring',
|
|
441
|
+
category: 'commerce',
|
|
442
|
+
capabilities: ['commerce.price', 'data.compare'],
|
|
443
|
+
permission: 'read',
|
|
444
|
+
input: {
|
|
445
|
+
type: 'object',
|
|
446
|
+
properties: {
|
|
447
|
+
query: { type: 'string', minLength: 1 },
|
|
448
|
+
category: { type: 'string' },
|
|
449
|
+
sources: { type: 'array', items: Types.Url },
|
|
450
|
+
fairnessWeight: { type: 'number', minimum: 0, maximum: 1 },
|
|
451
|
+
},
|
|
452
|
+
required: ['query'],
|
|
453
|
+
},
|
|
454
|
+
output: {
|
|
455
|
+
type: 'object',
|
|
456
|
+
properties: {
|
|
457
|
+
results: { type: 'array', items: { type: 'object', properties: {
|
|
458
|
+
source: { type: 'string' },
|
|
459
|
+
price: { type: 'number' },
|
|
460
|
+
currency: { type: 'string' },
|
|
461
|
+
fairnessScore: { type: 'number' },
|
|
462
|
+
hasDecepitvePatterns: { type: 'boolean' },
|
|
463
|
+
}}},
|
|
464
|
+
bestDeal: { type: 'object' },
|
|
465
|
+
comparison: { type: 'object' },
|
|
466
|
+
},
|
|
467
|
+
},
|
|
468
|
+
idempotent: true,
|
|
469
|
+
timeout: 60000,
|
|
470
|
+
});
|
|
471
|
+
|
|
472
|
+
// Negotiate
|
|
473
|
+
registerCommand({
|
|
474
|
+
name: 'wab.commerce.negotiate',
|
|
475
|
+
version: '1.0.0',
|
|
476
|
+
description: 'Open or continue a price negotiation session',
|
|
477
|
+
category: 'commerce',
|
|
478
|
+
capabilities: ['commerce.negotiate'],
|
|
479
|
+
permission: 'execute',
|
|
480
|
+
input: {
|
|
481
|
+
type: 'object',
|
|
482
|
+
properties: {
|
|
483
|
+
siteId: Types.SiteId,
|
|
484
|
+
productId: { type: 'string' },
|
|
485
|
+
proposal: { type: 'object', properties: {
|
|
486
|
+
type: { type: 'string', enum: ['discount', 'bundle', 'loyalty', 'counter'] },
|
|
487
|
+
amount: { type: 'number' },
|
|
488
|
+
justification: { type: 'string' },
|
|
489
|
+
}, required: ['type', 'amount'] },
|
|
490
|
+
sessionId: { type: 'string' },
|
|
491
|
+
},
|
|
492
|
+
required: ['siteId', 'proposal'],
|
|
493
|
+
},
|
|
494
|
+
output: {
|
|
495
|
+
type: 'object',
|
|
496
|
+
properties: {
|
|
497
|
+
sessionId: { type: 'string' },
|
|
498
|
+
status: { type: 'string', enum: ['pending', 'accepted', 'rejected', 'counter'] },
|
|
499
|
+
counterOffer: { type: 'object' },
|
|
500
|
+
finalPrice: { type: 'number' },
|
|
501
|
+
},
|
|
502
|
+
},
|
|
503
|
+
idempotent: false,
|
|
504
|
+
});
|
|
505
|
+
|
|
506
|
+
// Agent mesh message
|
|
507
|
+
registerCommand({
|
|
508
|
+
name: 'wab.mesh.send',
|
|
509
|
+
version: '1.0.0',
|
|
510
|
+
description: 'Send a message through the agent mesh',
|
|
511
|
+
category: 'communication',
|
|
512
|
+
capabilities: ['agent.communicate'],
|
|
513
|
+
permission: 'write',
|
|
514
|
+
input: {
|
|
515
|
+
type: 'object',
|
|
516
|
+
properties: {
|
|
517
|
+
channel: { type: 'string' },
|
|
518
|
+
topic: { type: 'string' },
|
|
519
|
+
payload: {},
|
|
520
|
+
targetAgent: Types.AgentId,
|
|
521
|
+
broadcast: { type: 'boolean' },
|
|
522
|
+
},
|
|
523
|
+
required: ['channel', 'topic', 'payload'],
|
|
524
|
+
},
|
|
525
|
+
output: {
|
|
526
|
+
type: 'object',
|
|
527
|
+
properties: {
|
|
528
|
+
messageId: { type: 'string' },
|
|
529
|
+
delivered: { type: 'number' },
|
|
530
|
+
timestamp: Types.Timestamp,
|
|
531
|
+
},
|
|
532
|
+
},
|
|
533
|
+
});
|
|
534
|
+
|
|
535
|
+
// LLM Inference
|
|
536
|
+
registerCommand({
|
|
537
|
+
name: 'wab.ai.infer',
|
|
538
|
+
version: '1.0.0',
|
|
539
|
+
description: 'Run LLM inference through the model abstraction layer',
|
|
540
|
+
category: 'ai',
|
|
541
|
+
capabilities: ['ai.infer'],
|
|
542
|
+
permission: 'execute',
|
|
543
|
+
input: {
|
|
544
|
+
type: 'object',
|
|
545
|
+
properties: {
|
|
546
|
+
prompt: { type: 'string', minLength: 1 },
|
|
547
|
+
model: { type: 'string' },
|
|
548
|
+
provider: { type: 'string', enum: ['openai', 'anthropic', 'ollama', 'custom'] },
|
|
549
|
+
options: { type: 'object', properties: {
|
|
550
|
+
temperature: { type: 'number', minimum: 0, maximum: 2 },
|
|
551
|
+
maxTokens: { type: 'number', minimum: 1, maximum: 128000 },
|
|
552
|
+
systemPrompt: { type: 'string' },
|
|
553
|
+
}},
|
|
554
|
+
},
|
|
555
|
+
required: ['prompt'],
|
|
556
|
+
},
|
|
557
|
+
output: {
|
|
558
|
+
type: 'object',
|
|
559
|
+
properties: {
|
|
560
|
+
text: { type: 'string' },
|
|
561
|
+
model: { type: 'string' },
|
|
562
|
+
provider: { type: 'string' },
|
|
563
|
+
usage: { type: 'object', properties: {
|
|
564
|
+
promptTokens: { type: 'number' },
|
|
565
|
+
completionTokens: { type: 'number' },
|
|
566
|
+
totalTokens: { type: 'number' },
|
|
567
|
+
}},
|
|
568
|
+
duration: { type: 'number' },
|
|
569
|
+
},
|
|
570
|
+
},
|
|
571
|
+
timeout: 120000,
|
|
572
|
+
});
|
|
573
|
+
|
|
574
|
+
module.exports = {
|
|
575
|
+
PROTOCOL_VERSION,
|
|
576
|
+
Types,
|
|
577
|
+
Capabilities,
|
|
578
|
+
PermissionLevels,
|
|
579
|
+
registerCommand,
|
|
580
|
+
getCommand,
|
|
581
|
+
listCommands,
|
|
582
|
+
validateInput,
|
|
583
|
+
validateOutput,
|
|
584
|
+
};
|