opc-agent 1.1.3 → 1.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.
Files changed (156) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/CONTRIBUTING.md +75 -75
  3. package/README.md +429 -429
  4. package/README.zh-CN.md +415 -415
  5. package/dist/channels/web.js +256 -256
  6. package/dist/core/streaming.d.ts +56 -0
  7. package/dist/core/streaming.js +160 -0
  8. package/dist/deploy/hermes.js +22 -22
  9. package/dist/deploy/openclaw.js +31 -31
  10. package/dist/index.d.ts +4 -0
  11. package/dist/index.js +7 -1
  12. package/dist/providers/index.d.ts +1 -1
  13. package/dist/providers/index.js +13 -148
  14. package/dist/schema/oad.d.ts +3 -3
  15. package/dist/templates/code-reviewer.js +5 -5
  16. package/dist/templates/customer-service.js +2 -2
  17. package/dist/templates/data-analyst.js +5 -5
  18. package/dist/templates/knowledge-base.js +2 -2
  19. package/dist/templates/sales-assistant.js +4 -4
  20. package/dist/templates/teacher.js +6 -6
  21. package/dist/tools/gateway.d.ts +28 -0
  22. package/dist/tools/gateway.js +177 -0
  23. package/docs/.vitepress/config.ts +103 -103
  24. package/docs/api/cli.md +48 -48
  25. package/docs/api/oad-schema.md +64 -64
  26. package/docs/api/sdk.md +80 -80
  27. package/docs/guide/concepts.md +51 -51
  28. package/docs/guide/configuration.md +79 -79
  29. package/docs/guide/deployment.md +42 -42
  30. package/docs/guide/getting-started.md +44 -44
  31. package/docs/guide/templates.md +28 -28
  32. package/docs/guide/testing.md +84 -84
  33. package/docs/index.md +27 -27
  34. package/docs/zh/api/cli.md +54 -54
  35. package/docs/zh/api/oad-schema.md +87 -87
  36. package/docs/zh/api/sdk.md +102 -102
  37. package/docs/zh/guide/concepts.md +104 -104
  38. package/docs/zh/guide/configuration.md +135 -135
  39. package/docs/zh/guide/deployment.md +81 -81
  40. package/docs/zh/guide/getting-started.md +82 -82
  41. package/docs/zh/guide/templates.md +84 -84
  42. package/docs/zh/guide/testing.md +88 -88
  43. package/docs/zh/index.md +27 -27
  44. package/examples/customer-service-demo/README.md +90 -90
  45. package/examples/customer-service-demo/oad.yaml +107 -107
  46. package/package.json +1 -1
  47. package/src/analytics/index.ts +66 -66
  48. package/src/channels/discord.ts +192 -192
  49. package/src/channels/email.ts +177 -177
  50. package/src/channels/feishu.ts +236 -236
  51. package/src/channels/index.ts +15 -15
  52. package/src/channels/slack.ts +160 -160
  53. package/src/channels/telegram.ts +90 -90
  54. package/src/channels/voice.ts +106 -106
  55. package/src/channels/web.ts +17 -17
  56. package/src/channels/webhook.ts +199 -199
  57. package/src/channels/websocket.ts +87 -87
  58. package/src/channels/wechat.ts +149 -149
  59. package/src/core/a2a.ts +143 -143
  60. package/src/core/agent.ts +152 -152
  61. package/src/core/analytics-engine.ts +186 -186
  62. package/src/core/auth.ts +57 -57
  63. package/src/core/cache.ts +141 -141
  64. package/src/core/compose.ts +77 -77
  65. package/src/core/config.ts +14 -14
  66. package/src/core/errors.ts +148 -148
  67. package/src/core/hitl.ts +138 -138
  68. package/src/core/knowledge.ts +49 -4
  69. package/src/core/logger.ts +57 -57
  70. package/src/core/orchestrator.ts +215 -215
  71. package/src/core/performance.ts +187 -187
  72. package/src/core/rate-limiter.ts +128 -128
  73. package/src/core/room.ts +109 -109
  74. package/src/core/runtime.ts +152 -152
  75. package/src/core/sandbox.ts +101 -101
  76. package/src/core/security.ts +171 -171
  77. package/src/core/streaming.ts +195 -0
  78. package/src/core/types.ts +68 -68
  79. package/src/core/versioning.ts +106 -106
  80. package/src/core/watch.ts +178 -178
  81. package/src/core/workflow.ts +235 -235
  82. package/src/deploy/hermes.ts +156 -156
  83. package/src/deploy/openclaw.ts +200 -200
  84. package/src/dtv/data.ts +29 -29
  85. package/src/dtv/trust.ts +43 -43
  86. package/src/dtv/value.ts +47 -47
  87. package/src/i18n/index.ts +216 -216
  88. package/src/index.ts +6 -0
  89. package/src/marketplace/index.ts +223 -223
  90. package/src/memory/deepbrain.ts +108 -108
  91. package/src/memory/index.ts +34 -34
  92. package/src/plugins/index.ts +208 -208
  93. package/src/providers/index.ts +12 -3
  94. package/src/schema/oad.ts +155 -155
  95. package/src/skills/base.ts +16 -16
  96. package/src/skills/document.ts +100 -100
  97. package/src/skills/http.ts +35 -35
  98. package/src/skills/index.ts +27 -27
  99. package/src/skills/scheduler.ts +80 -80
  100. package/src/skills/webhook-trigger.ts +59 -59
  101. package/src/templates/code-reviewer.ts +34 -34
  102. package/src/templates/customer-service.ts +80 -80
  103. package/src/templates/data-analyst.ts +70 -70
  104. package/src/templates/executive-assistant.ts +71 -71
  105. package/src/templates/financial-advisor.ts +60 -60
  106. package/src/templates/knowledge-base.ts +31 -31
  107. package/src/templates/legal-assistant.ts +71 -71
  108. package/src/templates/sales-assistant.ts +79 -79
  109. package/src/templates/teacher.ts +79 -79
  110. package/src/testing/index.ts +181 -181
  111. package/src/tools/calculator.ts +73 -73
  112. package/src/tools/datetime.ts +149 -149
  113. package/src/tools/gateway.ts +220 -0
  114. package/src/tools/json-transform.ts +187 -187
  115. package/src/tools/mcp.ts +76 -76
  116. package/src/tools/text-analysis.ts +116 -116
  117. package/templates/Dockerfile +15 -15
  118. package/templates/code-reviewer/README.md +27 -27
  119. package/templates/code-reviewer/oad.yaml +41 -41
  120. package/templates/customer-service/README.md +22 -22
  121. package/templates/customer-service/oad.yaml +36 -36
  122. package/templates/docker-compose.yml +21 -21
  123. package/templates/ecommerce-assistant/README.md +45 -0
  124. package/templates/ecommerce-assistant/oad.yaml +47 -0
  125. package/templates/knowledge-base/README.md +28 -28
  126. package/templates/knowledge-base/oad.yaml +38 -38
  127. package/templates/sales-assistant/README.md +26 -26
  128. package/templates/sales-assistant/oad.yaml +43 -43
  129. package/templates/tech-support/README.md +43 -0
  130. package/templates/tech-support/oad.yaml +45 -0
  131. package/tests/a2a.test.ts +66 -66
  132. package/tests/agent.test.ts +72 -72
  133. package/tests/analytics.test.ts +50 -50
  134. package/tests/channel.test.ts +39 -39
  135. package/tests/e2e.test.ts +134 -134
  136. package/tests/errors.test.ts +83 -83
  137. package/tests/gateway.test.ts +71 -0
  138. package/tests/hitl.test.ts +71 -71
  139. package/tests/i18n.test.ts +41 -41
  140. package/tests/mcp.test.ts +54 -54
  141. package/tests/oad.test.ts +68 -68
  142. package/tests/performance.test.ts +115 -115
  143. package/tests/plugin.test.ts +74 -74
  144. package/tests/room.test.ts +106 -106
  145. package/tests/runtime.test.ts +42 -42
  146. package/tests/sandbox.test.ts +46 -46
  147. package/tests/security.test.ts +60 -60
  148. package/tests/streaming.test.ts +109 -0
  149. package/tests/templates.test.ts +77 -77
  150. package/tests/v070.test.ts +76 -76
  151. package/tests/versioning.test.ts +75 -75
  152. package/tests/voice.test.ts +61 -61
  153. package/tests/webhook.test.ts +29 -29
  154. package/tests/workflow.test.ts +143 -143
  155. package/tsconfig.json +19 -19
  156. package/vitest.config.ts +9 -9
