openclaw-cascade-plugin 1.0.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.
Files changed (98) hide show
  1. package/PHASE1_SUMMARY.md +191 -0
  2. package/PHASE3_SUMMARY.md +195 -0
  3. package/README.md +43 -0
  4. package/dist/a2a-client.d.ts +17 -0
  5. package/dist/a2a-client.d.ts.map +1 -0
  6. package/dist/a2a-client.js +47 -0
  7. package/dist/a2a-client.js.map +1 -0
  8. package/dist/cascade-client.d.ts +53 -0
  9. package/dist/cascade-client.d.ts.map +1 -0
  10. package/dist/cascade-client.js +179 -0
  11. package/dist/cascade-client.js.map +1 -0
  12. package/dist/config.d.ts +26 -0
  13. package/dist/config.d.ts.map +1 -0
  14. package/dist/config.js +116 -0
  15. package/dist/config.js.map +1 -0
  16. package/dist/index.d.ts +29 -0
  17. package/dist/index.d.ts.map +1 -0
  18. package/dist/index.js +136 -0
  19. package/dist/index.js.map +1 -0
  20. package/dist/python-manager.d.ts +59 -0
  21. package/dist/python-manager.d.ts.map +1 -0
  22. package/dist/python-manager.js +190 -0
  23. package/dist/python-manager.js.map +1 -0
  24. package/dist/test-utils/helpers.d.ts +20 -0
  25. package/dist/test-utils/helpers.d.ts.map +1 -0
  26. package/dist/test-utils/helpers.js +89 -0
  27. package/dist/test-utils/helpers.js.map +1 -0
  28. package/dist/test-utils/index.d.ts +3 -0
  29. package/dist/test-utils/index.d.ts.map +1 -0
  30. package/dist/test-utils/index.js +19 -0
  31. package/dist/test-utils/index.js.map +1 -0
  32. package/dist/test-utils/mocks.d.ts +51 -0
  33. package/dist/test-utils/mocks.d.ts.map +1 -0
  34. package/dist/test-utils/mocks.js +84 -0
  35. package/dist/test-utils/mocks.js.map +1 -0
  36. package/dist/tools/a2a-tools.d.ts +9 -0
  37. package/dist/tools/a2a-tools.d.ts.map +1 -0
  38. package/dist/tools/a2a-tools.js +147 -0
  39. package/dist/tools/a2a-tools.js.map +1 -0
  40. package/dist/tools/api-tools.d.ts +9 -0
  41. package/dist/tools/api-tools.d.ts.map +1 -0
  42. package/dist/tools/api-tools.js +102 -0
  43. package/dist/tools/api-tools.js.map +1 -0
  44. package/dist/tools/desktop-automation.d.ts +10 -0
  45. package/dist/tools/desktop-automation.d.ts.map +1 -0
  46. package/dist/tools/desktop-automation.js +330 -0
  47. package/dist/tools/desktop-automation.js.map +1 -0
  48. package/dist/tools/index.d.ts +12 -0
  49. package/dist/tools/index.d.ts.map +1 -0
  50. package/dist/tools/index.js +35 -0
  51. package/dist/tools/index.js.map +1 -0
  52. package/dist/tools/response-helpers.d.ts +25 -0
  53. package/dist/tools/response-helpers.d.ts.map +1 -0
  54. package/dist/tools/response-helpers.js +71 -0
  55. package/dist/tools/response-helpers.js.map +1 -0
  56. package/dist/tools/sandbox-tools.d.ts +9 -0
  57. package/dist/tools/sandbox-tools.d.ts.map +1 -0
  58. package/dist/tools/sandbox-tools.js +79 -0
  59. package/dist/tools/sandbox-tools.js.map +1 -0
  60. package/dist/tools/tool-registry.d.ts +34 -0
  61. package/dist/tools/tool-registry.d.ts.map +1 -0
  62. package/dist/tools/tool-registry.js +50 -0
  63. package/dist/tools/tool-registry.js.map +1 -0
  64. package/dist/tools/web-automation.d.ts +9 -0
  65. package/dist/tools/web-automation.d.ts.map +1 -0
  66. package/dist/tools/web-automation.js +471 -0
  67. package/dist/tools/web-automation.js.map +1 -0
  68. package/dist/types/index.d.ts +111 -0
  69. package/dist/types/index.d.ts.map +1 -0
  70. package/dist/types/index.js +38 -0
  71. package/dist/types/index.js.map +1 -0
  72. package/jest.setup.js +19 -0
  73. package/openclaw-cascade-plugin-1.0.0.tgz +0 -0
  74. package/openclaw.plugin.json +116 -0
  75. package/package.json +74 -0
  76. package/src/a2a-client.ts +66 -0
  77. package/src/cascade-client.test.ts +400 -0
  78. package/src/cascade-client.ts +198 -0
  79. package/src/config.test.ts +194 -0
  80. package/src/config.ts +135 -0
  81. package/src/index.ts +164 -0
  82. package/src/python-manager.test.ts +187 -0
  83. package/src/python-manager.ts +230 -0
  84. package/src/test-utils/helpers.ts +107 -0
  85. package/src/test-utils/index.ts +2 -0
  86. package/src/test-utils/mocks.ts +101 -0
  87. package/src/tools/a2a-tools.ts +162 -0
  88. package/src/tools/api-tools.ts +110 -0
  89. package/src/tools/desktop-automation.test.ts +305 -0
  90. package/src/tools/desktop-automation.ts +366 -0
  91. package/src/tools/index.ts +13 -0
  92. package/src/tools/response-helpers.ts +78 -0
  93. package/src/tools/sandbox-tools.ts +83 -0
  94. package/src/tools/tool-registry.ts +51 -0
  95. package/src/tools/web-automation.test.ts +177 -0
  96. package/src/tools/web-automation.ts +518 -0
  97. package/src/types/index.ts +132 -0
  98. package/tsconfig.json +27 -0
