@coralai/sps-cli 0.49.16 → 0.50.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 (164) hide show
  1. package/dist/commands/cardAdd.d.ts.map +1 -1
  2. package/dist/commands/cardAdd.js +47 -67
  3. package/dist/commands/cardAdd.js.map +1 -1
  4. package/dist/commands/cardMarkComplete.js +1 -1
  5. package/dist/commands/cardMarkComplete.js.map +1 -1
  6. package/dist/commands/cardMarkStarted.js +1 -1
  7. package/dist/commands/cardMarkStarted.js.map +1 -1
  8. package/dist/commands/consoleCommand.js +3 -3
  9. package/dist/commands/consoleCommand.js.map +1 -1
  10. package/dist/commands/hookCommand.d.ts +25 -0
  11. package/dist/commands/hookCommand.d.ts.map +1 -1
  12. package/dist/commands/hookCommand.js +2 -2
  13. package/dist/commands/hookCommand.js.map +1 -1
  14. package/dist/console/index.d.ts +15 -0
  15. package/dist/console/index.d.ts.map +1 -0
  16. package/dist/console/index.js +177 -0
  17. package/dist/console/index.js.map +1 -0
  18. package/dist/console/lib/lockFile.d.ts +17 -0
  19. package/dist/console/lib/lockFile.d.ts.map +1 -0
  20. package/dist/console/lib/lockFile.js +61 -0
  21. package/dist/console/lib/lockFile.js.map +1 -0
  22. package/dist/console/lib/portPicker.d.ts +3 -0
  23. package/dist/console/lib/portPicker.d.ts.map +1 -0
  24. package/dist/console/lib/portPicker.js +25 -0
  25. package/dist/console/lib/portPicker.js.map +1 -0
  26. package/dist/console/lib/resultToJson.d.ts +18 -0
  27. package/dist/console/lib/resultToJson.d.ts.map +1 -0
  28. package/dist/console/lib/resultToJson.js +19 -0
  29. package/dist/console/lib/resultToJson.js.map +1 -0
  30. package/dist/console/routes/cards.d.ts +26 -0
  31. package/dist/console/routes/cards.d.ts.map +1 -0
  32. package/dist/console/routes/cards.js +85 -0
  33. package/dist/console/routes/cards.js.map +1 -0
  34. package/dist/console/routes/chat.d.ts +36 -0
  35. package/dist/console/routes/chat.d.ts.map +1 -0
  36. package/dist/console/routes/chat.js +458 -0
  37. package/dist/console/routes/chat.js.map +1 -0
  38. package/dist/console/routes/logs.d.ts +17 -0
  39. package/dist/console/routes/logs.d.ts.map +1 -0
  40. package/dist/console/routes/logs.js +159 -0
  41. package/dist/console/routes/logs.js.map +1 -0
  42. package/dist/console/routes/pipeline.d.ts +10 -0
  43. package/dist/console/routes/pipeline.d.ts.map +1 -0
  44. package/dist/console/routes/pipeline.js +39 -0
  45. package/dist/console/routes/pipeline.js.map +1 -0
  46. package/dist/console/routes/projects.d.ts +15 -0
  47. package/dist/console/routes/projects.d.ts.map +1 -0
  48. package/dist/console/routes/projects.js +132 -0
  49. package/dist/console/routes/projects.js.map +1 -0
  50. package/dist/console/routes/skills.d.ts +4 -0
  51. package/dist/console/routes/skills.d.ts.map +1 -0
  52. package/dist/console/routes/skills.js +91 -0
  53. package/dist/console/routes/skills.js.map +1 -0
  54. package/dist/console/routes/system.d.ts +7 -0
  55. package/dist/console/routes/system.d.ts.map +1 -0
  56. package/dist/console/routes/system.js +288 -0
  57. package/dist/console/routes/system.js.map +1 -0
  58. package/dist/console/routes/workers.d.ts +5 -0
  59. package/dist/console/routes/workers.d.ts.map +1 -0
  60. package/dist/console/routes/workers.js +149 -0
  61. package/dist/console/routes/workers.js.map +1 -0
  62. package/dist/console/sse/eventBus.d.ts +25 -0
  63. package/dist/console/sse/eventBus.d.ts.map +1 -0
  64. package/dist/console/sse/eventBus.js +32 -0
  65. package/dist/console/sse/eventBus.js.map +1 -0
  66. package/dist/console/sse/projectStream.d.ts +13 -0
  67. package/dist/console/sse/projectStream.d.ts.map +1 -0
  68. package/dist/console/sse/projectStream.js +84 -0
  69. package/dist/console/sse/projectStream.js.map +1 -0
  70. package/dist/infra/chokidarWatchers.d.ts +38 -0
  71. package/dist/infra/chokidarWatchers.d.ts.map +1 -0
  72. package/dist/infra/chokidarWatchers.js +213 -0
  73. package/dist/infra/chokidarWatchers.js.map +1 -0
  74. package/dist/infra/clock.d.ts +35 -0
  75. package/dist/infra/clock.d.ts.map +1 -0
  76. package/dist/infra/clock.js +45 -0
  77. package/dist/infra/clock.js.map +1 -0
  78. package/dist/infra/filesystem.d.ts +89 -0
  79. package/dist/infra/filesystem.d.ts.map +1 -0
  80. package/dist/infra/filesystem.js +247 -0
  81. package/dist/infra/filesystem.js.map +1 -0
  82. package/dist/infra/spawn.d.ts +67 -0
  83. package/dist/infra/spawn.d.ts.map +1 -0
  84. package/dist/infra/spawn.js +116 -0
  85. package/dist/infra/spawn.js.map +1 -0
  86. package/dist/infra/sseBus.d.ts +36 -0
  87. package/dist/infra/sseBus.d.ts.map +1 -0
  88. package/dist/infra/sseBus.js +72 -0
  89. package/dist/infra/sseBus.js.map +1 -0
  90. package/dist/interfaces/ACPClient.d.ts +1 -1
  91. package/dist/interfaces/ACPClient.d.ts.map +1 -1
  92. package/dist/main.js +1 -1
  93. package/dist/main.js.map +1 -1
  94. package/dist/providers/LocalACPClient.d.ts +1 -1
  95. package/dist/providers/LocalACPClient.d.ts.map +1 -1
  96. package/dist/services/CardService.d.ts +86 -0
  97. package/dist/services/CardService.d.ts.map +1 -0
  98. package/dist/services/CardService.js +313 -0
  99. package/dist/services/CardService.js.map +1 -0
  100. package/dist/services/ChatService.d.ts +62 -0
  101. package/dist/services/ChatService.d.ts.map +1 -0
  102. package/dist/services/ChatService.js +157 -0
  103. package/dist/services/ChatService.js.map +1 -0
  104. package/dist/services/LogService.d.ts +46 -0
  105. package/dist/services/LogService.d.ts.map +1 -0
  106. package/dist/services/LogService.js +185 -0
  107. package/dist/services/LogService.js.map +1 -0
  108. package/dist/services/PipelineService.d.ts +71 -0
  109. package/dist/services/PipelineService.d.ts.map +1 -0
  110. package/dist/services/PipelineService.js +349 -0
  111. package/dist/services/PipelineService.js.map +1 -0
  112. package/dist/services/ProjectService.d.ts +105 -0
  113. package/dist/services/ProjectService.d.ts.map +1 -0
  114. package/dist/services/ProjectService.js +346 -0
  115. package/dist/services/ProjectService.js.map +1 -0
  116. package/dist/services/SkillService.d.ts +38 -0
  117. package/dist/services/SkillService.d.ts.map +1 -0
  118. package/dist/services/SkillService.js +301 -0
  119. package/dist/services/SkillService.js.map +1 -0
  120. package/dist/services/WorkerService.d.ts +83 -0
  121. package/dist/services/WorkerService.d.ts.map +1 -0
  122. package/dist/services/WorkerService.js +262 -0
  123. package/dist/services/WorkerService.js.map +1 -0
  124. package/dist/services/container.d.ts +47 -0
  125. package/dist/services/container.d.ts.map +1 -0
  126. package/dist/services/container.js +77 -0
  127. package/dist/services/container.js.map +1 -0
  128. package/dist/services/defaults.d.ts +15 -0
  129. package/dist/services/defaults.d.ts.map +1 -0
  130. package/dist/services/defaults.js +11 -0
  131. package/dist/services/defaults.js.map +1 -0
  132. package/dist/services/executors.d.ts +38 -0
  133. package/dist/services/executors.d.ts.map +1 -0
  134. package/dist/services/executors.js +157 -0
  135. package/dist/services/executors.js.map +1 -0
  136. package/dist/services/ports.d.ts +17 -0
  137. package/dist/services/ports.d.ts.map +1 -0
  138. package/dist/services/ports.js +2 -0
  139. package/dist/services/ports.js.map +1 -0
  140. package/dist/shared/domainEvents.d.ts +118 -0
  141. package/dist/shared/domainEvents.d.ts.map +1 -0
  142. package/dist/shared/domainEvents.js +40 -0
  143. package/dist/shared/domainEvents.js.map +1 -0
  144. package/dist/shared/errors.d.ts +59 -0
  145. package/dist/shared/errors.d.ts.map +1 -0
  146. package/dist/shared/errors.js +79 -0
  147. package/dist/shared/errors.js.map +1 -0
  148. package/dist/shared/result.d.ts +51 -0
  149. package/dist/shared/result.d.ts.map +1 -0
  150. package/dist/shared/result.js +48 -0
  151. package/dist/shared/result.js.map +1 -0
  152. package/dist/shared/runtimePaths.d.ts +78 -0
  153. package/dist/shared/runtimePaths.d.ts.map +1 -0
  154. package/dist/shared/runtimePaths.js +179 -0
  155. package/dist/shared/runtimePaths.js.map +1 -0
  156. package/dist/shared/runtimeSchemas.d.ts +324 -0
  157. package/dist/shared/runtimeSchemas.d.ts.map +1 -0
  158. package/dist/shared/runtimeSchemas.js +124 -0
  159. package/dist/shared/runtimeSchemas.js.map +1 -0
  160. package/dist/shared/types.d.ts +12 -0
  161. package/dist/shared/types.d.ts.map +1 -0
  162. package/dist/shared/types.js +2 -0
  163. package/dist/shared/types.js.map +1 -0
  164. package/package.json +1 -1