package/src/core/watch.ts CHANGED
@@ -1,178 +1,178 @@
1
- import { EventEmitter } from 'events';
2
-
3
- /**
4
- * ProcessWatcher — Background process output monitoring with pattern matching.
5
- *
6
- * Inspired by Hermes Agent's watch_patterns feature.
7
- * Set patterns to watch for in background process output and get callbacks
8
- * when they match — no polling needed.
9
- *
10
- * Usage:
11
- * const watcher = new ProcessWatcher();
12
- * watcher.watch(childProcess.stdout, {
13
- * patterns: [
14
- * { regex: /listening on port (\d+)/, label: 'server-ready' },
15
- * { regex: /error|Error|ERROR/, label: 'error-detected' },
16
- * { regex: /build completed/, label: 'build-done', once: true },
17
- * ],
18
- * onMatch: (match) => console.log(`[${match.label}] ${match.line}`),
19
- * });
20
- */
21
-
22
- export interface WatchPattern {
23
- /** Regex to match against each line of output */
24
- regex: RegExp;
25
- /** Human-readable label for this pattern */
26
- label: string;
27
- /** If true, auto-remove after first match */
28
- once?: boolean;
29
- }
30
-
31
- export interface WatchMatch {
32
- /** Pattern label */
33
- label: string;
34
- /** The full line that matched */
35
- line: string;
36
- /** Regex match groups */
37
- groups: string[];
38
- /** Timestamp of match */
39
- timestamp: number;
40
- /** Stream source */
41
- stream: 'stdout' | 'stderr';
42
- }
43
-
44
- export interface WatchOptions {
45
- /** Patterns to match */
46
- patterns: WatchPattern[];
47
- /** Callback on match */
48
- onMatch: (match: WatchMatch) => void;
49
- /** Optional: max matches to keep in history (default: 100) */
50
- maxHistory?: number;
51
- }
52
-
53
- export class ProcessWatcher extends EventEmitter {
54
- private watchers = new Map<string, {
55
- patterns: WatchPattern[];
56
- onMatch: (match: WatchMatch) => void;
57
- history: WatchMatch[];
58
- maxHistory: number;
59
- }>();
60
-
61
- private watcherIdCounter = 0;
62
-
63
- /**
64
- * Start watching a readable stream for patterns.
65
- * Returns a watcher ID that can be used to stop watching.
66
- */
67
- watch(
68
- stream: NodeJS.ReadableStream,
69
- options: WatchOptions,
70
- streamName: 'stdout' | 'stderr' = 'stdout',
71
- ): string {
72
- const id = `watcher_${++this.watcherIdCounter}`;
73
- const state = {
74
- patterns: [...options.patterns],
75
- onMatch: options.onMatch,
76
- history: [] as WatchMatch[],
77
- maxHistory: options.maxHistory ?? 100,
78
- };
79
- this.watchers.set(id, state);
80
-
81
- let buffer = '';
82
-
83
- const onData = (chunk: Buffer | string) => {
84
- buffer += chunk.toString();
85
- const lines = buffer.split('\n');
86
- buffer = lines.pop() ?? ''; // Keep incomplete line in buffer
87
-
88
- for (const line of lines) {
89
- this.matchLine(id, line, streamName);
90
- }
91
- };
92
-
93
- stream.on('data', onData);
94
- stream.on('end', () => {
95
- // Process remaining buffer
96
- if (buffer) this.matchLine(id, buffer, streamName);
97
- this.watchers.delete(id);
98
- this.emit('watcher:end', id);
99
- });
100
-
101
- return id;
102
- }
103
-
104
- /**
105
- * Watch both stdout and stderr of a ChildProcess.
106
- */
107
- watchProcess(
108
- proc: { stdout?: NodeJS.ReadableStream | null; stderr?: NodeJS.ReadableStream | null },
109
- options: WatchOptions,
110
- ): string[] {
111
- const ids: string[] = [];
112
- if (proc.stdout) ids.push(this.watch(proc.stdout, options, 'stdout'));
113
- if (proc.stderr) ids.push(this.watch(proc.stderr, options, 'stderr'));
114
- return ids;
115
- }
116
-
117
- /** Stop a specific watcher */
118
- unwatch(id: string): void {
119
- this.watchers.delete(id);
120
- }
121
-
122
- /** Get match history for a watcher */
123
- getHistory(id: string): WatchMatch[] {
124
- return this.watchers.get(id)?.history ?? [];
125
- }
126
-
127
- /** Add a pattern to an existing watcher */
128
- addPattern(id: string, pattern: WatchPattern): void {
129
- const state = this.watchers.get(id);
130
- if (state) state.patterns.push(pattern);
131
- }
132
-
133
- /** Remove a pattern by label from a watcher */
134
- removePattern(id: string, label: string): void {
135
- const state = this.watchers.get(id);
136
- if (state) {
137
- state.patterns = state.patterns.filter(p => p.label !== label);
138
- }
139
- }
140
-
141
- private matchLine(watcherId: string, line: string, stream: 'stdout' | 'stderr'): void {
142
- const state = this.watchers.get(watcherId);
143
- if (!state) return;
144
-
145
- const toRemove: number[] = [];
146
-
147
- for (let i = 0; i < state.patterns.length; i++) {
148
- const pattern = state.patterns[i];
149
- const m = line.match(pattern.regex);
150
- if (m) {
151
- const match: WatchMatch = {
152
- label: pattern.label,
153
- line,
154
- groups: m.slice(1),
155
- timestamp: Date.now(),
156
- stream,
157
- };
158
-
159
- state.history.push(match);
160
- if (state.history.length > state.maxHistory) {
161
- state.history.shift();
162
- }
163
-
164
- state.onMatch(match);
165
- this.emit('match', match);
166
-
167
- if (pattern.once) {
168
- toRemove.push(i);
169
- }
170
- }
171
- }
172
-
173
- // Remove once-patterns in reverse order
174
- for (let i = toRemove.length - 1; i >= 0; i--) {
175
- state.patterns.splice(toRemove[i], 1);
176
- }
177
- }
178
- }
1
+ import { EventEmitter } from 'events';
2
+
3
+ /**
4
+ * ProcessWatcher — Background process output monitoring with pattern matching.
5
+ *
6
+ * Inspired by Hermes Agent's watch_patterns feature.
7
+ * Set patterns to watch for in background process output and get callbacks
8
+ * when they match — no polling needed.
9
+ *
10
+ * Usage:
11
+ * const watcher = new ProcessWatcher();
12
+ * watcher.watch(childProcess.stdout, {
13
+ * patterns: [
14
+ * { regex: /listening on port (\d+)/, label: 'server-ready' },
15
+ * { regex: /error|Error|ERROR/, label: 'error-detected' },
16
+ * { regex: /build completed/, label: 'build-done', once: true },
17
+ * ],
18
+ * onMatch: (match) => console.log(`[${match.label}] ${match.line}`),
19
+ * });
20
+ */
21
+
22
+ export interface WatchPattern {
23
+ /** Regex to match against each line of output */
24
+ regex: RegExp;
25
+ /** Human-readable label for this pattern */
26
+ label: string;
27
+ /** If true, auto-remove after first match */
28
+ once?: boolean;
29
+ }
30
+
31
+ export interface WatchMatch {
32
+ /** Pattern label */
33
+ label: string;
34
+ /** The full line that matched */
35
+ line: string;
36
+ /** Regex match groups */
37
+ groups: string[];
38
+ /** Timestamp of match */
39
+ timestamp: number;
40
+ /** Stream source */
41
+ stream: 'stdout' | 'stderr';
42
+ }
43
+
44
+ export interface WatchOptions {
45
+ /** Patterns to match */
46
+ patterns: WatchPattern[];
47
+ /** Callback on match */
48
+ onMatch: (match: WatchMatch) => void;
49
+ /** Optional: max matches to keep in history (default: 100) */
50
+ maxHistory?: number;
51
+ }
52
+
53
+ export class ProcessWatcher extends EventEmitter {
54
+ private watchers = new Map<string, {
55
+ patterns: WatchPattern[];
56
+ onMatch: (match: WatchMatch) => void;
57
+ history: WatchMatch[];
58
+ maxHistory: number;
59
+ }>();
60
+
61
+ private watcherIdCounter = 0;
62
+
63
+ /**
64
+ * Start watching a readable stream for patterns.
65
+ * Returns a watcher ID that can be used to stop watching.
66
+ */
67
+ watch(
68
+ stream: NodeJS.ReadableStream,
69
+ options: WatchOptions,
70
+ streamName: 'stdout' | 'stderr' = 'stdout',
71
+ ): string {
72
+ const id = `watcher_${++this.watcherIdCounter}`;
73
+ const state = {
74
+ patterns: [...options.patterns],
75
+ onMatch: options.onMatch,
76
+ history: [] as WatchMatch[],
77
+ maxHistory: options.maxHistory ?? 100,
78
+ };
79
+ this.watchers.set(id, state);
80
+
81
+ let buffer = '';
82
+
83
+ const onData = (chunk: Buffer | string) => {
84
+ buffer += chunk.toString();
85
+ const lines = buffer.split('\n');
86
+ buffer = lines.pop() ?? ''; // Keep incomplete line in buffer
87
+
88
+ for (const line of lines) {
89
+ this.matchLine(id, line, streamName);
90
+ }
91
+ };
92
+
93
+ stream.on('data', onData);
94
+ stream.on('end', () => {
95
+ // Process remaining buffer
96
+ if (buffer) this.matchLine(id, buffer, streamName);
97
+ this.watchers.delete(id);
98
+ this.emit('watcher:end', id);
99
+ });
100
+
101
+ return id;
102
+ }
103
+
104
+ /**
105
+ * Watch both stdout and stderr of a ChildProcess.
106
+ */
107
+ watchProcess(
108
+ proc: { stdout?: NodeJS.ReadableStream | null; stderr?: NodeJS.ReadableStream | null },
109
+ options: WatchOptions,
110
+ ): string[] {
111
+ const ids: string[] = [];
112
+ if (proc.stdout) ids.push(this.watch(proc.stdout, options, 'stdout'));
113
+ if (proc.stderr) ids.push(this.watch(proc.stderr, options, 'stderr'));
114
+ return ids;
115
+ }
116
+
117
+ /** Stop a specific watcher */
118
+ unwatch(id: string): void {
119
+ this.watchers.delete(id);
120
+ }
121
+
122
+ /** Get match history for a watcher */
123
+ getHistory(id: string): WatchMatch[] {
124
+ return this.watchers.get(id)?.history ?? [];
125
+ }
126
+
127
+ /** Add a pattern to an existing watcher */
128
+ addPattern(id: string, pattern: WatchPattern): void {
129
+ const state = this.watchers.get(id);
130
+ if (state) state.patterns.push(pattern);
131
+ }
132
+
133
+ /** Remove a pattern by label from a watcher */
134
+ removePattern(id: string, label: string): void {
135
+ const state = this.watchers.get(id);
136
+ if (state) {
137
+ state.patterns = state.patterns.filter(p => p.label !== label);
138
+ }
139
+ }
140
+
141
+ private matchLine(watcherId: string, line: string, stream: 'stdout' | 'stderr'): void {
142
+ const state = this.watchers.get(watcherId);
143
+ if (!state) return;
144
+
145
+ const toRemove: number[] = [];
146
+
147
+ for (let i = 0; i < state.patterns.length; i++) {
148
+ const pattern = state.patterns[i];
149
+ const m = line.match(pattern.regex);
150
+ if (m) {
151
+ const match: WatchMatch = {
152
+ label: pattern.label,
153
+ line,
154
+ groups: m.slice(1),
155
+ timestamp: Date.now(),
156
+ stream,
157
+ };
158
+
159
+ state.history.push(match);
160
+ if (state.history.length > state.maxHistory) {
161
+ state.history.shift();
162
+ }
163
+
164
+ state.onMatch(match);
165
+ this.emit('match', match);
166
+
167
+ if (pattern.once) {
168
+ toRemove.push(i);
169
+ }
170
+ }
171
+ }
172
+
173
+ // Remove once-patterns in reverse order
174
+ for (let i = toRemove.length - 1; i >= 0; i--) {
175
+ state.patterns.splice(toRemove[i], 1);
176
+ }
177
+ }
178
+ }