@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.
- package/dist/commands/cardAdd.d.ts.map +1 -1
- package/dist/commands/cardAdd.js +47 -67
- package/dist/commands/cardAdd.js.map +1 -1
- package/dist/commands/cardMarkComplete.js +1 -1
- package/dist/commands/cardMarkComplete.js.map +1 -1
- package/dist/commands/cardMarkStarted.js +1 -1
- package/dist/commands/cardMarkStarted.js.map +1 -1
- package/dist/commands/consoleCommand.js +3 -3
- package/dist/commands/consoleCommand.js.map +1 -1
- package/dist/commands/hookCommand.d.ts +25 -0
- package/dist/commands/hookCommand.d.ts.map +1 -1
- package/dist/commands/hookCommand.js +2 -2
- package/dist/commands/hookCommand.js.map +1 -1
- package/dist/console/index.d.ts +15 -0
- package/dist/console/index.d.ts.map +1 -0
- package/dist/console/index.js +177 -0
- package/dist/console/index.js.map +1 -0
- package/dist/console/lib/lockFile.d.ts +17 -0
- package/dist/console/lib/lockFile.d.ts.map +1 -0
- package/dist/console/lib/lockFile.js +61 -0
- package/dist/console/lib/lockFile.js.map +1 -0
- package/dist/console/lib/portPicker.d.ts +3 -0
- package/dist/console/lib/portPicker.d.ts.map +1 -0
- package/dist/console/lib/portPicker.js +25 -0
- package/dist/console/lib/portPicker.js.map +1 -0
- package/dist/console/lib/resultToJson.d.ts +18 -0
- package/dist/console/lib/resultToJson.d.ts.map +1 -0
- package/dist/console/lib/resultToJson.js +19 -0
- package/dist/console/lib/resultToJson.js.map +1 -0
- package/dist/console/routes/cards.d.ts +26 -0
- package/dist/console/routes/cards.d.ts.map +1 -0
- package/dist/console/routes/cards.js +85 -0
- package/dist/console/routes/cards.js.map +1 -0
- package/dist/console/routes/chat.d.ts +36 -0
- package/dist/console/routes/chat.d.ts.map +1 -0
- package/dist/console/routes/chat.js +458 -0
- package/dist/console/routes/chat.js.map +1 -0
- package/dist/console/routes/logs.d.ts +17 -0
- package/dist/console/routes/logs.d.ts.map +1 -0
- package/dist/console/routes/logs.js +159 -0
- package/dist/console/routes/logs.js.map +1 -0
- package/dist/console/routes/pipeline.d.ts +10 -0
- package/dist/console/routes/pipeline.d.ts.map +1 -0
- package/dist/console/routes/pipeline.js +39 -0
- package/dist/console/routes/pipeline.js.map +1 -0
- package/dist/console/routes/projects.d.ts +15 -0
- package/dist/console/routes/projects.d.ts.map +1 -0
- package/dist/console/routes/projects.js +132 -0
- package/dist/console/routes/projects.js.map +1 -0
- package/dist/console/routes/skills.d.ts +4 -0
- package/dist/console/routes/skills.d.ts.map +1 -0
- package/dist/console/routes/skills.js +91 -0
- package/dist/console/routes/skills.js.map +1 -0
- package/dist/console/routes/system.d.ts +7 -0
- package/dist/console/routes/system.d.ts.map +1 -0
- package/dist/console/routes/system.js +288 -0
- package/dist/console/routes/system.js.map +1 -0
- package/dist/console/routes/workers.d.ts +5 -0
- package/dist/console/routes/workers.d.ts.map +1 -0
- package/dist/console/routes/workers.js +149 -0
- package/dist/console/routes/workers.js.map +1 -0
- package/dist/console/sse/eventBus.d.ts +25 -0
- package/dist/console/sse/eventBus.d.ts.map +1 -0
- package/dist/console/sse/eventBus.js +32 -0
- package/dist/console/sse/eventBus.js.map +1 -0
- package/dist/console/sse/projectStream.d.ts +13 -0
- package/dist/console/sse/projectStream.d.ts.map +1 -0
- package/dist/console/sse/projectStream.js +84 -0
- package/dist/console/sse/projectStream.js.map +1 -0
- package/dist/infra/chokidarWatchers.d.ts +38 -0
- package/dist/infra/chokidarWatchers.d.ts.map +1 -0
- package/dist/infra/chokidarWatchers.js +213 -0
- package/dist/infra/chokidarWatchers.js.map +1 -0
- package/dist/infra/clock.d.ts +35 -0
- package/dist/infra/clock.d.ts.map +1 -0
- package/dist/infra/clock.js +45 -0
- package/dist/infra/clock.js.map +1 -0
- package/dist/infra/filesystem.d.ts +89 -0
- package/dist/infra/filesystem.d.ts.map +1 -0
- package/dist/infra/filesystem.js +247 -0
- package/dist/infra/filesystem.js.map +1 -0
- package/dist/infra/spawn.d.ts +67 -0
- package/dist/infra/spawn.d.ts.map +1 -0
- package/dist/infra/spawn.js +116 -0
- package/dist/infra/spawn.js.map +1 -0
- package/dist/infra/sseBus.d.ts +36 -0
- package/dist/infra/sseBus.d.ts.map +1 -0
- package/dist/infra/sseBus.js +72 -0
- package/dist/infra/sseBus.js.map +1 -0
- package/dist/interfaces/ACPClient.d.ts +1 -1
- package/dist/interfaces/ACPClient.d.ts.map +1 -1
- package/dist/main.js +1 -1
- package/dist/main.js.map +1 -1
- package/dist/providers/LocalACPClient.d.ts +1 -1
- package/dist/providers/LocalACPClient.d.ts.map +1 -1
- package/dist/services/CardService.d.ts +86 -0
- package/dist/services/CardService.d.ts.map +1 -0
- package/dist/services/CardService.js +313 -0
- package/dist/services/CardService.js.map +1 -0
- package/dist/services/ChatService.d.ts +62 -0
- package/dist/services/ChatService.d.ts.map +1 -0
- package/dist/services/ChatService.js +157 -0
- package/dist/services/ChatService.js.map +1 -0
- package/dist/services/LogService.d.ts +46 -0
- package/dist/services/LogService.d.ts.map +1 -0
- package/dist/services/LogService.js +185 -0
- package/dist/services/LogService.js.map +1 -0
- package/dist/services/PipelineService.d.ts +71 -0
- package/dist/services/PipelineService.d.ts.map +1 -0
- package/dist/services/PipelineService.js +349 -0
- package/dist/services/PipelineService.js.map +1 -0
- package/dist/services/ProjectService.d.ts +105 -0
- package/dist/services/ProjectService.d.ts.map +1 -0
- package/dist/services/ProjectService.js +346 -0
- package/dist/services/ProjectService.js.map +1 -0
- package/dist/services/SkillService.d.ts +38 -0
- package/dist/services/SkillService.d.ts.map +1 -0
- package/dist/services/SkillService.js +301 -0
- package/dist/services/SkillService.js.map +1 -0
- package/dist/services/WorkerService.d.ts +83 -0
- package/dist/services/WorkerService.d.ts.map +1 -0
- package/dist/services/WorkerService.js +262 -0
- package/dist/services/WorkerService.js.map +1 -0
- package/dist/services/container.d.ts +47 -0
- package/dist/services/container.d.ts.map +1 -0
- package/dist/services/container.js +77 -0
- package/dist/services/container.js.map +1 -0
- package/dist/services/defaults.d.ts +15 -0
- package/dist/services/defaults.d.ts.map +1 -0
- package/dist/services/defaults.js +11 -0
- package/dist/services/defaults.js.map +1 -0
- package/dist/services/executors.d.ts +38 -0
- package/dist/services/executors.d.ts.map +1 -0
- package/dist/services/executors.js +157 -0
- package/dist/services/executors.js.map +1 -0
- package/dist/services/ports.d.ts +17 -0
- package/dist/services/ports.d.ts.map +1 -0
- package/dist/services/ports.js +2 -0
- package/dist/services/ports.js.map +1 -0
- package/dist/shared/domainEvents.d.ts +118 -0
- package/dist/shared/domainEvents.d.ts.map +1 -0
- package/dist/shared/domainEvents.js +40 -0
- package/dist/shared/domainEvents.js.map +1 -0
- package/dist/shared/errors.d.ts +59 -0
- package/dist/shared/errors.d.ts.map +1 -0
- package/dist/shared/errors.js +79 -0
- package/dist/shared/errors.js.map +1 -0
- package/dist/shared/result.d.ts +51 -0
- package/dist/shared/result.d.ts.map +1 -0
- package/dist/shared/result.js +48 -0
- package/dist/shared/result.js.map +1 -0
- package/dist/shared/runtimePaths.d.ts +78 -0
- package/dist/shared/runtimePaths.d.ts.map +1 -0
- package/dist/shared/runtimePaths.js +179 -0
- package/dist/shared/runtimePaths.js.map +1 -0
- package/dist/shared/runtimeSchemas.d.ts +324 -0
- package/dist/shared/runtimeSchemas.d.ts.map +1 -0
- package/dist/shared/runtimeSchemas.js +124 -0
- package/dist/shared/runtimeSchemas.js.map +1 -0
- package/dist/shared/types.d.ts +12 -0
- package/dist/shared/types.d.ts.map +1 -0
- package/dist/shared/types.js +2 -0
- package/dist/shared/types.js.map +1 -0
- 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
|