@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.
- package/README.md +1 -1
- package/dist/console-assets/assets/index-BGs71Um2.css +1 -0
- package/dist/console-assets/assets/index-BsHLiTM2.js +236 -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 +24 -0
- package/dist/console-server/routes/chat.d.ts.map +1 -0
- package/dist/console-server/routes/chat.js +335 -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,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"}
|
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module console-server/routes/skills
|
|
3
|
+
* @description Skill REST API - 复用 core/skillStore
|
|
4
|
+
*/
|
|
5
|
+
import { Hono } from 'hono';
|
|
6
|
+
import { existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
|
|
7
|
+
import { resolve } from 'node:path';
|
|
8
|
+
import { addSkillToProject, ensureSkillsGitignore, freezeSkillInProject, inspectProjectSkill, listUserSkills, projectSkillsDir, removeSkillFromProject, syncBundledSkillsToUser, unfreezeSkillInProject, userSkillsRoot, } from '../../core/skillStore.js';
|
|
9
|
+
const HOME = process.env.HOME || '/home/coral';
|
|
10
|
+
const PROJECTS_DIR = resolve(HOME, '.coral', 'projects');
|
|
11
|
+
function findProjectRepoDir(projectName) {
|
|
12
|
+
const confPath = resolve(PROJECTS_DIR, projectName, 'conf');
|
|
13
|
+
if (!existsSync(confPath))
|
|
14
|
+
return null;
|
|
15
|
+
try {
|
|
16
|
+
const content = readFileSync(confPath, 'utf-8');
|
|
17
|
+
const match = content.match(/^(?:export\s+)?REPO_DIR=["']?([^"'\n]+)["']?/m);
|
|
18
|
+
return match ? (match[1] ?? null) : null;
|
|
19
|
+
}
|
|
20
|
+
catch {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
function classifyCategory(name) {
|
|
25
|
+
const LANGUAGES = new Set(['python', 'typescript', 'golang', 'rust', 'kotlin', 'swift', 'java']);
|
|
26
|
+
const ENDS = new Set(['frontend', 'backend', 'mobile', 'database', 'devops']);
|
|
27
|
+
const PERSONAS = new Set([
|
|
28
|
+
'backend-architect', 'frontend-developer', 'code-reviewer', 'database-optimizer',
|
|
29
|
+
'devops-automator', 'security-engineer', 'qa-tester',
|
|
30
|
+
]);
|
|
31
|
+
const WORKFLOWS = new Set([
|
|
32
|
+
'coding-standards', 'tdd-workflow', 'git-workflow',
|
|
33
|
+
'architecture-decision-records', 'debugging-workflow',
|
|
34
|
+
]);
|
|
35
|
+
if (LANGUAGES.has(name))
|
|
36
|
+
return 'language';
|
|
37
|
+
if (ENDS.has(name))
|
|
38
|
+
return 'end';
|
|
39
|
+
if (PERSONAS.has(name))
|
|
40
|
+
return 'persona';
|
|
41
|
+
if (WORKFLOWS.has(name))
|
|
42
|
+
return 'workflow';
|
|
43
|
+
return 'other';
|
|
44
|
+
}
|
|
45
|
+
function readFrontmatter(path) {
|
|
46
|
+
try {
|
|
47
|
+
const raw = readFileSync(path, 'utf-8');
|
|
48
|
+
if (!raw.startsWith('---'))
|
|
49
|
+
return { description: '', origin: '' };
|
|
50
|
+
const end = raw.indexOf('\n---', 3);
|
|
51
|
+
if (end === -1)
|
|
52
|
+
return { description: '', origin: '' };
|
|
53
|
+
const yaml = raw.slice(3, end);
|
|
54
|
+
const desc = yaml.match(/^description:\s*(.+)$/m)?.[1] ?? '';
|
|
55
|
+
const origin = yaml.match(/^origin:\s*(.+)$/m)?.[1] ?? '';
|
|
56
|
+
return {
|
|
57
|
+
description: desc.trim().replace(/^["']|["']$/g, ''),
|
|
58
|
+
origin: origin.trim().replace(/^["']|["']$/g, ''),
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
catch {
|
|
62
|
+
return { description: '', origin: '' };
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
function collectLinkedProjects(skillName) {
|
|
66
|
+
if (!existsSync(PROJECTS_DIR))
|
|
67
|
+
return [];
|
|
68
|
+
const out = [];
|
|
69
|
+
const projects = readdirSync(PROJECTS_DIR, { withFileTypes: true })
|
|
70
|
+
.filter((e) => e.isDirectory())
|
|
71
|
+
.map((e) => e.name);
|
|
72
|
+
for (const name of projects) {
|
|
73
|
+
const repo = findProjectRepoDir(name);
|
|
74
|
+
if (!repo || !existsSync(repo))
|
|
75
|
+
continue;
|
|
76
|
+
const info = inspectProjectSkill(repo, skillName);
|
|
77
|
+
if (info && info.state !== 'absent') {
|
|
78
|
+
out.push({ project: name, state: info.state });
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
return out;
|
|
82
|
+
}
|
|
83
|
+
function findBundledSkillsDir() {
|
|
84
|
+
// same strategy as consoleCommand
|
|
85
|
+
const candidates = [
|
|
86
|
+
resolve(process.cwd(), 'skills'),
|
|
87
|
+
// npm package root/skills
|
|
88
|
+
resolve(import.meta.url.replace(/^file:\/\//, ''), '..', '..', '..', '..', 'skills'),
|
|
89
|
+
];
|
|
90
|
+
for (const c of candidates) {
|
|
91
|
+
try {
|
|
92
|
+
if (existsSync(c) && statSync(c).isDirectory())
|
|
93
|
+
return c;
|
|
94
|
+
}
|
|
95
|
+
catch {
|
|
96
|
+
/* ignore */
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
export function createSkillsRoute() {
|
|
102
|
+
const app = new Hono();
|
|
103
|
+
app.get('/', (c) => {
|
|
104
|
+
const users = listUserSkills();
|
|
105
|
+
const forProject = c.req.query('project');
|
|
106
|
+
const enriched = users.map((u) => {
|
|
107
|
+
const fm = readFrontmatter(resolve(u.userPath, 'SKILL.md'));
|
|
108
|
+
const base = {
|
|
109
|
+
name: u.name,
|
|
110
|
+
category: classifyCategory(u.name),
|
|
111
|
+
description: fm.description,
|
|
112
|
+
origin: fm.origin,
|
|
113
|
+
linkedProjects: collectLinkedProjects(u.name).map((p) => p.project),
|
|
114
|
+
};
|
|
115
|
+
if (forProject) {
|
|
116
|
+
const repo = findProjectRepoDir(forProject);
|
|
117
|
+
const state = repo ? inspectProjectSkill(repo, u.name)?.state ?? 'absent' : 'absent';
|
|
118
|
+
return { ...base, stateInProject: state };
|
|
119
|
+
}
|
|
120
|
+
return base;
|
|
121
|
+
});
|
|
122
|
+
return c.json({ data: enriched });
|
|
123
|
+
});
|
|
124
|
+
app.get('/:name', (c) => {
|
|
125
|
+
const name = c.req.param('name');
|
|
126
|
+
const users = listUserSkills();
|
|
127
|
+
const user = users.find((u) => u.name === name);
|
|
128
|
+
if (!user) {
|
|
129
|
+
return c.json({ type: 'not-found', title: 'Skill not found', status: 404, detail: name }, 404);
|
|
130
|
+
}
|
|
131
|
+
const fm = readFrontmatter(resolve(user.userPath, 'SKILL.md'));
|
|
132
|
+
const body = (() => {
|
|
133
|
+
try {
|
|
134
|
+
return readFileSync(resolve(user.userPath, 'SKILL.md'), 'utf-8');
|
|
135
|
+
}
|
|
136
|
+
catch {
|
|
137
|
+
return '';
|
|
138
|
+
}
|
|
139
|
+
})();
|
|
140
|
+
const references = existsSync(resolve(user.userPath, 'references'))
|
|
141
|
+
? readdirSync(resolve(user.userPath, 'references'))
|
|
142
|
+
.filter((f) => f.endsWith('.md'))
|
|
143
|
+
.map((f) => {
|
|
144
|
+
const full = resolve(user.userPath, 'references', f);
|
|
145
|
+
try {
|
|
146
|
+
return { name: f, lines: readFileSync(full, 'utf-8').split('\n').length };
|
|
147
|
+
}
|
|
148
|
+
catch {
|
|
149
|
+
return { name: f, lines: 0 };
|
|
150
|
+
}
|
|
151
|
+
})
|
|
152
|
+
: [];
|
|
153
|
+
const linkedProjects = collectLinkedProjects(name);
|
|
154
|
+
return c.json({
|
|
155
|
+
name,
|
|
156
|
+
category: classifyCategory(name),
|
|
157
|
+
description: fm.description,
|
|
158
|
+
origin: fm.origin,
|
|
159
|
+
body,
|
|
160
|
+
references,
|
|
161
|
+
linkedProjects,
|
|
162
|
+
});
|
|
163
|
+
});
|
|
164
|
+
app.post('/:name/link', async (c) => {
|
|
165
|
+
const name = c.req.param('name');
|
|
166
|
+
const body = await c.req.json().catch(() => null);
|
|
167
|
+
if (!body?.project) {
|
|
168
|
+
return c.json({ type: 'validation', title: 'project required', status: 422 }, 422);
|
|
169
|
+
}
|
|
170
|
+
const repo = findProjectRepoDir(body.project);
|
|
171
|
+
if (!repo) {
|
|
172
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404 }, 404);
|
|
173
|
+
}
|
|
174
|
+
const result = addSkillToProject(repo, name);
|
|
175
|
+
if (result === 'skipped-absent') {
|
|
176
|
+
return c.json({ type: 'not-found', title: 'Skill not in ~/.coral/skills/', status: 404 }, 404);
|
|
177
|
+
}
|
|
178
|
+
ensureSkillsGitignore(repo);
|
|
179
|
+
return c.json({ state: result });
|
|
180
|
+
});
|
|
181
|
+
app.delete('/:name/link', (c) => {
|
|
182
|
+
const name = c.req.param('name');
|
|
183
|
+
const project = c.req.query('project');
|
|
184
|
+
if (!project) {
|
|
185
|
+
return c.json({ type: 'validation', title: 'project required', status: 422 }, 422);
|
|
186
|
+
}
|
|
187
|
+
const repo = findProjectRepoDir(project);
|
|
188
|
+
if (!repo) {
|
|
189
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404 }, 404);
|
|
190
|
+
}
|
|
191
|
+
const removed = removeSkillFromProject(repo, name);
|
|
192
|
+
return c.json({ removed });
|
|
193
|
+
});
|
|
194
|
+
app.post('/:name/freeze', async (c) => {
|
|
195
|
+
const name = c.req.param('name');
|
|
196
|
+
const body = await c.req.json().catch(() => null);
|
|
197
|
+
if (!body?.project) {
|
|
198
|
+
return c.json({ type: 'validation', title: 'project required', status: 422 }, 422);
|
|
199
|
+
}
|
|
200
|
+
const repo = findProjectRepoDir(body.project);
|
|
201
|
+
if (!repo)
|
|
202
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404 }, 404);
|
|
203
|
+
const ok = freezeSkillInProject(repo, name);
|
|
204
|
+
return c.json({ ok, state: ok ? 'frozen' : 'unchanged' });
|
|
205
|
+
});
|
|
206
|
+
app.post('/:name/unfreeze', async (c) => {
|
|
207
|
+
const name = c.req.param('name');
|
|
208
|
+
const body = await c.req.json().catch(() => null);
|
|
209
|
+
if (!body?.project) {
|
|
210
|
+
return c.json({ type: 'validation', title: 'project required', status: 422 }, 422);
|
|
211
|
+
}
|
|
212
|
+
const repo = findProjectRepoDir(body.project);
|
|
213
|
+
if (!repo)
|
|
214
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404 }, 404);
|
|
215
|
+
const ok = unfreezeSkillInProject(repo, name);
|
|
216
|
+
return c.json({ ok, state: ok ? 'linked' : 'unchanged' });
|
|
217
|
+
});
|
|
218
|
+
app.post('/sync', (c) => {
|
|
219
|
+
const bundled = findBundledSkillsDir();
|
|
220
|
+
if (!bundled) {
|
|
221
|
+
return c.json({ copied: 0, skipped: 0, note: 'bundled dir not found' });
|
|
222
|
+
}
|
|
223
|
+
const result = syncBundledSkillsToUser(bundled);
|
|
224
|
+
return c.json({ ...result, userRoot: userSkillsRoot() });
|
|
225
|
+
});
|
|
226
|
+
// 当前项目的 .claude/skills 列表(用于调试)
|
|
227
|
+
app.get('/links/:project', (c) => {
|
|
228
|
+
const project = c.req.param('project');
|
|
229
|
+
const repo = findProjectRepoDir(project);
|
|
230
|
+
if (!repo) {
|
|
231
|
+
return c.json({ type: 'not-found', title: 'Project not found', status: 404 }, 404);
|
|
232
|
+
}
|
|
233
|
+
const dir = projectSkillsDir(repo);
|
|
234
|
+
return c.json({
|
|
235
|
+
repo,
|
|
236
|
+
skillsDir: dir,
|
|
237
|
+
links: existsSync(dir) ? readdirSync(dir) : [],
|
|
238
|
+
});
|
|
239
|
+
});
|
|
240
|
+
return app;
|
|
241
|
+
}
|
|
242
|
+
//# sourceMappingURL=skills.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"skills.js","sourceRoot":"","sources":["../../../src/console-server/routes/skills.ts"],"names":[],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"system.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/system.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"system.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/system.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AA0B5B,wBAAgB,iBAAiB,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,EAAE,IAAI,GAAG,IAAI,CAiExE"}
|
|
@@ -1,8 +1,30 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* @module console-server/routes/system
|
|
3
|
-
* @description
|
|
3
|
+
* @description 系统信息:版本、运行时、env(脱敏)、doctor 聚合
|
|
4
4
|
*/
|
|
5
5
|
import { Hono } from 'hono';
|
|
6
|
+
import { existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
|
|
7
|
+
import { resolve } from 'node:path';
|
|
8
|
+
const HOME = process.env.HOME || '/home/coral';
|
|
9
|
+
const ENV_PATH = resolve(HOME, '.coral', 'env');
|
|
10
|
+
function maskSecret(value) {
|
|
11
|
+
if (!value)
|
|
12
|
+
return '';
|
|
13
|
+
if (value.length <= 6)
|
|
14
|
+
return '****';
|
|
15
|
+
return value.slice(0, 4) + '****';
|
|
16
|
+
}
|
|
17
|
+
const SECRET_KEY_PATTERNS = [
|
|
18
|
+
/_TOKEN$/,
|
|
19
|
+
/_KEY$/,
|
|
20
|
+
/_SECRET$/,
|
|
21
|
+
/_PASSWORD$/,
|
|
22
|
+
/_PASS$/,
|
|
23
|
+
/^(ANTHROPIC|OPENAI|CLAUDE|PLANE|TRELLO|MATRIX)_/,
|
|
24
|
+
];
|
|
25
|
+
function isSecret(key) {
|
|
26
|
+
return SECRET_KEY_PATTERNS.some((p) => p.test(key));
|
|
27
|
+
}
|
|
6
28
|
export function createSystemRoute(version, startedAt) {
|
|
7
29
|
const app = new Hono();
|
|
8
30
|
app.get('/info', (c) => {
|
|
@@ -12,8 +34,63 @@ export function createSystemRoute(version, startedAt) {
|
|
|
12
34
|
startedAt: startedAt.toISOString(),
|
|
13
35
|
uptimeMs: Date.now() - startedAt.getTime(),
|
|
14
36
|
platform: process.platform,
|
|
37
|
+
pid: process.pid,
|
|
15
38
|
});
|
|
16
39
|
});
|
|
40
|
+
app.get('/env', (c) => {
|
|
41
|
+
if (!existsSync(ENV_PATH)) {
|
|
42
|
+
return c.json({ path: ENV_PATH, exists: false, entries: [] });
|
|
43
|
+
}
|
|
44
|
+
const raw = readFileSync(ENV_PATH, 'utf-8');
|
|
45
|
+
const entries = [];
|
|
46
|
+
for (const line of raw.split('\n')) {
|
|
47
|
+
const t = line.trim();
|
|
48
|
+
if (!t || t.startsWith('#'))
|
|
49
|
+
continue;
|
|
50
|
+
const m = t.match(/^(?:export\s+)?([A-Z_][A-Z0-9_]*)=["']?(.*?)["']?\s*$/);
|
|
51
|
+
if (!m)
|
|
52
|
+
continue;
|
|
53
|
+
const key = m[1] ?? '';
|
|
54
|
+
const value = m[2] ?? '';
|
|
55
|
+
const masked = isSecret(key);
|
|
56
|
+
entries.push({ key, value: masked ? maskSecret(value) : value, masked });
|
|
57
|
+
}
|
|
58
|
+
return c.json({ path: ENV_PATH, exists: true, entries });
|
|
59
|
+
});
|
|
60
|
+
app.get('/doctor/all', (c) => {
|
|
61
|
+
const projectsDir = resolve(HOME, '.coral', 'projects');
|
|
62
|
+
const report = [];
|
|
63
|
+
if (existsSync(projectsDir)) {
|
|
64
|
+
const names = readdirSync(projectsDir, { withFileTypes: true })
|
|
65
|
+
.filter((e) => e.isDirectory())
|
|
66
|
+
.map((e) => e.name);
|
|
67
|
+
for (const name of names) {
|
|
68
|
+
const issues = [];
|
|
69
|
+
const dir = resolve(projectsDir, name);
|
|
70
|
+
if (!existsSync(resolve(dir, 'conf')))
|
|
71
|
+
issues.push('missing conf');
|
|
72
|
+
if (!existsSync(resolve(dir, 'cards')))
|
|
73
|
+
issues.push('missing cards/');
|
|
74
|
+
const runtime = resolve(dir, 'runtime');
|
|
75
|
+
if (existsSync(runtime)) {
|
|
76
|
+
const markers = readdirSync(runtime).filter((f) => /worker-\d+-current\.json$/.test(f));
|
|
77
|
+
for (const mf of markers) {
|
|
78
|
+
try {
|
|
79
|
+
const stat = statSync(resolve(runtime, mf));
|
|
80
|
+
const ageMin = Math.floor((Date.now() - stat.mtimeMs) / 60000);
|
|
81
|
+
if (ageMin > 60)
|
|
82
|
+
issues.push(`stale marker ${mf} (${ageMin}m)`);
|
|
83
|
+
}
|
|
84
|
+
catch {
|
|
85
|
+
/* ignore */
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
report.push({ project: name, issues, ok: issues.length === 0 });
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return c.json({ data: report });
|
|
93
|
+
});
|
|
17
94
|
return app;
|
|
18
95
|
}
|
|
19
96
|
//# sourceMappingURL=system.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"system.js","sourceRoot":"","sources":["../../../src/console-server/routes/system.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"system.js","sourceRoot":"","sources":["../../../src/console-server/routes/system.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"workers.d.ts","sourceRoot":"","sources":["../../../src/console-server/routes/workers.ts"],"names":[],"mappings":"AAAA;;;GAGG;AACH,OAAO,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAwG5B,wBAAgB,kBAAkB,IAAI,IAAI,CA2CzC"}
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @module console-server/routes/workers
|
|
3
|
+
* @description Worker slot 列表 + kill/launch
|
|
4
|
+
*/
|
|
5
|
+
import { Hono } from 'hono';
|
|
6
|
+
import { existsSync, readdirSync, readFileSync, statSync } from 'node:fs';
|
|
7
|
+
import { resolve } from 'node:path';
|
|
8
|
+
import { spawnCliSync } from '../lib/spawnCli.js';
|
|
9
|
+
const HOME = process.env.HOME || '/home/coral';
|
|
10
|
+
const STUCK_THRESHOLD_MS = 5 * 60 * 1000; // 5 min 无 marker 更新 = stuck
|
|
11
|
+
function projectRuntimeDir(project) {
|
|
12
|
+
return resolve(HOME, '.coral', 'projects', project, 'runtime');
|
|
13
|
+
}
|
|
14
|
+
function isPidAlive(pid) {
|
|
15
|
+
if (!pid)
|
|
16
|
+
return false;
|
|
17
|
+
try {
|
|
18
|
+
process.kill(pid, 0);
|
|
19
|
+
return true;
|
|
20
|
+
}
|
|
21
|
+
catch {
|
|
22
|
+
return false;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
function parseMarker(markerPath) {
|
|
26
|
+
const basename = markerPath.split('/').pop() ?? '';
|
|
27
|
+
const m = basename.match(/^worker-(\d+)-current\.json$/);
|
|
28
|
+
if (!m)
|
|
29
|
+
return null;
|
|
30
|
+
const slot = Number.parseInt(m[1] ?? '', 10);
|
|
31
|
+
if (!Number.isFinite(slot))
|
|
32
|
+
return null;
|
|
33
|
+
try {
|
|
34
|
+
const data = JSON.parse(readFileSync(markerPath, 'utf-8'));
|
|
35
|
+
return { slot, data };
|
|
36
|
+
}
|
|
37
|
+
catch {
|
|
38
|
+
return { slot, data: null };
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
function workerFromMarker(project, slot, markerPath) {
|
|
42
|
+
const now = Date.now();
|
|
43
|
+
let pid = null;
|
|
44
|
+
let card = null;
|
|
45
|
+
let stage = null;
|
|
46
|
+
let startedAt = null;
|
|
47
|
+
let markerUpdatedAt = null;
|
|
48
|
+
const parsed = parseMarker(markerPath);
|
|
49
|
+
if (parsed?.data) {
|
|
50
|
+
const d = parsed.data;
|
|
51
|
+
if (typeof d.pid === 'number')
|
|
52
|
+
pid = d.pid;
|
|
53
|
+
if (typeof d.seq === 'number' && typeof d.title === 'string') {
|
|
54
|
+
card = { seq: d.seq, title: d.title };
|
|
55
|
+
}
|
|
56
|
+
else if (typeof d.seq === 'number') {
|
|
57
|
+
card = { seq: d.seq, title: `#${d.seq}` };
|
|
58
|
+
}
|
|
59
|
+
if (typeof d.stage === 'string')
|
|
60
|
+
stage = d.stage;
|
|
61
|
+
if (typeof d.startedAt === 'string')
|
|
62
|
+
startedAt = d.startedAt;
|
|
63
|
+
if (typeof d.startTime === 'string')
|
|
64
|
+
startedAt = startedAt ?? d.startTime;
|
|
65
|
+
}
|
|
66
|
+
try {
|
|
67
|
+
const stat = statSync(markerPath);
|
|
68
|
+
markerUpdatedAt = new Date(stat.mtimeMs).toISOString();
|
|
69
|
+
}
|
|
70
|
+
catch {
|
|
71
|
+
/* ignore */
|
|
72
|
+
}
|
|
73
|
+
const alive = isPidAlive(pid);
|
|
74
|
+
const fresh = markerUpdatedAt ? now - new Date(markerUpdatedAt).getTime() < STUCK_THRESHOLD_MS : false;
|
|
75
|
+
const state = alive
|
|
76
|
+
? fresh
|
|
77
|
+
? 'running'
|
|
78
|
+
: 'stuck'
|
|
79
|
+
: (card !== null)
|
|
80
|
+
? 'crashed'
|
|
81
|
+
: 'idle';
|
|
82
|
+
const runtimeMs = startedAt ? now - new Date(startedAt).getTime() : null;
|
|
83
|
+
return { slot, pid, state, card, stage, startedAt, runtimeMs, markerUpdatedAt };
|
|
84
|
+
}
|
|
85
|
+
function listWorkerMarkerPaths(project) {
|
|
86
|
+
const dir = projectRuntimeDir(project);
|
|
87
|
+
if (!existsSync(dir))
|
|
88
|
+
return [];
|
|
89
|
+
return readdirSync(dir)
|
|
90
|
+
.filter((f) => /^worker-\d+-current\.json$/.test(f))
|
|
91
|
+
.map((f) => ({
|
|
92
|
+
slot: Number.parseInt((f.match(/^worker-(\d+)-/) ?? ['', '0'])[1] ?? '0', 10),
|
|
93
|
+
path: resolve(dir, f),
|
|
94
|
+
}))
|
|
95
|
+
.filter((m) => Number.isFinite(m.slot) && m.slot > 0)
|
|
96
|
+
.sort((a, b) => a.slot - b.slot);
|
|
97
|
+
}
|
|
98
|
+
export function createWorkersRoute() {
|
|
99
|
+
const app = new Hono();
|
|
100
|
+
app.get('/:project/workers', (c) => {
|
|
101
|
+
const project = c.req.param('project');
|
|
102
|
+
if (!existsSync(projectRuntimeDir(project))) {
|
|
103
|
+
return c.json({ data: [] });
|
|
104
|
+
}
|
|
105
|
+
const markers = listWorkerMarkerPaths(project);
|
|
106
|
+
const workers = markers.map((m) => workerFromMarker(project, m.slot, m.path));
|
|
107
|
+
return c.json({ data: workers });
|
|
108
|
+
});
|
|
109
|
+
app.post('/:project/workers/:slot/kill', async (c) => {
|
|
110
|
+
const project = c.req.param('project');
|
|
111
|
+
const slot = c.req.param('slot');
|
|
112
|
+
const result = await spawnCliSync(['worker', 'kill', project, slot], { timeoutMs: 15_000 });
|
|
113
|
+
if (result.exitCode !== 0) {
|
|
114
|
+
return c.json({ type: 'cli-error', title: 'kill failed', status: 500, detail: result.stderr }, 500);
|
|
115
|
+
}
|
|
116
|
+
return c.json({ ok: true });
|
|
117
|
+
});
|
|
118
|
+
app.post('/:project/workers/:slot/launch', async (c) => {
|
|
119
|
+
const project = c.req.param('project');
|
|
120
|
+
const slot = c.req.param('slot');
|
|
121
|
+
const body = await c.req.json().catch(() => ({}));
|
|
122
|
+
const args = ['worker', 'launch', project, slot];
|
|
123
|
+
if (typeof body?.seq === 'number')
|
|
124
|
+
args.push(String(body.seq));
|
|
125
|
+
const result = await spawnCliSync(args, { timeoutMs: 15_000 });
|
|
126
|
+
if (result.exitCode !== 0) {
|
|
127
|
+
return c.json({ type: 'cli-error', title: 'launch failed', status: 500, detail: result.stderr }, 500);
|
|
128
|
+
}
|
|
129
|
+
return c.json({ ok: true });
|
|
130
|
+
});
|
|
131
|
+
return app;
|
|
132
|
+
}
|
|
133
|
+
//# sourceMappingURL=workers.js.map
|