@@ -0,0 +1,518 @@
1
+ /**
2
+ * Web Automation Tools (Playwright)
3
+ *
4
+ * 14 tools for web browser automation
5
+ */
6
+
7
+ import { ToolRegistry } from './tool-registry';
8
+ import { CascadeMcpClient } from '../cascade-client';
9
+ import { ToolResponse } from '../types';
10
+ import { errorResponse, formatSuccess } from './response-helpers';
11
+
12
+ export function registerWebTools(registry: ToolRegistry, client: CascadeMcpClient): void {
13
+ // Navigation Tools (5)
14
+
15
+ // 1. cascade_pw_goto
16
+ registry.register({
17
+ name: 'cascade_pw_goto',
18
+ description: 'Navigate to a URL in the browser',
19
+ inputSchema: {
20
+ type: 'object',
21
+ properties: {
22
+ url: {
23
+ type: 'string',
24
+ description: 'URL to navigate to'
25
+ },
26
+ wait_until: {
27
+ type: 'string',
28
+ enum: ['load', 'domcontentloaded', 'networkidle'],
29
+ default: 'networkidle',
30
+ description: 'When to consider navigation complete'
31
+ }
32
+ },
33
+ required: ['url']
34
+ },
35
+ handler: async (args): Promise<ToolResponse> => {
36
+ try {
37
+ if (!args.url) {
38
+ return errorResponse('url is required');
39
+ }
40
+
41
+ const result = await client.callTool('pw_goto', {
42
+ url: args.url,
43
+ wait_until: args.wait_until || 'networkidle'
44
+ });
45
+ return formatSuccess(result);
46
+ } catch (error) {
47
+ return errorResponse(
48
+ error instanceof Error ? error.message : 'Failed to navigate',
49
+ 'Check that the URL is valid and accessible'
50
+ );
51
+ }
52
+ }
53
+ });
54
+
55
+ // 2. cascade_pw_back
56
+ registry.register({
57
+ name: 'cascade_pw_back',
58
+ description: 'Go back to the previous page in browser history',
59
+ inputSchema: {
60
+ type: 'object',
61
+ properties: {}
62
+ },
63
+ handler: async (): Promise<ToolResponse> => {
64
+ try {
65
+ const result = await client.callTool('pw_back', {});
66
+ return formatSuccess(result);
67
+ } catch (error) {
68
+ return errorResponse(
69
+ error instanceof Error ? error.message : 'Failed to go back'
70
+ );
71
+ }
72
+ }
73
+ });
74
+
75
+ // 3. cascade_pw_forward
76
+ registry.register({
77
+ name: 'cascade_pw_forward',
78
+ description: 'Go forward to the next page in browser history',
79
+ inputSchema: {
80
+ type: 'object',
81
+ properties: {}
82
+ },
83
+ handler: async (): Promise<ToolResponse> => {
84
+ try {
85
+ const result = await client.callTool('pw_forward', {});
86
+ return formatSuccess(result);
87
+ } catch (error) {
88
+ return errorResponse(
89
+ error instanceof Error ? error.message : 'Failed to go forward'
90
+ );
91
+ }
92
+ }
93
+ });
94
+
95
+ // 4. cascade_pw_reload
96
+ registry.register({
97
+ name: 'cascade_pw_reload',
98
+ description: 'Reload the current page',
99
+ inputSchema: {
100
+ type: 'object',
101
+ properties: {
102
+ wait_until: {
103
+ type: 'string',
104
+ enum: ['load', 'domcontentloaded', 'networkidle'],
105
+ default: 'networkidle'
106
+ }
107
+ }
108
+ },
109
+ handler: async (args): Promise<ToolResponse> => {
110
+ try {
111
+ const result = await client.callTool('pw_reload', {
112
+ wait_until: args.wait_until || 'networkidle'
113
+ });
114
+ return formatSuccess(result);
115
+ } catch (error) {
116
+ return errorResponse(
117
+ error instanceof Error ? error.message : 'Failed to reload page'
118
+ );
119
+ }
120
+ }
121
+ });
122
+
123
+ // 5. cascade_pw_wait_for_url
124
+ registry.register({
125
+ name: 'cascade_pw_wait_for_url',
126
+ description: 'Wait for the current URL to match a pattern',
127
+ inputSchema: {
128
+ type: 'object',
129
+ properties: {
130
+ url: {
131
+ type: 'string',
132
+ description: 'URL pattern to wait for (glob or regex)'
133
+ },
134
+ timeout_ms: {
135
+ type: 'integer',
136
+ default: 10000,
137
+ description: 'Timeout in milliseconds'
138
+ }
139
+ },
140
+ required: ['url']
141
+ },
142
+ handler: async (args): Promise<ToolResponse> => {
143
+ try {
144
+ if (!args.url) {
145
+ return errorResponse('url is required');
146
+ }
147
+
148
+ const result = await client.callTool('pw_wait_for_url', {
149
+ url: args.url,
150
+ timeout_ms: args.timeout_ms || 10000
151
+ });
152
+ return formatSuccess(result);
153
+ } catch (error) {
154
+ return errorResponse(
155
+ error instanceof Error ? error.message : 'Failed to wait for URL',
156
+ 'The page may be loading slowly or the URL pattern may not match'
157
+ );
158
+ }
159
+ }
160
+ });
161
+
162
+ // Locator Tools (2)
163
+
164
+ // 6. cascade_pw_locator_count
165
+ registry.register({
166
+ name: 'cascade_pw_locator_count',
167
+ description: 'Count elements matching a selector',
168
+ inputSchema: {
169
+ type: 'object',
170
+ properties: {
171
+ selector: {
172
+ type: 'string',
173
+ description: 'CSS selector or Playwright locator'
174
+ }
175
+ },
176
+ required: ['selector']
177
+ },
178
+ handler: async (args): Promise<ToolResponse> => {
179
+ try {
180
+ if (!args.selector) {
181
+ return errorResponse('selector is required');
182
+ }
183
+
184
+ const result = await client.callTool('pw_locator_count', {
185
+ selector: args.selector
186
+ });
187
+ return formatSuccess({ count: result.count });
188
+ } catch (error) {
189
+ return errorResponse(
190
+ error instanceof Error ? error.message : 'Failed to count elements'
191
+ );
192
+ }
193
+ }
194
+ });
195
+
196
+ // 7. cascade_pw_locator_text
197
+ registry.register({
198
+ name: 'cascade_pw_locator_text',
199
+ description: 'Get text content of the first matching element',
200
+ inputSchema: {
201
+ type: 'object',
202
+ properties: {
203
+ selector: {
204
+ type: 'string',
205
+ description: 'CSS selector or Playwright locator'
206
+ },
207
+ timeout_ms: {
208
+ type: 'integer',
209
+ default: 8000,
210
+ description: 'Timeout in milliseconds'
211
+ }
212
+ },
213
+ required: ['selector']
214
+ },
215
+ handler: async (args): Promise<ToolResponse> => {
216
+ try {
217
+ if (!args.selector) {
218
+ return errorResponse('selector is required');
219
+ }
220
+
221
+ const result = await client.callTool('pw_locator_text', {
222
+ selector: args.selector,
223
+ timeout_ms: args.timeout_ms || 8000
224
+ });
225
+ return formatSuccess({ text: result.text });
226
+ } catch (error) {
227
+ return errorResponse(
228
+ error instanceof Error ? error.message : 'Failed to get element text'
229
+ );
230
+ }
231
+ }
232
+ });
233
+
234
+ // Interaction Tools (5)
235
+
236
+ // 8. cascade_pw_click
237
+ registry.register({
238
+ name: 'cascade_pw_click',
239
+ description: 'Click on an element',
240
+ inputSchema: {
241
+ type: 'object',
242
+ properties: {
243
+ selector: {
244
+ type: 'string',
245
+ description: 'CSS selector or Playwright locator'
246
+ },
247
+ timeout_ms: {
248
+ type: 'integer',
249
+ default: 8000
250
+ }
251
+ },
252
+ required: ['selector']
253
+ },
254
+ handler: async (args): Promise<ToolResponse> => {
255
+ try {
256
+ if (!args.selector) {
257
+ return errorResponse('selector is required');
258
+ }
259
+
260
+ const result = await client.callTool('pw_click', {
261
+ selector: args.selector,
262
+ timeout_ms: args.timeout_ms || 8000
263
+ });
264
+ return formatSuccess(result);
265
+ } catch (error) {
266
+ return errorResponse(
267
+ error instanceof Error ? error.message : 'Failed to click element',
268
+ 'Make sure the element exists and is visible'
269
+ );
270
+ }
271
+ }
272
+ });
273
+
274
+ // 9. cascade_pw_fill
275
+ registry.register({
276
+ name: 'cascade_pw_fill',
277
+ description: 'Fill a form field with text',
278
+ inputSchema: {
279
+ type: 'object',
280
+ properties: {
281
+ selector: {
282
+ type: 'string',
283
+ description: 'CSS selector or Playwright locator'
284
+ },
285
+ text: {
286
+ type: 'string',
287
+ description: 'Text to fill'
288
+ },
289
+ timeout_ms: {
290
+ type: 'integer',
291
+ default: 8000
292
+ }
293
+ },
294
+ required: ['selector', 'text']
295
+ },
296
+ handler: async (args): Promise<ToolResponse> => {
297
+ try {
298
+ if (!args.selector) {
299
+ return errorResponse('selector is required');
300
+ }
301
+ if (!args.text) {
302
+ return errorResponse('text is required');
303
+ }
304
+
305
+ const result = await client.callTool('pw_fill', {
306
+ selector: args.selector,
307
+ text: args.text,
308
+ timeout_ms: args.timeout_ms || 8000
309
+ });
310
+ return formatSuccess(result);
311
+ } catch (error) {
312
+ return errorResponse(
313
+ error instanceof Error ? error.message : 'Failed to fill field'
314
+ );
315
+ }
316
+ }
317
+ });
318
+
319
+ // 10. cascade_pw_press
320
+ registry.register({
321
+ name: 'cascade_pw_press',
322
+ description: 'Press a key on an element',
323
+ inputSchema: {
324
+ type: 'object',
325
+ properties: {
326
+ selector: {
327
+ type: 'string',
328
+ description: 'CSS selector or Playwright locator'
329
+ },
330
+ key: {
331
+ type: 'string',
332
+ description: 'Key to press (e.g., Enter, Tab, Escape)'
333
+ },
334
+ timeout_ms: {
335
+ type: 'integer',
336
+ default: 8000
337
+ }
338
+ },
339
+ required: ['selector', 'key']
340
+ },
341
+ handler: async (args): Promise<ToolResponse> => {
342
+ try {
343
+ if (!args.selector) {
344
+ return errorResponse('selector is required');
345
+ }
346
+ if (!args.key) {
347
+ return errorResponse('key is required');
348
+ }
349
+
350
+ const result = await client.callTool('pw_press', {
351
+ selector: args.selector,
352
+ key: args.key,
353
+ timeout_ms: args.timeout_ms || 8000
354
+ });
355
+ return formatSuccess(result);
356
+ } catch (error) {
357
+ return errorResponse(
358
+ error instanceof Error ? error.message : 'Failed to press key'
359
+ );
360
+ }
361
+ }
362
+ });
363
+
364
+ // 11. cascade_pw_select_option
365
+ registry.register({
366
+ name: 'cascade_pw_select_option',
367
+ description: 'Select option(s) in a dropdown',
368
+ inputSchema: {
369
+ type: 'object',
370
+ properties: {
371
+ selector: {
372
+ type: 'string',
373
+ description: 'CSS selector for select element'
374
+ },
375
+ values: {
376
+ type: 'array',
377
+ items: { type: 'string' },
378
+ description: 'Values to select'
379
+ }
380
+ },
381
+ required: ['selector', 'values']
382
+ },
383
+ handler: async (args): Promise<ToolResponse> => {
384
+ try {
385
+ if (!args.selector) {
386
+ return errorResponse('selector is required');
387
+ }
388
+ if (!args.values || !Array.isArray(args.values)) {
389
+ return errorResponse('values is required and must be an array');
390
+ }
391
+
392
+ const result = await client.callTool('pw_select_option', {
393
+ selector: args.selector,
394
+ values: args.values
395
+ });
396
+ return formatSuccess({ selected: result.selected });
397
+ } catch (error) {
398
+ return errorResponse(
399
+ error instanceof Error ? error.message : 'Failed to select option'
400
+ );
401
+ }
402
+ }
403
+ });
404
+
405
+ // Evaluation Tools (3)
406
+
407
+ // 12. cascade_pw_eval
408
+ registry.register({
409
+ name: 'cascade_pw_eval',
410
+ description: 'Evaluate JavaScript in the page context',
411
+ inputSchema: {
412
+ type: 'object',
413
+ properties: {
414
+ expression: {
415
+ type: 'string',
416
+ description: 'JavaScript expression to evaluate'
417
+ }
418
+ },
419
+ required: ['expression']
420
+ },
421
+ handler: async (args): Promise<ToolResponse> => {
422
+ try {
423
+ if (!args.expression) {
424
+ return errorResponse('expression is required');
425
+ }
426
+
427
+ const result = await client.callTool('pw_eval', {
428
+ expression: args.expression
429
+ });
430
+ return formatSuccess({ result: result.result });
431
+ } catch (error) {
432
+ return errorResponse(
433
+ error instanceof Error ? error.message : 'Failed to evaluate expression'
434
+ );
435
+ }
436
+ }
437
+ });
438
+
439
+ // 13. cascade_pw_eval_on_selector
440
+ registry.register({
441
+ name: 'cascade_pw_eval_on_selector',
442
+ description: 'Evaluate JavaScript on a specific element',
443
+ inputSchema: {
444
+ type: 'object',
445
+ properties: {
446
+ selector: {
447
+ type: 'string',
448
+ description: 'CSS selector for target element'
449
+ },
450
+ expression: {
451
+ type: 'string',
452
+ description: 'JavaScript function body (receives element as argument)'
453
+ }
454
+ },
455
+ required: ['selector', 'expression']
456
+ },
457
+ handler: async (args): Promise<ToolResponse> => {
458
+ try {
459
+ if (!args.selector) {
460
+ return errorResponse('selector is required');
461
+ }
462
+ if (!args.expression) {
463
+ return errorResponse('expression is required');
464
+ }
465
+
466
+ const result = await client.callTool('pw_eval_on_selector', {
467
+ selector: args.selector,
468
+ expression: args.expression
469
+ });
470
+ return formatSuccess({ result: result.result });
471
+ } catch (error) {
472
+ return errorResponse(
473
+ error instanceof Error ? error.message : 'Failed to evaluate on element'
474
+ );
475
+ }
476
+ }
477
+ });
478
+
479
+ // 14. cascade_pw_list_frames
480
+ registry.register({
481
+ name: 'cascade_pw_list_frames',
482
+ description: 'List all frames (iframes) on the current page',
483
+ inputSchema: {
484
+ type: 'object',
485
+ properties: {}
486
+ },
487
+ handler: async (): Promise<ToolResponse> => {
488
+ try {
489
+ const result = await client.callTool('pw_list_frames', {});
490
+ return formatSuccess({ frames: result.frames });
491
+ } catch (error) {
492
+ return errorResponse(
493
+ error instanceof Error ? error.message : 'Failed to list frames'
494
+ );
495
+ }
496
+ }
497
+ });
498
+
499
+ // 15. cascade_pw_get_cookies
500
+ registry.register({
501
+ name: 'cascade_pw_get_cookies',
502
+ description: 'Get all cookies for the current context',
503
+ inputSchema: {
504
+ type: 'object',
505
+ properties: {}
506
+ },
507
+ handler: async (): Promise<ToolResponse> => {
508
+ try {
509
+ const result = await client.callTool('pw_get_cookies', {});
510
+ return formatSuccess({ cookies: result.cookies });
511
+ } catch (error) {
512
+ return errorResponse(
513
+ error instanceof Error ? error.message : 'Failed to get cookies'
514
+ );
515
+ }
516
+ }
517
+ });
518
+ }
@@ -0,0 +1,132 @@
1
+ /**
2
+ * Type definitions for OpenClaw Plugin
3
+ */
4
+
5
+ // Configuration Types
6
+ export interface CascadePluginConfig {
7
+ cascadeGrpcEndpoint: string;
8
+ cascadePythonPath?: string;
9
+ firestoreProjectId?: string;
10
+ firestoreCredentialsPath?: string;
11
+ headless?: boolean;
12
+ actionTimeoutMs?: number;
13
+ enableA2A?: boolean;
14
+ allowedAgents?: AgentRole[];
15
+ requireAgentConfirmation?: boolean;
16
+ verbose?: boolean;
17
+ screenshotMode?: 'embed' | 'disk' | 'auto';
18
+ screenshotDir?: string;
19
+ }
20
+
21
+ export type AgentRole = 'explorer' | 'worker' | 'orchestrator';
22
+
23
+ // Tool Types
24
+ export interface ToolSchema {
25
+ name: string;
26
+ description: string;
27
+ inputSchema: Record<string, any>;
28
+ }
29
+
30
+ export interface ToolRegistry {
31
+ register(tool: ToolSchema & { handler: ToolHandler }): void;
32
+ getAll(): Array<ToolSchema & { handler: ToolHandler }>;
33
+ get(name: string): (ToolSchema & { handler: ToolHandler }) | undefined;
34
+ }
35
+
36
+ export type ToolHandler = (args: Record<string, any>) => Promise<ToolResponse>;
37
+
38
+ export interface ToolResponse {
39
+ content: Array<{ type: string; [key: string]: any }>;
40
+ isError?: boolean;
41
+ }
42
+
43
+ // MCP Types
44
+ export interface McpRequest {
45
+ jsonrpc: '2.0';
46
+ id: number;
47
+ method: string;
48
+ params?: Record<string, any>;
49
+ }
50
+
51
+ export interface McpResponse {
52
+ jsonrpc: '2.0';
53
+ id: number;
54
+ result?: any;
55
+ error?: {
56
+ code: number;
57
+ message: string;
58
+ };
59
+ }
60
+
61
+ // A2A Types
62
+ export interface A2AMessage {
63
+ type: string;
64
+ source: 'openclaw' | AgentRole;
65
+ timestamp: number;
66
+ payload: any;
67
+ runId?: string;
68
+ }
69
+
70
+ export interface AgentMessage {
71
+ messageId: string;
72
+ userId: string;
73
+ appId: string;
74
+ senderAgentId: string;
75
+ senderRole: string;
76
+ targetAgentId?: string;
77
+ targetRole?: string;
78
+ runId?: string;
79
+ headers: Record<string, string>;
80
+ jsonPayload: string;
81
+ createdAtMs: number;
82
+ }
83
+
84
+ // OpenClaw API Types
85
+ export interface OpenClawApi {
86
+ config: {
87
+ plugins: {
88
+ entries: {
89
+ cascade?: {
90
+ config?: CascadePluginConfig;
91
+ };
92
+ };
93
+ };
94
+ };
95
+ registerTool(tool: ToolSchema & { handler: ToolHandler }): void;
96
+ registerGatewayMethod(name: string, handler: (context: any) => void): void;
97
+ registerCli(handler: (context: { program: any }) => void): void;
98
+ notify(message: string): void;
99
+ }
100
+
101
+ // Error Types
102
+ export class CascadeError extends Error {
103
+ constructor(
104
+ message: string,
105
+ public code: string,
106
+ public suggestion?: string
107
+ ) {
108
+ super(message);
109
+ this.name = 'CascadeError';
110
+ }
111
+ }
112
+
113
+ export class A2ADisabledError extends CascadeError {
114
+ constructor(message: string) {
115
+ super(message, 'A2A_DISABLED');
116
+ this.name = 'A2ADisabledError';
117
+ }
118
+ }
119
+
120
+ export class AgentNotAllowedError extends CascadeError {
121
+ constructor(message: string) {
122
+ super(message, 'AGENT_NOT_ALLOWED');
123
+ this.name = 'AgentNotAllowedError';
124
+ }
125
+ }
126
+
127
+ export class UserCancelledError extends CascadeError {
128
+ constructor(message: string) {
129
+ super(message, 'USER_CANCELLED');
130
+ this.name = 'UserCancelledError';
131
+ }
132
+ }
package/tsconfig.json ADDED
@@ -0,0 +1,27 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2020",
4
+ "module": "CommonJS",
5
+ "lib": ["ES2020"],
6
+ "outDir": "./dist",
7
+ "rootDir": "./src",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "skipLibCheck": true,
11
+ "forceConsistentCasingInFileNames": true,
12
+ "resolveJsonModule": true,
13
+ "declaration": true,
14
+ "declarationMap": true,
15
+ "sourceMap": true,
16
+ "noUnusedLocals": true,
17
+ "noUnusedParameters": true,
18
+ "noImplicitReturns": true,
19
+ "noFallthroughCasesInSwitch": true,
20
+ "baseUrl": ".",
21
+ "paths": {
22
+ "@/*": ["src/*"]
23
+ }
24
+ },
25
+ "include": ["src/**/*"],
26
+ "exclude": ["node_modules", "dist", "**/*.test.ts"]
27
+ }