opc-agent 1.3.0 → 1.3.2

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 (153) hide show
  1. package/CHANGELOG.md +6 -0
  2. package/CONTRIBUTING.md +75 -75
  3. package/README.md +358 -235
  4. package/README.zh-CN.md +415 -415
  5. package/dist/core/dashboard.d.ts +35 -0
  6. package/dist/core/dashboard.js +157 -0
  7. package/dist/core/priority.d.ts +52 -0
  8. package/dist/core/priority.js +102 -0
  9. package/dist/core/streaming.d.ts +56 -0
  10. package/dist/core/streaming.js +160 -0
  11. package/dist/deploy/hermes.js +22 -22
  12. package/dist/deploy/openclaw.js +31 -31
  13. package/dist/index.d.ts +8 -0
  14. package/dist/index.js +12 -1
  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 +50 -50
  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/webhook.ts +199 -199
  56. package/src/channels/websocket.ts +87 -87
  57. package/src/channels/wechat.ts +149 -149
  58. package/src/cli.ts +1 -119
  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/dashboard.ts +219 -0
  67. package/src/core/errors.ts +148 -148
  68. package/src/core/hitl.ts +138 -138
  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/priority.ts +140 -0
  73. package/src/core/rate-limiter.ts +128 -128
  74. package/src/core/room.ts +109 -109
  75. package/src/core/runtime.ts +152 -152
  76. package/src/core/sandbox.ts +101 -101
  77. package/src/core/security.ts +171 -171
  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 -0
  85. package/src/dtv/trust.ts +43 -0
  86. package/src/dtv/value.ts +47 -0
  87. package/src/i18n/index.ts +216 -216
  88. package/src/index.ts +6 -4
  89. package/src/marketplace/index.ts +223 -0
  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/schema/oad.ts +155 -154
  94. package/src/skills/base.ts +16 -16
  95. package/src/skills/document.ts +100 -100
  96. package/src/skills/http.ts +35 -35
  97. package/src/skills/index.ts +27 -27
  98. package/src/skills/scheduler.ts +80 -80
  99. package/src/skills/webhook-trigger.ts +59 -59
  100. package/src/templates/code-reviewer.ts +34 -30
  101. package/src/templates/customer-service.ts +80 -76
  102. package/src/templates/data-analyst.ts +70 -66
  103. package/src/templates/executive-assistant.ts +71 -71
  104. package/src/templates/financial-advisor.ts +60 -60
  105. package/src/templates/knowledge-base.ts +31 -27
  106. package/src/templates/legal-assistant.ts +71 -71
  107. package/src/templates/sales-assistant.ts +79 -75
  108. package/src/templates/teacher.ts +79 -75
  109. package/src/testing/index.ts +181 -181
  110. package/src/tools/calculator.ts +73 -73
  111. package/src/tools/datetime.ts +149 -149
  112. package/src/tools/json-transform.ts +187 -187
  113. package/src/tools/mcp.ts +76 -76
  114. package/src/tools/text-analysis.ts +116 -116
  115. package/templates/Dockerfile +15 -15
  116. package/templates/code-reviewer/README.md +27 -27
  117. package/templates/code-reviewer/oad.yaml +41 -41
  118. package/templates/customer-service/README.md +22 -22
  119. package/templates/customer-service/oad.yaml +36 -36
  120. package/templates/docker-compose.yml +21 -21
  121. package/templates/ecommerce-assistant/README.md +45 -45
  122. package/templates/ecommerce-assistant/oad.yaml +47 -47
  123. package/templates/knowledge-base/README.md +28 -28
  124. package/templates/knowledge-base/oad.yaml +38 -38
  125. package/templates/sales-assistant/README.md +26 -26
  126. package/templates/sales-assistant/oad.yaml +43 -43
  127. package/templates/tech-support/README.md +43 -43
  128. package/templates/tech-support/oad.yaml +45 -45
  129. package/tests/a2a.test.ts +66 -66
  130. package/tests/agent.test.ts +72 -72
  131. package/tests/analytics.test.ts +50 -50
  132. package/tests/channel.test.ts +39 -39
  133. package/tests/e2e.test.ts +134 -134
  134. package/tests/errors.test.ts +83 -83
  135. package/tests/hitl.test.ts +71 -71
  136. package/tests/i18n.test.ts +41 -41
  137. package/tests/mcp.test.ts +54 -54
  138. package/tests/oad.test.ts +68 -68
  139. package/tests/performance.test.ts +115 -115
  140. package/tests/plugin.test.ts +74 -74
  141. package/tests/room.test.ts +106 -106
  142. package/tests/runtime.test.ts +42 -42
  143. package/tests/sandbox.test.ts +46 -46
  144. package/tests/security.test.ts +60 -60
  145. package/tests/templates.test.ts +77 -77
  146. package/tests/v070.test.ts +76 -76
  147. package/tests/versioning.test.ts +75 -75
  148. package/tests/voice.test.ts +61 -61
  149. package/tests/webhook.test.ts +29 -29
  150. package/tests/workflow.test.ts +143 -143
  151. package/tsconfig.json +19 -19
  152. package/vitest.config.ts +9 -9
  153. package/src/traces/index.ts +0 -132
