@coralai/sps-cli 0.44.0 → 0.45.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 (62) hide show
  1. package/README.md +1 -1
  2. package/dist/console-assets/assets/index-BGs71Um2.css +1 -0
  3. package/dist/console-assets/assets/index-BsHLiTM2.js +236 -0
  4. package/dist/console-assets/index.html +2 -2
  5. package/dist/console-server/index.d.ts.map +1 -1
  6. package/dist/console-server/index.js +25 -5
  7. package/dist/console-server/index.js.map +1 -1
  8. package/dist/console-server/lib/cardReader.d.ts +27 -0
  9. package/dist/console-server/lib/cardReader.d.ts.map +1 -0
  10. package/dist/console-server/lib/cardReader.js +131 -0
  11. package/dist/console-server/lib/cardReader.js.map +1 -0
  12. package/dist/console-server/lib/spawnCli.d.ts +31 -0
  13. package/dist/console-server/lib/spawnCli.d.ts.map +1 -0
  14. package/dist/console-server/lib/spawnCli.js +64 -0
  15. package/dist/console-server/lib/spawnCli.js.map +1 -0
  16. package/dist/console-server/routes/cards.d.ts +7 -0
  17. package/dist/console-server/routes/cards.d.ts.map +1 -0
  18. package/dist/console-server/routes/cards.js +81 -0
  19. package/dist/console-server/routes/cards.js.map +1 -0
  20. package/dist/console-server/routes/chat.d.ts +24 -0
  21. package/dist/console-server/routes/chat.d.ts.map +1 -0
  22. package/dist/console-server/routes/chat.js +335 -0
  23. package/dist/console-server/routes/chat.js.map +1 -0
  24. package/dist/console-server/routes/logs.d.ts +14 -0
  25. package/dist/console-server/routes/logs.d.ts.map +1 -0
  26. package/dist/console-server/routes/logs.js +215 -0
  27. package/dist/console-server/routes/logs.js.map +1 -0
  28. package/dist/console-server/routes/pipeline.d.ts +8 -0
  29. package/dist/console-server/routes/pipeline.d.ts.map +1 -0
  30. package/dist/console-server/routes/pipeline.js +96 -0
  31. package/dist/console-server/routes/pipeline.js.map +1 -0
  32. package/dist/console-server/routes/skills.d.ts +7 -0
  33. package/dist/console-server/routes/skills.d.ts.map +1 -0
  34. package/dist/console-server/routes/skills.js +242 -0
  35. package/dist/console-server/routes/skills.js.map +1 -0
  36. package/dist/console-server/routes/system.d.ts +1 -1
  37. package/dist/console-server/routes/system.d.ts.map +1 -1
  38. package/dist/console-server/routes/system.js +78 -1
  39. package/dist/console-server/routes/system.js.map +1 -1
  40. package/dist/console-server/routes/workers.d.ts +7 -0
  41. package/dist/console-server/routes/workers.d.ts.map +1 -0
  42. package/dist/console-server/routes/workers.js +133 -0
  43. package/dist/console-server/routes/workers.js.map +1 -0
  44. package/dist/console-server/sse/projectStream.d.ts +10 -0
  45. package/dist/console-server/sse/projectStream.d.ts.map +1 -0
  46. package/dist/console-server/sse/projectStream.js +86 -0
  47. package/dist/console-server/sse/projectStream.js.map +1 -0
  48. package/dist/console-server/watchers/cardWatcher.d.ts +1 -3
  49. package/dist/console-server/watchers/cardWatcher.d.ts.map +1 -1
  50. package/dist/console-server/watchers/cardWatcher.js +28 -22
  51. package/dist/console-server/watchers/cardWatcher.js.map +1 -1
  52. package/dist/console-server/watchers/markerWatcher.d.ts +7 -0
  53. package/dist/console-server/watchers/markerWatcher.d.ts.map +1 -0
  54. package/dist/console-server/watchers/markerWatcher.js +53 -0
  55. package/dist/console-server/watchers/markerWatcher.js.map +1 -0
  56. package/dist/console-server/watchers/pipelinePoller.d.ts +2 -0
  57. package/dist/console-server/watchers/pipelinePoller.d.ts.map +1 -0
  58. package/dist/console-server/watchers/pipelinePoller.js +52 -0
  59. package/dist/console-server/watchers/pipelinePoller.js.map +1 -0
  60. package/package.json +1 -1
  61. package/dist/console-assets/assets/index-Bhd2f9AP.js +0 -125
  62. package/dist/console-assets/assets/index-bsAN2a12.css +0 -1
