@coralai/sps-cli 0.44.0 → 0.45.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.
- package/README.md +1 -1
- package/dist/console-assets/assets/index-COKveGOw.js +231 -0
- package/dist/console-assets/assets/index-DTOYQgOY.css +1 -0
- package/dist/console-assets/index.html +2 -2
- package/dist/console-server/index.d.ts.map +1 -1
- package/dist/console-server/index.js +25 -5
- package/dist/console-server/index.js.map +1 -1
- package/dist/console-server/lib/cardReader.d.ts +27 -0
- package/dist/console-server/lib/cardReader.d.ts.map +1 -0
- package/dist/console-server/lib/cardReader.js +131 -0
- package/dist/console-server/lib/cardReader.js.map +1 -0
- package/dist/console-server/lib/spawnCli.d.ts +31 -0
- package/dist/console-server/lib/spawnCli.d.ts.map +1 -0
- package/dist/console-server/lib/spawnCli.js +64 -0
- package/dist/console-server/lib/spawnCli.js.map +1 -0
- package/dist/console-server/routes/cards.d.ts +7 -0
- package/dist/console-server/routes/cards.d.ts.map +1 -0
- package/dist/console-server/routes/cards.js +81 -0
- package/dist/console-server/routes/cards.js.map +1 -0
- package/dist/console-server/routes/chat.d.ts +21 -0
- package/dist/console-server/routes/chat.d.ts.map +1 -0
- package/dist/console-server/routes/chat.js +233 -0
- package/dist/console-server/routes/chat.js.map +1 -0
- package/dist/console-server/routes/logs.d.ts +14 -0
- package/dist/console-server/routes/logs.d.ts.map +1 -0
- package/dist/console-server/routes/logs.js +215 -0
- package/dist/console-server/routes/logs.js.map +1 -0
- package/dist/console-server/routes/pipeline.d.ts +8 -0
- package/dist/console-server/routes/pipeline.d.ts.map +1 -0
- package/dist/console-server/routes/pipeline.js +96 -0
- package/dist/console-server/routes/pipeline.js.map +1 -0
- package/dist/console-server/routes/skills.d.ts +7 -0
- package/dist/console-server/routes/skills.d.ts.map +1 -0
- package/dist/console-server/routes/skills.js +242 -0
- package/dist/console-server/routes/skills.js.map +1 -0
- package/dist/console-server/routes/system.d.ts +1 -1
- package/dist/console-server/routes/system.d.ts.map +1 -1
- package/dist/console-server/routes/system.js +78 -1
- package/dist/console-server/routes/system.js.map +1 -1
- package/dist/console-server/routes/workers.d.ts +7 -0
- package/dist/console-server/routes/workers.d.ts.map +1 -0
- package/dist/console-server/routes/workers.js +133 -0
- package/dist/console-server/routes/workers.js.map +1 -0
- package/dist/console-server/sse/projectStream.d.ts +10 -0
- package/dist/console-server/sse/projectStream.d.ts.map +1 -0
- package/dist/console-server/sse/projectStream.js +86 -0
- package/dist/console-server/sse/projectStream.js.map +1 -0
- package/dist/console-server/watchers/cardWatcher.d.ts +1 -3
- package/dist/console-server/watchers/cardWatcher.d.ts.map +1 -1
- package/dist/console-server/watchers/cardWatcher.js +28 -22
- package/dist/console-server/watchers/cardWatcher.js.map +1 -1
- package/dist/console-server/watchers/markerWatcher.d.ts +7 -0
- package/dist/console-server/watchers/markerWatcher.d.ts.map +1 -0
- package/dist/console-server/watchers/markerWatcher.js +53 -0
- package/dist/console-server/watchers/markerWatcher.js.map +1 -0
- package/dist/console-server/watchers/pipelinePoller.d.ts +2 -0
- package/dist/console-server/watchers/pipelinePoller.d.ts.map +1 -0
- package/dist/console-server/watchers/pipelinePoller.js +52 -0
- package/dist/console-server/watchers/pipelinePoller.js.map +1 -0
- package/package.json +1 -1
- package/dist/console-assets/assets/index-Bhd2f9AP.js +0 -125
- package/dist/console-assets/assets/index-bsAN2a12.css +0 -1
|
@@ -0,0 +1,233 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module console-server/routes/chat
|
|
3
|
+
* @description Chat sessions:spawn `sps agent` 做一次性 prompt,结果缓存到内存 session
|
|
4
|
+
*
|
|
5
|
+
* 简化版(M5 v1):
|
|
6
|
+
* - 无流式 SSE(ACP daemon 集成留 v0.45)
|
|
7
|
+
* - 每个 session 是一个 list of messages(user + assistant),持久化在 ~/.coral/chat-sessions/
|
|
8
|
+
* - POST messages → spawnCliSync("agent", prompt) → assistant reply → append
|
|
9
|
+
* - SSE endpoint 只做"消息变更"通知
|
|
10
|
+
*/
|
|
11
|
+
import { Hono } from 'hono';
|
|
12
|
+
import { randomUUID } from 'node:crypto';
|
|
13
|
+
import { existsSync, mkdirSync, readdirSync, readFileSync, rmSync, writeFileSync, } from 'node:fs';
|
|
14
|
+
import { resolve } from 'node:path';
|
|
15
|
+
import { spawnCliSync } from '../lib/spawnCli.js';
|
|
16
|
+
import { eventBus } from '../sse/eventBus.js';
|
|
17
|
+
const HOME = process.env.HOME || '/home/coral';
|
|
18
|
+
const SESSIONS_DIR = resolve(HOME, '.coral', 'chat-sessions');
|
|
19
|
+
function ensureDir() {
|
|
20
|
+
if (!existsSync(SESSIONS_DIR))
|
|
21
|
+
mkdirSync(SESSIONS_DIR, { recursive: true });
|
|
22
|
+
}
|
|
23
|
+
function sessionPath(id) {
|
|
24
|
+
return resolve(SESSIONS_DIR, `${id}.json`);
|
|
25
|
+
}
|
|
26
|
+
function readSession(id) {
|
|
27
|
+
const p = sessionPath(id);
|
|
28
|
+
if (!existsSync(p))
|
|
29
|
+
return null;
|
|
30
|
+
try {
|
|
31
|
+
return JSON.parse(readFileSync(p, 'utf-8'));
|
|
32
|
+
}
|
|
33
|
+
catch {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
function writeSession(s) {
|
|
38
|
+
ensureDir();
|
|
39
|
+
writeFileSync(sessionPath(s.id), JSON.stringify(s, null, 2));
|
|
40
|
+
}
|
|
41
|
+
function listSessions() {
|
|
42
|
+
if (!existsSync(SESSIONS_DIR))
|
|
43
|
+
return [];
|
|
44
|
+
return readdirSync(SESSIONS_DIR)
|
|
45
|
+
.filter((f) => f.endsWith('.json'))
|
|
46
|
+
.map((f) => {
|
|
47
|
+
try {
|
|
48
|
+
return JSON.parse(readFileSync(resolve(SESSIONS_DIR, f), 'utf-8'));
|
|
49
|
+
}
|
|
50
|
+
catch {
|
|
51
|
+
return null;
|
|
52
|
+
}
|
|
53
|
+
})
|
|
54
|
+
.filter((s) => s !== null)
|
|
55
|
+
.sort((a, b) => {
|
|
56
|
+
const ta = a.lastMessageAt ?? a.createdAt;
|
|
57
|
+
const tb = b.lastMessageAt ?? b.createdAt;
|
|
58
|
+
return tb.localeCompare(ta);
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
function summarize(s) {
|
|
62
|
+
const { messages, ...rest } = s;
|
|
63
|
+
void messages;
|
|
64
|
+
return rest;
|
|
65
|
+
}
|
|
66
|
+
function formatTitle(firstMessage) {
|
|
67
|
+
const clean = firstMessage.replace(/\s+/g, ' ').trim();
|
|
68
|
+
return clean.length > 60 ? `${clean.slice(0, 57)}...` : clean;
|
|
69
|
+
}
|
|
70
|
+
export function createChatRoute(log) {
|
|
71
|
+
const app = new Hono();
|
|
72
|
+
app.get('/sessions', (c) => {
|
|
73
|
+
const sessions = listSessions().map(summarize);
|
|
74
|
+
return c.json({ data: sessions });
|
|
75
|
+
});
|
|
76
|
+
app.post('/sessions', async (c) => {
|
|
77
|
+
const body = await c.req.json().catch(() => ({}));
|
|
78
|
+
const session = {
|
|
79
|
+
id: randomUUID(),
|
|
80
|
+
createdAt: new Date().toISOString(),
|
|
81
|
+
lastMessageAt: null,
|
|
82
|
+
title: typeof body?.title === 'string' && body.title ? body.title : '新对话',
|
|
83
|
+
project: typeof body?.project === 'string' ? body.project : null,
|
|
84
|
+
messageCount: 0,
|
|
85
|
+
messages: [],
|
|
86
|
+
};
|
|
87
|
+
writeSession(session);
|
|
88
|
+
eventBus.publish('chat.session.created', { sessionId: session.id });
|
|
89
|
+
return c.json(summarize(session), 201);
|
|
90
|
+
});
|
|
91
|
+
app.get('/sessions/:id', (c) => {
|
|
92
|
+
const id = c.req.param('id');
|
|
93
|
+
const s = readSession(id);
|
|
94
|
+
if (!s) {
|
|
95
|
+
return c.json({ type: 'not-found', title: 'Session not found', status: 404 }, 404);
|
|
96
|
+
}
|
|
97
|
+
return c.json(s);
|
|
98
|
+
});
|
|
99
|
+
app.delete('/sessions/:id', (c) => {
|
|
100
|
+
const id = c.req.param('id');
|
|
101
|
+
const p = sessionPath(id);
|
|
102
|
+
if (existsSync(p))
|
|
103
|
+
rmSync(p);
|
|
104
|
+
eventBus.publish('chat.session.deleted', { sessionId: id });
|
|
105
|
+
return c.body(null, 204);
|
|
106
|
+
});
|
|
107
|
+
app.post('/sessions/:id/messages', async (c) => {
|
|
108
|
+
const id = c.req.param('id');
|
|
109
|
+
const s = readSession(id);
|
|
110
|
+
if (!s) {
|
|
111
|
+
return c.json({ type: 'not-found', title: 'Session not found', status: 404 }, 404);
|
|
112
|
+
}
|
|
113
|
+
const body = await c.req.json().catch(() => null);
|
|
114
|
+
if (!body?.content || typeof body.content !== 'string') {
|
|
115
|
+
return c.json({ type: 'validation', title: 'content required', status: 422 }, 422);
|
|
116
|
+
}
|
|
117
|
+
const userMsg = {
|
|
118
|
+
id: randomUUID(),
|
|
119
|
+
role: 'user',
|
|
120
|
+
content: body.content,
|
|
121
|
+
ts: new Date().toISOString(),
|
|
122
|
+
};
|
|
123
|
+
s.messages.push(userMsg);
|
|
124
|
+
s.lastMessageAt = userMsg.ts;
|
|
125
|
+
s.messageCount = s.messages.length;
|
|
126
|
+
if (s.messages.length === 1)
|
|
127
|
+
s.title = formatTitle(body.content);
|
|
128
|
+
writeSession(s);
|
|
129
|
+
eventBus.publish('chat.message', { sessionId: id, message: userMsg });
|
|
130
|
+
// 派 sps agent 跑一次 prompt(非流式)
|
|
131
|
+
log.info(`chat[${id}]: spawning sps agent`);
|
|
132
|
+
const result = await spawnCliSync(['agent', body.content], { timeoutMs: 5 * 60 * 1000 });
|
|
133
|
+
let assistant;
|
|
134
|
+
if (result.exitCode === 0) {
|
|
135
|
+
// 尝试去掉 log banner(匹配 Logger 输出的时间戳前缀)
|
|
136
|
+
const cleaned = result.stdout
|
|
137
|
+
.split('\n')
|
|
138
|
+
.filter((line) => !/^\u001b?\[?\d{4}-\d{2}-\d{2}.*\[(agent|setup|ACP)\]/.test(line))
|
|
139
|
+
.join('\n')
|
|
140
|
+
.trim();
|
|
141
|
+
assistant = {
|
|
142
|
+
id: randomUUID(),
|
|
143
|
+
role: 'assistant',
|
|
144
|
+
content: cleaned || result.stdout.trim() || '(no output)',
|
|
145
|
+
ts: new Date().toISOString(),
|
|
146
|
+
};
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
assistant = {
|
|
150
|
+
id: randomUUID(),
|
|
151
|
+
role: 'error',
|
|
152
|
+
content: result.stderr.trim() || result.stdout.trim() || `agent exit ${result.exitCode}`,
|
|
153
|
+
ts: new Date().toISOString(),
|
|
154
|
+
};
|
|
155
|
+
}
|
|
156
|
+
s.messages.push(assistant);
|
|
157
|
+
s.lastMessageAt = assistant.ts;
|
|
158
|
+
s.messageCount = s.messages.length;
|
|
159
|
+
writeSession(s);
|
|
160
|
+
eventBus.publish('chat.message', { sessionId: id, message: assistant });
|
|
161
|
+
return c.json({ user: userMsg, assistant });
|
|
162
|
+
});
|
|
163
|
+
app.get('/sessions/:id/meta', (c) => {
|
|
164
|
+
const id = c.req.param('id');
|
|
165
|
+
const s = readSession(id);
|
|
166
|
+
if (!s) {
|
|
167
|
+
return c.json({ type: 'not-found', title: 'Session not found', status: 404 }, 404);
|
|
168
|
+
}
|
|
169
|
+
return c.json(summarize(s));
|
|
170
|
+
});
|
|
171
|
+
return app;
|
|
172
|
+
}
|
|
173
|
+
// ── SSE: /stream/chat/:id ───────────────────────────────────────────────
|
|
174
|
+
export function createChatStreamRoute() {
|
|
175
|
+
const app = new Hono();
|
|
176
|
+
app.get('/:id', (c) => {
|
|
177
|
+
const sessionId = c.req.param('id');
|
|
178
|
+
const stream = new ReadableStream({
|
|
179
|
+
start(controller) {
|
|
180
|
+
const enc = new TextEncoder();
|
|
181
|
+
let closed = false;
|
|
182
|
+
const send = (event, data) => {
|
|
183
|
+
if (closed)
|
|
184
|
+
return;
|
|
185
|
+
try {
|
|
186
|
+
controller.enqueue(enc.encode(`event: ${event}\ndata: ${JSON.stringify(data)}\n\n`));
|
|
187
|
+
}
|
|
188
|
+
catch {
|
|
189
|
+
closed = true;
|
|
190
|
+
}
|
|
191
|
+
};
|
|
192
|
+
const onChatMessage = (data) => {
|
|
193
|
+
if (typeof data !== 'object' ||
|
|
194
|
+
data === null ||
|
|
195
|
+
data.sessionId !== sessionId)
|
|
196
|
+
return;
|
|
197
|
+
send('chat.message', data);
|
|
198
|
+
};
|
|
199
|
+
eventBus.on('chat.message', onChatMessage);
|
|
200
|
+
const heartbeat = setInterval(() => {
|
|
201
|
+
if (closed)
|
|
202
|
+
return;
|
|
203
|
+
try {
|
|
204
|
+
controller.enqueue(enc.encode(`: heartbeat ${Date.now()}\n\n`));
|
|
205
|
+
}
|
|
206
|
+
catch {
|
|
207
|
+
closed = true;
|
|
208
|
+
}
|
|
209
|
+
}, 15_000);
|
|
210
|
+
c.req.raw.signal?.addEventListener('abort', () => {
|
|
211
|
+
closed = true;
|
|
212
|
+
eventBus.off('chat.message', onChatMessage);
|
|
213
|
+
clearInterval(heartbeat);
|
|
214
|
+
try {
|
|
215
|
+
controller.close();
|
|
216
|
+
}
|
|
217
|
+
catch {
|
|
218
|
+
/* ignore */
|
|
219
|
+
}
|
|
220
|
+
});
|
|
221
|
+
},
|
|
222
|
+
});
|
|
223
|
+
return new Response(stream, {
|
|
224
|
+
headers: {
|
|
225
|
+
'Content-Type': 'text/event-stream',
|
|
226
|
+
'Cache-Control': 'no-cache, no-transform',
|
|
227
|
+
Connection: 'keep-alive',
|
|
228
|
+
},
|
|
229
|
+
});
|
|
230
|
+
});
|
|
231
|
+
return app;
|
|
232
|
+
}
|
|
233
|
+
//# 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"}
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module console-server/routes/pipeline
|
|
3
|
+
* @description 流水线控制:start / stop / reset / status
|
|
4
|
+
*/
|
|
5
|
+
import { Hono } from 'hono';
|
|
6
|
+
import { existsSync, mkdirSync, readFileSync } from 'node:fs';
|
|
7
|
+
import { resolve } from 'node:path';
|
|
8
|
+
import { spawnCliDetached, spawnCliSync } from '../lib/spawnCli.js';
|
|
9
|
+
const HOME = process.env.HOME || '/home/coral';
|
|
10
|
+
function projectDir(name) {
|
|
11
|
+
return resolve(HOME, '.coral', 'projects', name);
|
|
12
|
+
}
|
|
13
|
+
function supervisorPid(name) {
|
|
14
|
+
const pidFile = resolve(projectDir(name), 'runtime', 'supervisor.pid');
|
|
15
|
+
if (!existsSync(pidFile))
|
|
16
|
+
return null;
|
|
17
|
+
try {
|
|
18
|
+
const pid = Number.parseInt(readFileSync(pidFile, 'utf-8').trim(), 10);
|
|
19
|
+
if (!Number.isFinite(pid) || pid <= 0)
|
|
20
|
+
return null;
|
|
21
|
+
try {
|
|
22
|
+
process.kill(pid, 0);
|
|
23
|
+
return pid;
|
|
24
|
+
}
|
|
25
|
+
catch {
|
|
26
|
+
return null;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
catch {
|
|
30
|
+
return null;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
export function createPipelineRoute(log) {
|
|
34
|
+
const app = new Hono();
|
|
35
|
+
app.post('/:project/pipeline/start', (c) => {
|
|
36
|
+
const project = c.req.param('project');
|
|
37
|
+
if (!existsSync(projectDir(project))) {
|
|
38
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404, detail: project }, 404);
|
|
39
|
+
}
|
|
40
|
+
const existing = supervisorPid(project);
|
|
41
|
+
if (existing) {
|
|
42
|
+
return c.json({ ok: true, status: 'running', pid: existing });
|
|
43
|
+
}
|
|
44
|
+
// log 目录
|
|
45
|
+
const logsDir = resolve(projectDir(project), 'logs');
|
|
46
|
+
if (!existsSync(logsDir))
|
|
47
|
+
mkdirSync(logsDir, { recursive: true });
|
|
48
|
+
const logPath = resolve(logsDir, `console-tick-${new Date().toISOString().slice(0, 10)}.log`);
|
|
49
|
+
try {
|
|
50
|
+
const child = spawnCliDetached(['tick', project], { logPath });
|
|
51
|
+
log.ok(`Pipeline for "${project}" spawned (pid ${child.pid})`);
|
|
52
|
+
return c.json({ ok: true, status: 'running', pid: child.pid ?? null });
|
|
53
|
+
}
|
|
54
|
+
catch (err) {
|
|
55
|
+
return c.json({
|
|
56
|
+
type: 'spawn-error',
|
|
57
|
+
title: 'Failed to start pipeline',
|
|
58
|
+
status: 500,
|
|
59
|
+
detail: err instanceof Error ? err.message : String(err),
|
|
60
|
+
}, 500);
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
app.post('/:project/pipeline/stop', async (c) => {
|
|
64
|
+
const project = c.req.param('project');
|
|
65
|
+
const result = await spawnCliSync(['stop', project], { timeoutMs: 20_000 });
|
|
66
|
+
if (result.exitCode !== 0) {
|
|
67
|
+
return c.json({ type: 'cli-error', title: 'stop failed', status: 500, detail: result.stderr }, 500);
|
|
68
|
+
}
|
|
69
|
+
return c.json({ ok: true, output: result.stdout.trim() });
|
|
70
|
+
});
|
|
71
|
+
app.post('/:project/pipeline/reset', async (c) => {
|
|
72
|
+
const project = c.req.param('project');
|
|
73
|
+
const body = await c.req.json().catch(() => ({}));
|
|
74
|
+
const args = ['reset', project];
|
|
75
|
+
if (body?.all)
|
|
76
|
+
args.push('--all');
|
|
77
|
+
else if (Array.isArray(body?.cards) && body.cards.length > 0) {
|
|
78
|
+
args.push('--card', body.cards.join(','));
|
|
79
|
+
}
|
|
80
|
+
const result = await spawnCliSync(args, { timeoutMs: 60_000 });
|
|
81
|
+
if (result.exitCode !== 0) {
|
|
82
|
+
return c.json({ type: 'cli-error', title: 'reset failed', status: 500, detail: result.stderr }, 500);
|
|
83
|
+
}
|
|
84
|
+
return c.json({ ok: true });
|
|
85
|
+
});
|
|
86
|
+
app.get('/:project/pipeline/status', (c) => {
|
|
87
|
+
const project = c.req.param('project');
|
|
88
|
+
const pid = supervisorPid(project);
|
|
89
|
+
return c.json({
|
|
90
|
+
status: pid ? 'running' : 'idle',
|
|
91
|
+
pid,
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
return app;
|
|
95
|
+
}
|
|
96
|
+
//# sourceMappingURL=pipeline.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../../../src/console-server/routes/pipeline.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"skills.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/skills.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAuG5B,wBAAgB,iBAAiB,IAAI,IAAI,CAoJxC"}
|