@@ -1,106 +1,106 @@
1
- import * as fs from 'fs';
2
- import * as path from 'path';
3
- import { Logger } from './logger';
4
-
5
- // ── Versioning Types ────────────────────────────────────────
6
-
7
- export interface VersionEntry {
8
- version: string;
9
- timestamp: number;
10
- description?: string;
11
- oadSnapshot: string; // serialized OAD YAML
12
- }
13
-
14
- export interface Migration {
15
- fromVersion: string;
16
- toVersion: string;
17
- migrate: (oad: Record<string, unknown>) => Record<string, unknown>;
18
- }
19
-
20
- // ── Version Manager ─────────────────────────────────────────
21
-
22
- export class VersionManager {
23
- private versions: VersionEntry[] = [];
24
- private migrations: Migration[] = [];
25
- private storePath: string;
26
- private logger = new Logger('versioning');
27
-
28
- constructor(storePath?: string) {
29
- this.storePath = storePath ?? '.opc-versions.json';
30
- this.load();
31
- }
32
-
33
- private load(): void {
34
- try {
35
- if (fs.existsSync(this.storePath)) {
36
- const data = JSON.parse(fs.readFileSync(this.storePath, 'utf-8'));
37
- this.versions = data.versions ?? [];
38
- }
39
- } catch {
40
- this.versions = [];
41
- }
42
- }
43
-
44
- private save(): void {
45
- fs.writeFileSync(this.storePath, JSON.stringify({ versions: this.versions }, null, 2));
46
- }
47
-
48
- snapshot(version: string, oadYaml: string, description?: string): void {
49
- this.versions.push({
50
- version,
51
- timestamp: Date.now(),
52
- description,
53
- oadSnapshot: oadYaml,
54
- });
55
- this.save();
56
- this.logger.info('Version snapshot saved', { version });
57
- }
58
-
59
- list(): VersionEntry[] {
60
- return [...this.versions];
61
- }
62
-
63
- get(version: string): VersionEntry | undefined {
64
- return this.versions.find(v => v.version === version);
65
- }
66
-
67
- getCurrent(): VersionEntry | undefined {
68
- return this.versions[this.versions.length - 1];
69
- }
70
-
71
- rollback(version: string): string | null {
72
- const entry = this.get(version);
73
- if (!entry) {
74
- this.logger.warn('Version not found', { version });
75
- return null;
76
- }
77
- this.logger.info('Rolling back to version', { version });
78
- return entry.oadSnapshot;
79
- }
80
-
81
- registerMigration(migration: Migration): void {
82
- this.migrations.push(migration);
83
- }
84
-
85
- migrate(oad: Record<string, unknown>, fromVersion: string, toVersion: string): Record<string, unknown> {
86
- let current = fromVersion;
87
- let result = { ...oad };
88
-
89
- while (current !== toVersion) {
90
- const migration = this.migrations.find(m => m.fromVersion === current);
91
- if (!migration) {
92
- throw new Error(`No migration path from ${current} to ${toVersion}`);
93
- }
94
- result = migration.migrate(result);
95
- current = migration.toVersion;
96
- this.logger.info('Migration applied', { from: migration.fromVersion, to: migration.toVersion });
97
- }
98
-
99
- return result;
100
- }
101
-
102
- clear(): void {
103
- this.versions = [];
104
- this.save();
105
- }
106
- }
1
+ import * as fs from 'fs';
2
+ import * as path from 'path';
3
+ import { Logger } from './logger';
4
+
5
+ // ── Versioning Types ────────────────────────────────────────
6
+
7
+ export interface VersionEntry {
8
+ version: string;
9
+ timestamp: number;
10
+ description?: string;
11
+ oadSnapshot: string; // serialized OAD YAML
12
+ }
13
+
14
+ export interface Migration {
15
+ fromVersion: string;
16
+ toVersion: string;
17
+ migrate: (oad: Record<string, unknown>) => Record<string, unknown>;
18
+ }
19
+
20
+ // ── Version Manager ─────────────────────────────────────────
21
+
22
+ export class VersionManager {
23
+ private versions: VersionEntry[] = [];
24
+ private migrations: Migration[] = [];
25
+ private storePath: string;
26
+ private logger = new Logger('versioning');
27
+
28
+ constructor(storePath?: string) {
29
+ this.storePath = storePath ?? '.opc-versions.json';
30
+ this.load();
31
+ }
32
+
33
+ private load(): void {
34
+ try {
35
+ if (fs.existsSync(this.storePath)) {
36
+ const data = JSON.parse(fs.readFileSync(this.storePath, 'utf-8'));
37
+ this.versions = data.versions ?? [];
38
+ }
39
+ } catch {
40
+ this.versions = [];
41
+ }
42
+ }
43
+
44
+ private save(): void {
45
+ fs.writeFileSync(this.storePath, JSON.stringify({ versions: this.versions }, null, 2));
46
+ }
47
+
48
+ snapshot(version: string, oadYaml: string, description?: string): void {
49
+ this.versions.push({
50
+ version,
51
+ timestamp: Date.now(),
52
+ description,
53
+ oadSnapshot: oadYaml,
54
+ });
55
+ this.save();
56
+ this.logger.info('Version snapshot saved', { version });
57
+ }
58
+
59
+ list(): VersionEntry[] {
60
+ return [...this.versions];
61
+ }
62
+
63
+ get(version: string): VersionEntry | undefined {
64
+ return this.versions.find(v => v.version === version);
65
+ }
66
+
67
+ getCurrent(): VersionEntry | undefined {
68
+ return this.versions[this.versions.length - 1];
69
+ }
70
+
71
+ rollback(version: string): string | null {
72
+ const entry = this.get(version);
73
+ if (!entry) {
74
+ this.logger.warn('Version not found', { version });
75
+ return null;
76
+ }
77
+ this.logger.info('Rolling back to version', { version });
78
+ return entry.oadSnapshot;
79
+ }
80
+
81
+ registerMigration(migration: Migration): void {
82
+ this.migrations.push(migration);
83
+ }
84
+
85
+ migrate(oad: Record<string, unknown>, fromVersion: string, toVersion: string): Record<string, unknown> {
86
+ let current = fromVersion;
87
+ let result = { ...oad };
88
+
89
+ while (current !== toVersion) {
90
+ const migration = this.migrations.find(m => m.fromVersion === current);
91
+ if (!migration) {
92
+ throw new Error(`No migration path from ${current} to ${toVersion}`);
93
+ }
94
+ result = migration.migrate(result);
95
+ current = migration.toVersion;
96
+ this.logger.info('Migration applied', { from: migration.fromVersion, to: migration.toVersion });
97
+ }
98
+
99
+ return result;
100
+ }
101
+
102
+ clear(): void {
103
+ this.versions = [];
104
+ this.save();
105
+ }
106
+ }
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
+ }