@@ -0,0 +1,335 @@
1
+ /**
2
+ * @module console-server/routes/chat
3
+ * @description Chat sessions with real-time streaming
4
+ *
5
+ * 流式策略:
6
+ * - POST /sessions/:id/messages:立刻持久化 user msg,返回 user msg + pending assistantId
7
+ * - 异步 spawn sps agent,stdout chunk 逐份推 chat.message.chunk(带 assistantId)
8
+ * - 完成时推 chat.message.complete
9
+ * - 失败推 chat.message.complete(role=error)
10
+ *
11
+ * 前端按 assistantId 累积 chunk,组装完整消息。
12
+ */
13
+ import { Hono } from 'hono';
14
+ import { randomUUID } from 'node:crypto';
15
+ import { spawn } from 'node:child_process';
16
+ import { existsSync, mkdirSync, readdirSync, readFileSync, rmSync, writeFileSync, } from 'node:fs';
17
+ import { resolve } from 'node:path';
18
+ import { eventBus } from '../sse/eventBus.js';
19
+ const HOME = process.env.HOME || '/home/coral';
20
+ const SESSIONS_DIR = resolve(HOME, '.coral', 'chat-sessions');
21
+ function ensureDir() {
22
+ if (!existsSync(SESSIONS_DIR))
23
+ mkdirSync(SESSIONS_DIR, { recursive: true });
24
+ }
25
+ function sessionPath(id) {
26
+ return resolve(SESSIONS_DIR, `${id}.json`);
27
+ }
28
+ function readSession(id) {
29
+ const p = sessionPath(id);
30
+ if (!existsSync(p))
31
+ return null;
32
+ try {
33
+ return JSON.parse(readFileSync(p, 'utf-8'));
34
+ }
35
+ catch {
36
+ return null;
37
+ }
38
+ }
39
+ function writeSession(s) {
40
+ ensureDir();
41
+ writeFileSync(sessionPath(s.id), JSON.stringify(s, null, 2));
42
+ }
43
+ function listSessions() {
44
+ if (!existsSync(SESSIONS_DIR))
45
+ return [];
46
+ return readdirSync(SESSIONS_DIR)
47
+ .filter((f) => f.endsWith('.json'))
48
+ .map((f) => {
49
+ try {
50
+ return JSON.parse(readFileSync(resolve(SESSIONS_DIR, f), 'utf-8'));
51
+ }
52
+ catch {
53
+ return null;
54
+ }
55
+ })
56
+ .filter((s) => s !== null)
57
+ .sort((a, b) => {
58
+ const ta = a.lastMessageAt ?? a.createdAt;
59
+ const tb = b.lastMessageAt ?? b.createdAt;
60
+ return tb.localeCompare(ta);
61
+ });
62
+ }
63
+ function summarize(s) {
64
+ const { messages, ...rest } = s;
65
+ void messages;
66
+ return rest;
67
+ }
68
+ function formatTitle(firstMessage) {
69
+ const clean = firstMessage.replace(/\s+/g, ' ').trim();
70
+ return clean.length > 60 ? `${clean.slice(0, 57)}...` : clean;
71
+ }
72
+ /**
73
+ * 清理终端 ANSI escape + 回车 + spinner 字符,保留纯文本。
74
+ */
75
+ const SPINNER_CHARS = new Set(['⠋', '⠙', '⠹', '⠸', '⠼', '⠴', '⠦', '⠧', '⠇', '⠏']);
76
+ function cleanAnsi(raw) {
77
+ let s = raw.replace(/\u001b\[[0-9;?]*[a-zA-Z]/g, '');
78
+ s = s.replace(/\r(?!\n)/g, '');
79
+ s = Array.from(s).filter((ch) => !SPINNER_CHARS.has(ch)).join('');
80
+ return s;
81
+ }
82
+ /**
83
+ * Logger 输出的系统行([agent] / [setup] 等),不给用户看。
84
+ */
85
+ function isSystemLine(line) {
86
+ const cleaned = line.trim();
87
+ if (!cleaned)
88
+ return true;
89
+ if (/^\d{4}-\d{2}-\d{2}[T ][\d:.]+Z?\s+\[(agent|setup|ACP|console|skill|worker-\d+)\]/i.test(cleaned))
90
+ return true;
91
+ if (cleaned.startsWith('(node:') || cleaned.startsWith('Warning:'))
92
+ return true;
93
+ return false;
94
+ }
95
+ function cliEntry() {
96
+ return { node: process.argv[0] ?? 'node', entry: process.argv[1] ?? 'sps' };
97
+ }
98
+ function streamAssistantResponse(sessionId, assistantId, userContent, log) {
99
+ const { node, entry } = cliEntry();
100
+ log.info(`chat[${sessionId}]: spawning sps agent for ${assistantId}`);
101
+ const child = spawn(node, [entry, 'agent', userContent], {
102
+ stdio: ['ignore', 'pipe', 'pipe'],
103
+ env: {
104
+ ...process.env,
105
+ NODE_NO_WARNINGS: '1',
106
+ FORCE_COLOR: '0',
107
+ NO_COLOR: '1',
108
+ },
109
+ });
110
+ let accumulated = '';
111
+ let stderr = '';
112
+ let lineBuffer = '';
113
+ const emitChunk = (chunk) => {
114
+ if (!chunk)
115
+ return;
116
+ accumulated += chunk;
117
+ eventBus.publish('chat.message.chunk', {
118
+ sessionId,
119
+ assistantId,
120
+ chunk,
121
+ accumulated,
122
+ });
123
+ };
124
+ child.stdout?.on('data', (d) => {
125
+ const raw = d.toString('utf-8');
126
+ const cleaned = cleanAnsi(raw);
127
+ lineBuffer += cleaned;
128
+ const lines = lineBuffer.split('\n');
129
+ lineBuffer = lines.pop() ?? '';
130
+ const keep = lines.filter((line) => !isSystemLine(line));
131
+ if (keep.length > 0)
132
+ emitChunk(keep.join('\n') + '\n');
133
+ });
134
+ child.stderr?.on('data', (d) => {
135
+ stderr += d.toString('utf-8');
136
+ });
137
+ child.on('close', (code) => {
138
+ if (lineBuffer && !isSystemLine(lineBuffer))
139
+ emitChunk(lineBuffer);
140
+ const session = readSession(sessionId);
141
+ if (!session)
142
+ return;
143
+ const finalContent = accumulated.trim() || (code === 0 ? '(no output)' : '');
144
+ const ts = new Date().toISOString();
145
+ const msg = code === 0
146
+ ? { id: assistantId, role: 'assistant', content: finalContent, ts, status: 'complete' }
147
+ : {
148
+ id: assistantId,
149
+ role: 'error',
150
+ content: stderr.trim() || `sps agent exit ${code}`,
151
+ ts,
152
+ status: 'error',
153
+ };
154
+ session.messages.push(msg);
155
+ session.lastMessageAt = ts;
156
+ session.messageCount = session.messages.length;
157
+ writeSession(session);
158
+ eventBus.publish('chat.message.complete', {
159
+ sessionId,
160
+ assistantId,
161
+ message: msg,
162
+ });
163
+ if (code === 0)
164
+ log.ok(`chat[${sessionId}]: assistant complete (${accumulated.length} chars)`);
165
+ else
166
+ log.warn(`chat[${sessionId}]: assistant error (exit ${code})`);
167
+ });
168
+ child.on('error', (err) => {
169
+ const session = readSession(sessionId);
170
+ if (!session)
171
+ return;
172
+ const errMsg = {
173
+ id: assistantId,
174
+ role: 'error',
175
+ content: err.message,
176
+ ts: new Date().toISOString(),
177
+ status: 'error',
178
+ };
179
+ session.messages.push(errMsg);
180
+ session.messageCount = session.messages.length;
181
+ writeSession(session);
182
+ eventBus.publish('chat.message.complete', {
183
+ sessionId,
184
+ assistantId,
185
+ message: errMsg,
186
+ });
187
+ log.warn(`chat[${sessionId}]: spawn error ${err.message}`);
188
+ });
189
+ }
190
+ export function createChatRoute(log) {
191
+ const app = new Hono();
192
+ app.get('/sessions', (c) => {
193
+ return c.json({ data: listSessions().map(summarize) });
194
+ });
195
+ app.post('/sessions', async (c) => {
196
+ const body = await c.req.json().catch(() => ({}));
197
+ const session = {
198
+ id: randomUUID(),
199
+ createdAt: new Date().toISOString(),
200
+ lastMessageAt: null,
201
+ title: typeof body?.title === 'string' && body.title ? body.title : '新对话',
202
+ project: typeof body?.project === 'string' ? body.project : null,
203
+ messageCount: 0,
204
+ messages: [],
205
+ };
206
+ writeSession(session);
207
+ eventBus.publish('chat.session.created', { sessionId: session.id });
208
+ return c.json(summarize(session), 201);
209
+ });
210
+ app.get('/sessions/:id', (c) => {
211
+ const id = c.req.param('id');
212
+ const s = readSession(id);
213
+ if (!s)
214
+ return c.json({ type: 'not-found', title: 'Session not found', status: 404 }, 404);
215
+ return c.json(s);
216
+ });
217
+ app.delete('/sessions/:id', (c) => {
218
+ const id = c.req.param('id');
219
+ const p = sessionPath(id);
220
+ if (existsSync(p))
221
+ rmSync(p);
222
+ eventBus.publish('chat.session.deleted', { sessionId: id });
223
+ return c.body(null, 204);
224
+ });
225
+ /**
226
+ * POST messages: 非阻塞
227
+ * 1. 持久化 user msg → 立即返回 user + assistantId (202)
228
+ * 2. 后台 spawn sps agent,SSE 推 chunk + complete
229
+ */
230
+ app.post('/sessions/:id/messages', async (c) => {
231
+ const id = c.req.param('id');
232
+ const s = readSession(id);
233
+ if (!s)
234
+ return c.json({ type: 'not-found', title: 'Session not found', status: 404 }, 404);
235
+ const body = (await c.req.json().catch(() => null));
236
+ if (!body?.content || typeof body.content !== 'string') {
237
+ return c.json({ type: 'validation', title: 'content required', status: 422 }, 422);
238
+ }
239
+ const userMsg = {
240
+ id: randomUUID(),
241
+ role: 'user',
242
+ content: body.content,
243
+ ts: new Date().toISOString(),
244
+ status: 'complete',
245
+ };
246
+ s.messages.push(userMsg);
247
+ s.lastMessageAt = userMsg.ts;
248
+ s.messageCount = s.messages.length;
249
+ if (s.messages.length === 1)
250
+ s.title = formatTitle(body.content);
251
+ writeSession(s);
252
+ eventBus.publish('chat.message', { sessionId: id, message: userMsg });
253
+ const assistantId = randomUUID();
254
+ eventBus.publish('chat.message.pending', {
255
+ sessionId: id,
256
+ assistantId,
257
+ ts: new Date().toISOString(),
258
+ });
259
+ streamAssistantResponse(id, assistantId, body.content, log);
260
+ return c.json({ user: userMsg, assistantId }, 202);
261
+ });
262
+ return app;
263
+ }
264
+ // ── SSE: /stream/chat/:id ───────────────────────────────────────────────
265
+ export function createChatStreamRoute() {
266
+ const app = new Hono();
267
+ app.get('/:id', (c) => {
268
+ const sessionId = c.req.param('id');
269
+ const stream = new ReadableStream({
270
+ start(controller) {
271
+ const enc = new TextEncoder();
272
+ let closed = false;
273
+ const send = (event, data) => {
274
+ if (closed)
275
+ return;
276
+ try {
277
+ controller.enqueue(enc.encode(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`));
278
+ }
279
+ catch {
280
+ closed = true;
281
+ }
282
+ };
283
+ const eventTypes = [
284
+ 'chat.message',
285
+ 'chat.message.pending',
286
+ 'chat.message.chunk',
287
+ 'chat.message.complete',
288
+ ];
289
+ const handlers = [];
290
+ for (const event of eventTypes) {
291
+ const fn = (data) => {
292
+ if (typeof data !== 'object' ||
293
+ data === null ||
294
+ data.sessionId !== sessionId)
295
+ return;
296
+ send(event, data);
297
+ };
298
+ eventBus.on(event, fn);
299
+ handlers.push({ event, fn });
300
+ }
301
+ const heartbeat = setInterval(() => {
302
+ if (closed)
303
+ return;
304
+ try {
305
+ controller.enqueue(enc.encode(`: heartbeat ${Date.now()}\n\n`));
306
+ }
307
+ catch {
308
+ closed = true;
309
+ }
310
+ }, 15_000);
311
+ c.req.raw.signal?.addEventListener('abort', () => {
312
+ closed = true;
313
+ for (const h of handlers)
314
+ eventBus.off(h.event, h.fn);
315
+ clearInterval(heartbeat);
316
+ try {
317
+ controller.close();
318
+ }
319
+ catch {
320
+ /* ignore */
321
+ }
322
+ });
323
+ },
324
+ });
325
+ return new Response(stream, {
326
+ headers: {
327
+ 'Content-Type': 'text/event-stream',
328
+ 'Cache-Control': 'no-cache, no-transform',
329
+ Connection: 'keep-alive',
330
+ },
331
+ });
332
+ });
333
+ return app;
334
+ }
335
+ //# sourceMappingURL=chat.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"chat.js","sourceRoot":"","sources":["../../../src/console-server/routes/chat.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ /**
2
+ * @module console-server/routes/logs
3
+ * @description 日志:历史查询 + SSE tail
4
+ *
5
+ * 策略:
6
+ * - 项目有个大 log 文件(~/.coral/projects/<name>/logs/*.log)
7
+ * - GET /api/logs?project=x[&worker=N]&since=<offset>&limit=500 → 读尾部
8
+ * - GET /stream/logs?project=x 打开 fs.watch,文件追加 → SSE 广播
9
+ */
10
+ import { Hono } from 'hono';
11
+ import { Logger } from '../../core/logger.js';
12
+ export declare function createLogsRoute(log: Logger): Hono;
13
+ export declare function createLogsStreamRoute(log: Logger): Hono;
14
+ //# sourceMappingURL=logs.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logs.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/logs.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAW5B,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AA8E9C,wBAAgB,eAAe,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CA4BjD;AAGD,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CAsGvD"}
@@ -0,0 +1,215 @@
1
+ /**
2
+ * @module console-server/routes/logs
3
+ * @description 日志:历史查询 + SSE tail
4
+ *
5
+ * 策略:
6
+ * - 项目有个大 log 文件(~/.coral/projects/<name>/logs/*.log)
7
+ * - GET /api/logs?project=x[&worker=N]&since=<offset>&limit=500 → 读尾部
8
+ * - GET /stream/logs?project=x 打开 fs.watch,文件追加 → SSE 广播
9
+ */
10
+ import { Hono } from 'hono';
11
+ import { createReadStream, existsSync, readdirSync, statSync, watch as fsWatch, } from 'node:fs';
12
+ import { resolve } from 'node:path';
13
+ import { createInterface } from 'node:readline';
14
+ const HOME = process.env.HOME || '/home/coral';
15
+ const LOG_LINE_BUDGET = 500;
16
+ const SSE_HEARTBEAT_MS = 15_000;
17
+ const MAX_SCAN_BYTES = 4 * 1024 * 1024; // 只读最近 4MB
18
+ function projectLogsDir(project) {
19
+ return resolve(HOME, '.coral', 'projects', project, 'logs');
20
+ }
21
+ function findLogFiles(project, worker) {
22
+ const dir = projectLogsDir(project);
23
+ if (!existsSync(dir))
24
+ return [];
25
+ const files = readdirSync(dir)
26
+ .filter((f) => f.endsWith('.log'))
27
+ .filter((f) => (worker ? f.includes(`worker-${worker}`) || f.includes(`-${worker}-`) : true))
28
+ .map((f) => resolve(dir, f))
29
+ .sort((a, b) => {
30
+ try {
31
+ return statSync(b).mtimeMs - statSync(a).mtimeMs;
32
+ }
33
+ catch {
34
+ return 0;
35
+ }
36
+ });
37
+ return files;
38
+ }
39
+ function parseLine(raw) {
40
+ // 兼容几种:
41
+ // [2026-04-22 18:55:03.124] INFO worker-1 msg
42
+ // 2026-04-22T18:55:03.124Z INFO [worker-1] msg
43
+ // 普通 tsx 输出:任意字符串
44
+ const cleaned = raw.replace(/\u001b\[[0-9;]*m/g, ''); // 去 ANSI 颜色
45
+ const m = cleaned.match(/(\d{4}-\d{2}-\d{2}[T ][\d:.]+Z?)\s*(?:\[)?(DEBUG|INFO|WARN|WARNING|ERROR|TRACE)\]?\s*(?:\[?(worker-\d+|acp|claude|supervisor|event-handler|skill|console)\]?\s*)?(.*)$/i);
46
+ if (m) {
47
+ const tsRaw = m[1] ?? '';
48
+ const lvl = (m[2] ?? 'info').toLowerCase().replace('warning', 'warn');
49
+ const src = m[3] ?? '';
50
+ const msg = m[4] ?? cleaned;
51
+ let worker = null;
52
+ const wm = src.match(/worker-(\d+)/);
53
+ if (wm)
54
+ worker = Number.parseInt(wm[1] ?? '', 10);
55
+ return {
56
+ ts: tsRaw.includes('T') ? tsRaw : tsRaw.replace(' ', 'T') + 'Z',
57
+ worker,
58
+ level: lvl,
59
+ msg: src && !src.startsWith('worker-') ? `[${src}] ${msg}` : msg,
60
+ raw: cleaned,
61
+ };
62
+ }
63
+ return { ts: null, worker: null, level: 'info', msg: cleaned, raw: cleaned };
64
+ }
65
+ async function readTailLines(filePath, limit) {
66
+ const stat = statSync(filePath);
67
+ const start = Math.max(0, stat.size - MAX_SCAN_BYTES);
68
+ const lines = [];
69
+ await new Promise((done) => {
70
+ const stream = createReadStream(filePath, { start, encoding: 'utf-8' });
71
+ const rl = createInterface({ input: stream });
72
+ rl.on('line', (l) => {
73
+ lines.push(parseLine(l));
74
+ if (lines.length > limit * 3)
75
+ lines.splice(0, lines.length - limit * 3);
76
+ });
77
+ rl.on('close', () => done());
78
+ });
79
+ return lines.slice(-limit);
80
+ }
81
+ export function createLogsRoute(log) {
82
+ const app = new Hono();
83
+ app.get('/', async (c) => {
84
+ const project = c.req.query('project');
85
+ if (!project) {
86
+ return c.json({ type: 'validation', title: 'project required', status: 422 }, 422);
87
+ }
88
+ const worker = c.req.query('worker') || undefined;
89
+ const limit = Math.min(Number.parseInt(c.req.query('limit') ?? '500', 10) || 500, 2000);
90
+ const files = findLogFiles(project, worker);
91
+ if (files.length === 0)
92
+ return c.json({ data: [], files: [] });
93
+ const file = files[0];
94
+ try {
95
+ const lines = await readTailLines(file, limit);
96
+ return c.json({
97
+ data: lines,
98
+ file: file.replace(HOME, '~'),
99
+ files: files.map((f) => f.replace(HOME, '~')),
100
+ });
101
+ }
102
+ catch (err) {
103
+ log.warn(`log read failed: ${err instanceof Error ? err.message : String(err)}`);
104
+ return c.json({ data: [], file, files });
105
+ }
106
+ });
107
+ return app;
108
+ }
109
+ // ── SSE: /stream/logs?project=x[&worker=N] ────────────────────────────
110
+ export function createLogsStreamRoute(log) {
111
+ const app = new Hono();
112
+ app.get('/', (c) => {
113
+ const project = c.req.query('project');
114
+ if (!project) {
115
+ return c.text('project required', 422);
116
+ }
117
+ const worker = c.req.query('worker') || undefined;
118
+ const files = findLogFiles(project, worker);
119
+ const file = files[0];
120
+ const stream = new ReadableStream({
121
+ start(controller) {
122
+ const enc = new TextEncoder();
123
+ let closed = false;
124
+ const send = (event, data) => {
125
+ if (closed)
126
+ return;
127
+ try {
128
+ controller.enqueue(enc.encode(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`));
129
+ }
130
+ catch {
131
+ closed = true;
132
+ }
133
+ };
134
+ let watcher = null;
135
+ let lastSize = 0;
136
+ if (file && existsSync(file)) {
137
+ try {
138
+ lastSize = statSync(file).size;
139
+ }
140
+ catch {
141
+ lastSize = 0;
142
+ }
143
+ try {
144
+ watcher = fsWatch(file, async () => {
145
+ if (closed)
146
+ return;
147
+ try {
148
+ const stat = statSync(file);
149
+ if (stat.size <= lastSize) {
150
+ // 文件被截断或轮转,重新定位
151
+ lastSize = stat.size;
152
+ return;
153
+ }
154
+ const newChunk = [];
155
+ await new Promise((done) => {
156
+ const s = createReadStream(file, {
157
+ start: lastSize,
158
+ end: stat.size,
159
+ encoding: 'utf-8',
160
+ });
161
+ const rl = createInterface({ input: s });
162
+ rl.on('line', (l) => newChunk.push(l));
163
+ rl.on('close', () => done());
164
+ });
165
+ lastSize = stat.size;
166
+ for (const l of newChunk) {
167
+ if (!l.trim())
168
+ continue;
169
+ send('log.line', parseLine(l));
170
+ }
171
+ }
172
+ catch (err) {
173
+ log.warn(`log watch failed: ${err instanceof Error ? err.message : String(err)}`);
174
+ }
175
+ });
176
+ }
177
+ catch (err) {
178
+ log.warn(`fs.watch failed: ${err instanceof Error ? err.message : String(err)}`);
179
+ }
180
+ }
181
+ const heartbeat = setInterval(() => {
182
+ if (closed)
183
+ return;
184
+ try {
185
+ controller.enqueue(enc.encode(`: heartbeat ${Date.now()}\n\n`));
186
+ }
187
+ catch {
188
+ closed = true;
189
+ }
190
+ }, SSE_HEARTBEAT_MS);
191
+ send('log.init', { file: file?.replace(HOME, '~') ?? null });
192
+ c.req.raw.signal?.addEventListener('abort', () => {
193
+ closed = true;
194
+ watcher?.close();
195
+ clearInterval(heartbeat);
196
+ try {
197
+ controller.close();
198
+ }
199
+ catch {
200
+ /* ignore */
201
+ }
202
+ });
203
+ },
204
+ });
205
+ return new Response(stream, {
206
+ headers: {
207
+ 'Content-Type': 'text/event-stream',
208
+ 'Cache-Control': 'no-cache, no-transform',
209
+ Connection: 'keep-alive',
210
+ },
211
+ });
212
+ });
213
+ return app;
214
+ }
215
+ //# sourceMappingURL=logs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logs.js","sourceRoot":"","sources":["../../../src/console-server/routes/logs.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,8 @@
1
+ /**
2
+ * @module console-server/routes/pipeline
3
+ * @description 流水线控制:start / stop / reset / status
4
+ */
5
+ import { Hono } from 'hono';
6
+ import { Logger } from '../../core/logger.js';
7
+ export declare function createPipelineRoute(log: Logger): Hono;
8
+ //# sourceMappingURL=pipeline.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pipeline.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/pipeline.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAG5B,OAAO,EAAE,MAAM,EAAE,MAAM,sBAAsB,CAAC;AA0B9C,wBAAgB,mBAAmB,CAAC,GAAG,EAAE,MAAM,GAAG,IAAI,CA4ErD"}