@@ -0,0 +1,288 @@
1
+ /**
2
+ * @module console/routes/system
3
+ * @description 系统信息:版本、运行时、env(脱敏)、doctor 聚合
4
+ */
5
+ import { spawn } from 'node:child_process';
6
+ import { createHash } from 'node:crypto';
7
+ import { chmodSync, existsSync, mkdirSync, readdirSync, readFileSync, statSync, writeFileSync } from 'node:fs';
8
+ import { dirname, resolve } from 'node:path';
9
+ import { Hono } from 'hono';
10
+ const HOME = process.env.HOME || '/home/coral';
11
+ const ENV_PATH = resolve(HOME, '.coral', 'env');
12
+ function maskSecret(value) {
13
+ if (!value)
14
+ return '';
15
+ if (value.length <= 6)
16
+ return '****';
17
+ return value.slice(0, 4) + '****';
18
+ }
19
+ const SECRET_KEY_PATTERNS = [
20
+ /_TOKEN$/,
21
+ /_KEY$/,
22
+ /_SECRET$/,
23
+ /_PASSWORD$/,
24
+ /_PASS$/,
25
+ /^(ANTHROPIC|OPENAI|CLAUDE|PLANE|TRELLO|MATRIX)_/,
26
+ ];
27
+ function isSecret(key) {
28
+ return SECRET_KEY_PATTERNS.some((p) => p.test(key));
29
+ }
30
+ export function createSystemRoute(version, startedAt) {
31
+ const app = new Hono();
32
+ app.get('/info', (c) => {
33
+ return c.json({
34
+ version,
35
+ nodeVersion: process.version,
36
+ startedAt: startedAt.toISOString(),
37
+ uptimeMs: Date.now() - startedAt.getTime(),
38
+ platform: process.platform,
39
+ pid: process.pid,
40
+ });
41
+ });
42
+ app.get('/env', (c) => {
43
+ if (!existsSync(ENV_PATH)) {
44
+ return c.json({ path: ENV_PATH, exists: false, entries: [] });
45
+ }
46
+ const raw = readFileSync(ENV_PATH, 'utf-8');
47
+ const entries = [];
48
+ for (const line of raw.split('\n')) {
49
+ const t = line.trim();
50
+ if (!t || t.startsWith('#'))
51
+ continue;
52
+ const m = t.match(/^(?:export\s+)?([A-Z_][A-Z0-9_]*)=["']?(.*?)["']?\s*$/);
53
+ if (!m)
54
+ continue;
55
+ const key = m[1] ?? '';
56
+ const value = m[2] ?? '';
57
+ const masked = isSecret(key);
58
+ entries.push({ key, value: masked ? maskSecret(value) : value, masked });
59
+ }
60
+ return c.json({ path: ENV_PATH, exists: true, entries });
61
+ });
62
+ /**
63
+ * GET /api/system/env/raw — Return raw ~/.coral/env content + SHA-256 etag.
64
+ * Values NOT masked (for editing). UI should warn user before showing.
65
+ */
66
+ app.get('/env/raw', (c) => {
67
+ if (!existsSync(ENV_PATH)) {
68
+ return c.json({ path: ENV_PATH, exists: false, content: '', etag: '' });
69
+ }
70
+ const content = readFileSync(ENV_PATH, 'utf-8');
71
+ const etag = createHash('sha256').update(content).digest('hex').slice(0, 16);
72
+ return c.json({ path: ENV_PATH, exists: true, content, etag });
73
+ });
74
+ /**
75
+ * PATCH /api/system/env — Overwrite ~/.coral/env with body.content.
76
+ * Optimistic lock via body.etag (or If-Match). Keeps file mode 0600.
77
+ * Creates ~/.coral/ directory if missing.
78
+ */
79
+ app.patch('/env', async (c) => {
80
+ const body = (await c.req.json().catch(() => null));
81
+ if (!body || typeof body.content !== 'string') {
82
+ return c.json({ type: 'validation', title: 'content required', status: 422 }, 422);
83
+ }
84
+ const ifMatch = body.etag ?? c.req.header('If-Match');
85
+ const current = existsSync(ENV_PATH) ? readFileSync(ENV_PATH, 'utf-8') : '';
86
+ const currentEtag = existsSync(ENV_PATH)
87
+ ? createHash('sha256').update(current).digest('hex').slice(0, 16)
88
+ : '';
89
+ // Create-new allowed without etag; update requires etag match
90
+ if (existsSync(ENV_PATH)) {
91
+ if (!ifMatch) {
92
+ return c.json({ type: 'validation', title: 'etag required', status: 422 }, 422);
93
+ }
94
+ if (ifMatch !== currentEtag) {
95
+ return c.json({
96
+ type: 'conflict',
97
+ title: 'etag mismatch',
98
+ status: 409,
99
+ detail: 'env changed since you loaded it; reload and try again',
100
+ currentEtag,
101
+ }, 409);
102
+ }
103
+ }
104
+ try {
105
+ mkdirSync(dirname(ENV_PATH), { recursive: true });
106
+ writeFileSync(ENV_PATH, body.content);
107
+ chmodSync(ENV_PATH, 0o600);
108
+ }
109
+ catch (err) {
110
+ return c.json({
111
+ type: 'internal',
112
+ title: 'write failed',
113
+ status: 500,
114
+ detail: err instanceof Error ? err.message : String(err),
115
+ }, 500);
116
+ }
117
+ const newEtag = createHash('sha256').update(body.content).digest('hex').slice(0, 16);
118
+ return c.json({ etag: newEtag });
119
+ });
120
+ /**
121
+ * GET /api/system/latest-version — Fetch latest npm version for self-update check.
122
+ * Spawns `npm view @coralai/sps-cli version`. 10s timeout.
123
+ */
124
+ app.get('/latest-version', async (c) => {
125
+ const result = await new Promise((r) => {
126
+ const child = spawn('npm', ['view', '@coralai/sps-cli', 'version'], {
127
+ stdio: ['ignore', 'pipe', 'pipe'],
128
+ });
129
+ let stdout = '';
130
+ let stderr = '';
131
+ const timer = setTimeout(() => {
132
+ try {
133
+ child.kill('SIGKILL');
134
+ }
135
+ catch { /* noop */ }
136
+ r({ ok: false, error: 'npm view timeout (10s)' });
137
+ }, 10_000);
138
+ child.stdout?.on('data', (d) => (stdout += d.toString()));
139
+ child.stderr?.on('data', (d) => (stderr += d.toString()));
140
+ child.on('close', (code) => {
141
+ clearTimeout(timer);
142
+ if (code === 0)
143
+ r({ ok: true, version: stdout.trim() });
144
+ else
145
+ r({ ok: false, error: stderr.trim() || `npm exit ${code}` });
146
+ });
147
+ child.on('error', (err) => {
148
+ clearTimeout(timer);
149
+ r({ ok: false, error: err.message });
150
+ });
151
+ });
152
+ if (!result.ok) {
153
+ return c.json({ type: 'internal', title: 'version check failed', status: 500, detail: result.error }, 500);
154
+ }
155
+ return c.json({ current: version, latest: result.version, upToDate: version === result.version });
156
+ });
157
+ /**
158
+ * POST /api/system/upgrade — Run `npm i -g @coralai/sps-cli@latest`.
159
+ * Safety: refuses if any project has a running pipeline (supervisor.pid alive).
160
+ * Streams stdout/stderr back as plain text (user must refresh to re-bind new version).
161
+ */
162
+ app.post('/upgrade', async (c) => {
163
+ // Safety: check all projects for running pipelines
164
+ const projectsDir = resolve(HOME, '.coral', 'projects');
165
+ const runningProjects = [];
166
+ if (existsSync(projectsDir)) {
167
+ const names = readdirSync(projectsDir, { withFileTypes: true })
168
+ .filter((e) => e.isDirectory())
169
+ .map((e) => e.name);
170
+ for (const name of names) {
171
+ const pidFile = resolve(projectsDir, name, 'runtime', 'supervisor.pid');
172
+ if (!existsSync(pidFile))
173
+ continue;
174
+ try {
175
+ const pid = Number.parseInt(readFileSync(pidFile, 'utf-8').trim(), 10);
176
+ if (pid > 0) {
177
+ try {
178
+ process.kill(pid, 0);
179
+ runningProjects.push(name);
180
+ }
181
+ catch { /* dead */ }
182
+ }
183
+ }
184
+ catch { /* ignore */ }
185
+ }
186
+ }
187
+ if (runningProjects.length > 0) {
188
+ return c.json({
189
+ type: 'conflict',
190
+ title: 'pipelines running',
191
+ status: 409,
192
+ detail: `Stop pipelines first: ${runningProjects.join(', ')}`,
193
+ projects: runningProjects,
194
+ }, 409);
195
+ }
196
+ // Run npm i -g (streams output, returns combined log)
197
+ const result = await new Promise((r) => {
198
+ const child = spawn('npm', ['i', '-g', '@coralai/sps-cli@latest'], {
199
+ stdio: ['ignore', 'pipe', 'pipe'],
200
+ });
201
+ let output = '';
202
+ const timer = setTimeout(() => {
203
+ try {
204
+ child.kill('SIGKILL');
205
+ }
206
+ catch { /* noop */ }
207
+ r({ ok: false, output: output + '\n[timeout 120s]' });
208
+ }, 120_000);
209
+ child.stdout?.on('data', (d) => (output += d.toString()));
210
+ child.stderr?.on('data', (d) => (output += d.toString()));
211
+ child.on('close', (code) => {
212
+ clearTimeout(timer);
213
+ r({ ok: code === 0, output });
214
+ });
215
+ child.on('error', (err) => {
216
+ clearTimeout(timer);
217
+ r({ ok: false, output: output + `\n[spawn error: ${err.message}]` });
218
+ });
219
+ });
220
+ return c.json({ ok: result.ok, output: result.output });
221
+ });
222
+ app.get('/doctor/all', (c) => {
223
+ const projectsDir = resolve(HOME, '.coral', 'projects');
224
+ const report = [];
225
+ if (existsSync(projectsDir)) {
226
+ const names = readdirSync(projectsDir, { withFileTypes: true })
227
+ .filter((e) => e.isDirectory())
228
+ .map((e) => e.name);
229
+ for (const name of names) {
230
+ const issues = [];
231
+ const dir = resolve(projectsDir, name);
232
+ if (!existsSync(resolve(dir, 'conf')))
233
+ issues.push('missing conf');
234
+ if (!existsSync(resolve(dir, 'cards')))
235
+ issues.push('missing cards/');
236
+ const runtime = resolve(dir, 'runtime');
237
+ if (existsSync(runtime)) {
238
+ const markers = readdirSync(runtime).filter((f) => /worker-(?:worker-)?\d+-current\.json$/.test(f));
239
+ for (const mf of markers) {
240
+ try {
241
+ const stat = statSync(resolve(runtime, mf));
242
+ const ageMin = Math.floor((Date.now() - stat.mtimeMs) / 60000);
243
+ if (ageMin > 60)
244
+ issues.push(`stale marker ${mf} (${ageMin}m)`);
245
+ }
246
+ catch {
247
+ /* ignore */
248
+ }
249
+ }
250
+ }
251
+ report.push({ project: name, issues, ok: issues.length === 0 });
252
+ }
253
+ }
254
+ return c.json({ data: report });
255
+ });
256
+ /**
257
+ * POST /api/client-errors — Front-end error telemetry sink.
258
+ * Accepts { message, stack, url, ua, ts } and writes to server log.
259
+ * No persistence, just log for now. Limits body to 8KB to prevent abuse.
260
+ */
261
+ app.post('/client-errors', async (c) => {
262
+ const raw = await c.req.text().catch(() => '');
263
+ if (raw.length > 8192) {
264
+ return c.json({ type: 'validation', title: 'payload too large', status: 413 }, 413);
265
+ }
266
+ let body;
267
+ try {
268
+ body = JSON.parse(raw);
269
+ }
270
+ catch {
271
+ return c.json({ type: 'validation', title: 'invalid JSON', status: 422 }, 422);
272
+ }
273
+ // Log to stderr via simple structured line — server log collection reads stderr
274
+ const line = [
275
+ `[client-error]`,
276
+ body.ts ?? new Date().toISOString(),
277
+ body.url ?? '-',
278
+ body.message ?? '-',
279
+ ].join(' ');
280
+ process.stderr.write(line + '\n');
281
+ if (body.stack) {
282
+ process.stderr.write(body.stack.split('\n').map((l) => ` ${l}`).join('\n') + '\n');
283
+ }
284
+ return c.body(null, 204);
285
+ });
286
+ return app;
287
+ }
288
+ //# sourceMappingURL=system.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"system.js","sourceRoot":"","sources":["../../../src/console/routes/system.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import { Hono } from 'hono';
2
+ import type { WorkerService } from '../../services/WorkerService.js';
3
+ export declare function createWorkersRoute(workers: WorkerService): Hono;
4
+ export declare function createWorkersAggregateRoute(workers: WorkerService): Hono;
5
+ //# sourceMappingURL=workers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workers.d.ts","sourceRoot":"","sources":["../../../src/console/routes/workers.ts"],"names":[],"mappings":"AASA,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,iCAAiC,CAAC;AASrE,wBAAgB,kBAAkB,CAAC,OAAO,EAAE,aAAa,GAAG,IAAI,CA6C/D;AAED,wBAAgB,2BAA2B,CAAC,OAAO,EAAE,aAAa,GAAG,IAAI,CAuBxE"}
@@ -0,0 +1,149 @@
1
+ /**
2
+ * @module console/routes/workers
3
+ * @description Worker REST —— 全走 WorkerService
4
+ *
5
+ * @layer console
6
+ */
7
+ import { createReadStream, existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
8
+ import { resolve } from 'node:path';
9
+ import { createInterface } from 'node:readline';
10
+ import { Hono } from 'hono';
11
+ import { home, logsDir, workerLogLineTag, workerMarkerFile, } from '../../shared/runtimePaths.js';
12
+ import { sendResult } from '../lib/resultToJson.js';
13
+ export function createWorkersRoute(workers) {
14
+ const app = new Hono();
15
+ app.get('/:project/workers', async (c) => {
16
+ const r = await workers.listByProject(c.req.param('project'));
17
+ if (!r.ok)
18
+ return sendResult(c, r);
19
+ return c.json({ data: r.value });
20
+ });
21
+ app.get('/:project/workers/:slot', async (c) => {
22
+ const slot = Number.parseInt(c.req.param('slot'), 10);
23
+ const r = await workers.getBySlot(c.req.param('project'), slot);
24
+ if (!r.ok)
25
+ return sendResult(c, r);
26
+ const project = c.req.param('project');
27
+ const markerPath = workerMarkerFile(project, `worker-${slot}`);
28
+ let markerData = null;
29
+ try {
30
+ markerData = JSON.parse(readFileSync(markerPath, 'utf-8'));
31
+ }
32
+ catch {
33
+ /* ignore */
34
+ }
35
+ const recentLogs = await readWorkerLogTail(project, slot, 20);
36
+ return c.json({
37
+ ...r.value,
38
+ markerPath: markerPath.replace(home(), '~'),
39
+ markerData,
40
+ recentLogs,
41
+ });
42
+ });
43
+ app.post('/:project/workers/:slot/kill', async (c) => {
44
+ const slot = Number.parseInt(c.req.param('slot'), 10);
45
+ return sendResult(c, await workers.kill(c.req.param('project'), slot));
46
+ });
47
+ app.post('/:project/workers/:slot/launch', async (c) => {
48
+ const body = await c.req.json().catch(() => ({}));
49
+ const seq = typeof body?.seq === 'number' ? body.seq : Number.parseInt(String(body?.seq ?? 0), 10);
50
+ if (!Number.isInteger(seq) || seq <= 0) {
51
+ return c.json({ type: 'validation', title: 'seq required', status: 422 }, 422);
52
+ }
53
+ return sendResult(c, await workers.launch(c.req.param('project'), seq));
54
+ });
55
+ return app;
56
+ }
57
+ export function createWorkersAggregateRoute(workers) {
58
+ const app = new Hono();
59
+ app.get('/all', async (c) => {
60
+ const r = await workers.aggregate();
61
+ if (!r.ok)
62
+ return sendResult(c, r);
63
+ const enrich = async (list) => {
64
+ const out = [];
65
+ for (const w of list) {
66
+ out.push({ ...w, lastLogLine: await readLatestLogLine(w.project, w.slot) });
67
+ }
68
+ return out;
69
+ };
70
+ return c.json({
71
+ alerts: await enrich(r.value.alerts),
72
+ active: await enrich(r.value.active),
73
+ capacity: r.value.capacity,
74
+ });
75
+ });
76
+ return app;
77
+ }
78
+ // ─── 日志 tail helpers(Delivery 专属) ──────────────────────────────
79
+ async function readWorkerLogTail(project, slot, limit) {
80
+ const dir = logsDir(project);
81
+ if (!existsSync(dir))
82
+ return [];
83
+ const candidates = readdirSync(dir)
84
+ .filter((f) => f.endsWith('.log'))
85
+ .map((f) => ({ f, full: resolve(dir, f), mtime: statSync(resolve(dir, f)).mtimeMs }))
86
+ .sort((a, b) => b.mtime - a.mtime);
87
+ const pipeline = candidates.find((c) => c.f.startsWith('pipeline-'));
88
+ const file = pipeline?.full ?? candidates[0]?.full;
89
+ if (!file)
90
+ return [];
91
+ const MAX_BYTES = 4 * 1024 * 1024;
92
+ const stat = statSync(file);
93
+ const start = Math.max(0, stat.size - MAX_BYTES);
94
+ const matches = [];
95
+ const slotTag = workerLogLineTag(slot);
96
+ await new Promise((done) => {
97
+ const stream = createReadStream(file, { start, encoding: 'utf-8' });
98
+ const rl = createInterface({ input: stream });
99
+ rl.on('line', (raw) => {
100
+ if (!raw.includes(slotTag))
101
+ return;
102
+ const cleaned = raw.replace(/\[[0-9;]*m/g, '');
103
+ const m = cleaned.match(/(\d{4}-\d{2}-\d{2}[T ][\d:.]+Z?)\s*(?:\[)?(DEBUG|INFO|WARN|WARNING|ERROR|TRACE)\]?\s*(.*)$/i);
104
+ matches.push({
105
+ ts: m?.[1] ?? null,
106
+ level: (m?.[2] ?? 'info').toLowerCase().replace('warning', 'warn'),
107
+ msg: m?.[3] ?? cleaned,
108
+ });
109
+ if (matches.length > limit * 3)
110
+ matches.splice(0, matches.length - limit * 3);
111
+ });
112
+ rl.on('close', () => done());
113
+ rl.on('error', () => done());
114
+ });
115
+ return matches.slice(-limit);
116
+ }
117
+ async function readLatestLogLine(project, slot) {
118
+ const dir = logsDir(project);
119
+ if (!existsSync(dir))
120
+ return null;
121
+ const candidates = readdirSync(dir)
122
+ .filter((f) => f.endsWith('.log'))
123
+ .map((f) => ({ f, full: resolve(dir, f), mtime: statSync(resolve(dir, f)).mtimeMs }))
124
+ .sort((a, b) => b.mtime - a.mtime);
125
+ const pipeline = candidates.find((c) => c.f.startsWith('pipeline-'));
126
+ const file = pipeline?.full ?? candidates[0]?.full;
127
+ if (!file)
128
+ return null;
129
+ const MAX_BYTES = 512 * 1024;
130
+ const stat = statSync(file);
131
+ const start = Math.max(0, stat.size - MAX_BYTES);
132
+ const slotTag = workerLogLineTag(slot);
133
+ let latest = null;
134
+ await new Promise((done) => {
135
+ const stream = createReadStream(file, { start, encoding: 'utf-8' });
136
+ const rl = createInterface({ input: stream });
137
+ rl.on('line', (raw) => {
138
+ if (!raw.includes(slotTag))
139
+ return;
140
+ const cleaned = raw.replace(/\[[0-9;]*m/g, '');
141
+ const m = cleaned.match(/(\d{4}-\d{2}-\d{2}[T ][\d:.]+Z?)\s*(?:\[[^\]]+\]\s*)?(.*)$/);
142
+ latest = { ts: m?.[1] ?? null, msg: (m?.[2] ?? cleaned).slice(0, 200) };
143
+ });
144
+ rl.on('close', () => done());
145
+ rl.on('error', () => done());
146
+ });
147
+ return latest;
148
+ }
149
+ //# sourceMappingURL=workers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workers.js","sourceRoot":"","sources":["../../../src/console/routes/workers.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,25 @@
1
+ /**
2
+ * @module console/sse/eventBus
3
+ * @description 内部事件总线 —— watchers 推事件,SSE handlers 订阅推给客户端
4
+ *
5
+ * @role core
6
+ * @layer console
7
+ * @boundedContext console
8
+ */
9
+ import { EventEmitter } from 'node:events';
10
+ export interface BusEvent {
11
+ id: number;
12
+ event: string;
13
+ data: unknown;
14
+ ts: number;
15
+ }
16
+ declare class ConsoleEventBus extends EventEmitter {
17
+ private lastEventId;
18
+ private history;
19
+ publish(event: string, data: unknown): number;
20
+ since(lastEventId: number): BusEvent[];
21
+ clearHistory(): void;
22
+ }
23
+ export declare const eventBus: ConsoleEventBus;
24
+ export {};
25
+ //# sourceMappingURL=eventBus.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eventBus.d.ts","sourceRoot":"","sources":["../../../src/console/sse/eventBus.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAE3C,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,OAAO,CAAC;IACd,EAAE,EAAE,MAAM,CAAC;CACZ;AAID,cAAM,eAAgB,SAAQ,YAAY;IACxC,OAAO,CAAC,WAAW,CAAK;IACxB,OAAO,CAAC,OAAO,CAAkB;IAEjC,OAAO,CAAC,KAAK,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,GAAG,MAAM;IAU7C,KAAK,CAAC,WAAW,EAAE,MAAM,GAAG,QAAQ,EAAE;IAItC,YAAY,IAAI,IAAI;CAGrB;AAED,eAAO,MAAM,QAAQ,iBAAwB,CAAC"}
@@ -0,0 +1,32 @@
1
+ /**
2
+ * @module console/sse/eventBus
3
+ * @description 内部事件总线 —— watchers 推事件,SSE handlers 订阅推给客户端
4
+ *
5
+ * @role core
6
+ * @layer console
7
+ * @boundedContext console
8
+ */
9
+ import { EventEmitter } from 'node:events';
10
+ const MAX_HISTORY = 1000;
11
+ class ConsoleEventBus extends EventEmitter {
12
+ lastEventId = 0;
13
+ history = [];
14
+ publish(event, data) {
15
+ const id = ++this.lastEventId;
16
+ const record = { id, event, data, ts: Date.now() };
17
+ this.history.push(record);
18
+ if (this.history.length > MAX_HISTORY)
19
+ this.history.shift();
20
+ this.emit(event, data);
21
+ this.emit('*', record);
22
+ return id;
23
+ }
24
+ since(lastEventId) {
25
+ return this.history.filter((h) => h.id > lastEventId);
26
+ }
27
+ clearHistory() {
28
+ this.history = [];
29
+ }
30
+ }
31
+ export const eventBus = new ConsoleEventBus();
32
+ //# sourceMappingURL=eventBus.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"eventBus.js","sourceRoot":"","sources":["../../../src/console/sse/eventBus.ts"],"names":[],"mappings":"AAAA;;;;;;;GAOG;AACH,OAAO,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AAS3C,MAAM,WAAW,GAAG,IAAI,CAAC;AAEzB,MAAM,eAAgB,SAAQ,YAAY;IAChC,WAAW,GAAG,CAAC,CAAC;IAChB,OAAO,GAAe,EAAE,CAAC;IAEjC,OAAO,CAAC,KAAa,EAAE,IAAa;QAClC,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,WAAW,CAAC;QAC9B,MAAM,MAAM,GAAa,EAAE,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,EAAE,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;QAC7D,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAC1B,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,WAAW;YAAE,IAAI,CAAC,OAAO,CAAC,KAAK,EAAE,CAAC;QAC5D,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QACvB,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,KAAK,CAAC,WAAmB;QACvB,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,WAAW,CAAC,CAAC;IACxD,CAAC;IAED,YAAY;QACV,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC;IACpB,CAAC;CACF;AAED,MAAM,CAAC,MAAM,QAAQ,GAAG,IAAI,eAAe,EAAE,CAAC"}
@@ -0,0 +1,13 @@
1
+ /**
2
+ * @module console/sse/projectStream
3
+ * @description /stream/projects/:name —— 单项目的 card / worker / pipeline 事件流
4
+ *
5
+ * @layer console
6
+ *
7
+ * 订阅 SseEventBus,过滤 project 匹配的 DomainEvent,以 SSE 格式写给客户端。
8
+ * 支持 Last-Event-ID 断线补偿。
9
+ */
10
+ import { Hono } from 'hono';
11
+ import type { SseEventBus } from '../../infra/sseBus.js';
12
+ export declare function createProjectStreamRoute(bus: SseEventBus): Hono;
13
+ //# sourceMappingURL=projectStream.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projectStream.d.ts","sourceRoot":"","sources":["../../../src/console/sse/projectStream.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAC5B,OAAO,KAAK,EAAe,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAatE,wBAAgB,wBAAwB,CAAC,GAAG,EAAE,WAAW,GAAG,IAAI,CAqE/D"}
@@ -0,0 +1,84 @@
1
+ /**
2
+ * @module console/sse/projectStream
3
+ * @description /stream/projects/:name —— 单项目的 card / worker / pipeline 事件流
4
+ *
5
+ * @layer console
6
+ *
7
+ * 订阅 SseEventBus,过滤 project 匹配的 DomainEvent,以 SSE 格式写给客户端。
8
+ * 支持 Last-Event-ID 断线补偿。
9
+ */
10
+ import { Hono } from 'hono';
11
+ const HEARTBEAT_MS = 15_000;
12
+ function formatSse(id, event, data) {
13
+ return `id: ${id}\nevent: ${event}\ndata: ${JSON.stringify(data)}\n\n`;
14
+ }
15
+ function isProjectEvent(record, project) {
16
+ const d = record.data;
17
+ return d?.project === project;
18
+ }
19
+ export function createProjectStreamRoute(bus) {
20
+ const app = new Hono();
21
+ app.get('/:project', (c) => {
22
+ const project = c.req.param('project');
23
+ const lastEventId = Number.parseInt(c.req.header('last-event-id') ?? '0', 10) || 0;
24
+ c.header('Content-Type', 'text/event-stream');
25
+ c.header('Cache-Control', 'no-cache, no-transform');
26
+ c.header('Connection', 'keep-alive');
27
+ c.header('X-Accel-Buffering', 'no');
28
+ const stream = new ReadableStream({
29
+ start(controller) {
30
+ const enc = new TextEncoder();
31
+ let closed = false;
32
+ const safeWrite = (chunk) => {
33
+ if (closed)
34
+ return;
35
+ try {
36
+ controller.enqueue(enc.encode(chunk));
37
+ }
38
+ catch {
39
+ closed = true;
40
+ }
41
+ };
42
+ // 1. 补发断线期间的历史事件
43
+ if (lastEventId > 0) {
44
+ const history = bus.since(lastEventId).filter((r) => isProjectEvent(r, project));
45
+ for (const r of history) {
46
+ safeWrite(formatSse(r.id, r.event, r.data));
47
+ }
48
+ }
49
+ // 2. 订阅实时
50
+ const cancel = bus.subscribeRaw((record) => {
51
+ if (!isProjectEvent(record, project))
52
+ return;
53
+ safeWrite(formatSse(record.id, record.event, record.data));
54
+ });
55
+ // 3. 心跳保活
56
+ const heartbeat = setInterval(() => {
57
+ safeWrite(`: heartbeat ${Date.now()}\n\n`);
58
+ }, HEARTBEAT_MS);
59
+ // 4. 客户端断开 → 清理
60
+ c.req.raw.signal?.addEventListener('abort', () => {
61
+ closed = true;
62
+ cancel();
63
+ clearInterval(heartbeat);
64
+ try {
65
+ controller.close();
66
+ }
67
+ catch {
68
+ /* ignore */
69
+ }
70
+ });
71
+ },
72
+ });
73
+ return new Response(stream, {
74
+ headers: {
75
+ 'Content-Type': 'text/event-stream',
76
+ 'Cache-Control': 'no-cache, no-transform',
77
+ Connection: 'keep-alive',
78
+ 'X-Accel-Buffering': 'no',
79
+ },
80
+ });
81
+ });
82
+ return app;
83
+ }
84
+ //# sourceMappingURL=projectStream.js.map