agent-tasks 1.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (91) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +74 -0
  3. package/dist/context.d.ts +17 -0
  4. package/dist/context.d.ts.map +1 -0
  5. package/dist/context.js +37 -0
  6. package/dist/context.js.map +1 -0
  7. package/dist/db.d.ts +10 -0
  8. package/dist/db.d.ts.map +1 -0
  9. package/dist/db.js +112 -0
  10. package/dist/db.js.map +1 -0
  11. package/dist/domain/agent-bridge.d.ts +13 -0
  12. package/dist/domain/agent-bridge.d.ts.map +1 -0
  13. package/dist/domain/agent-bridge.js +99 -0
  14. package/dist/domain/agent-bridge.js.map +1 -0
  15. package/dist/domain/approvals.d.ts +18 -0
  16. package/dist/domain/approvals.d.ts.map +1 -0
  17. package/dist/domain/approvals.js +89 -0
  18. package/dist/domain/approvals.js.map +1 -0
  19. package/dist/domain/cleanup.d.ts +28 -0
  20. package/dist/domain/cleanup.d.ts.map +1 -0
  21. package/dist/domain/cleanup.js +68 -0
  22. package/dist/domain/cleanup.js.map +1 -0
  23. package/dist/domain/collaborators.d.ts +14 -0
  24. package/dist/domain/collaborators.d.ts.map +1 -0
  25. package/dist/domain/collaborators.js +59 -0
  26. package/dist/domain/collaborators.js.map +1 -0
  27. package/dist/domain/comments.d.ts +14 -0
  28. package/dist/domain/comments.d.ts.map +1 -0
  29. package/dist/domain/comments.js +63 -0
  30. package/dist/domain/comments.js.map +1 -0
  31. package/dist/domain/events.d.ts +9 -0
  32. package/dist/domain/events.d.ts.map +1 -0
  33. package/dist/domain/events.js +52 -0
  34. package/dist/domain/events.js.map +1 -0
  35. package/dist/domain/rules.d.ts +2 -0
  36. package/dist/domain/rules.d.ts.map +1 -0
  37. package/dist/domain/rules.js +67 -0
  38. package/dist/domain/rules.js.map +1 -0
  39. package/dist/domain/tasks.d.ts +60 -0
  40. package/dist/domain/tasks.d.ts.map +1 -0
  41. package/dist/domain/tasks.js +616 -0
  42. package/dist/domain/tasks.js.map +1 -0
  43. package/dist/domain/validate.d.ts +14 -0
  44. package/dist/domain/validate.d.ts.map +1 -0
  45. package/dist/domain/validate.js +29 -0
  46. package/dist/domain/validate.js.map +1 -0
  47. package/dist/event-bus.d.ts +10 -0
  48. package/dist/event-bus.d.ts.map +1 -0
  49. package/dist/event-bus.js +38 -0
  50. package/dist/event-bus.js.map +1 -0
  51. package/dist/index.d.ts +3 -0
  52. package/dist/index.d.ts.map +1 -0
  53. package/dist/index.js +121 -0
  54. package/dist/index.js.map +1 -0
  55. package/dist/server.d.ts +10 -0
  56. package/dist/server.d.ts.map +1 -0
  57. package/dist/server.js +95 -0
  58. package/dist/server.js.map +1 -0
  59. package/dist/session.d.ts +7 -0
  60. package/dist/session.d.ts.map +1 -0
  61. package/dist/session.js +11 -0
  62. package/dist/session.js.map +1 -0
  63. package/dist/storage/database.d.ts +15 -0
  64. package/dist/storage/database.d.ts.map +1 -0
  65. package/dist/storage/database.js +215 -0
  66. package/dist/storage/database.js.map +1 -0
  67. package/dist/tasks.d.ts +32 -0
  68. package/dist/tasks.d.ts.map +1 -0
  69. package/dist/tasks.js +410 -0
  70. package/dist/tasks.js.map +1 -0
  71. package/dist/transport/mcp.d.ts +6 -0
  72. package/dist/transport/mcp.d.ts.map +1 -0
  73. package/dist/transport/mcp.js +573 -0
  74. package/dist/transport/mcp.js.map +1 -0
  75. package/dist/transport/rest.d.ts +4 -0
  76. package/dist/transport/rest.d.ts.map +1 -0
  77. package/dist/transport/rest.js +382 -0
  78. package/dist/transport/rest.js.map +1 -0
  79. package/dist/transport/ws.d.ts +10 -0
  80. package/dist/transport/ws.d.ts.map +1 -0
  81. package/dist/transport/ws.js +177 -0
  82. package/dist/transport/ws.js.map +1 -0
  83. package/dist/types.d.ts +146 -0
  84. package/dist/types.d.ts.map +1 -0
  85. package/dist/types.js +35 -0
  86. package/dist/types.js.map +1 -0
  87. package/dist/ui/app.js +648 -0
  88. package/dist/ui/index.html +82 -0
  89. package/dist/ui/morphdom.min.js +1 -0
  90. package/dist/ui/styles.css +805 -0
  91. package/package.json +78 -0
@@ -0,0 +1,382 @@
1
+ // =============================================================================
2
+ // agent-tasks — REST transport
3
+ //
4
+ // Lightweight HTTP API using only node:http. No framework dependencies.
5
+ // Serves both the JSON API and the static web UI.
6
+ // =============================================================================
7
+ import { readFileSync, realpathSync, existsSync } from 'fs';
8
+ import { join, extname, resolve, dirname } from 'path';
9
+ import { fileURLToPath } from 'url';
10
+ import { TasksError, ValidationError } from '../types.js';
11
+ const MAX_BODY_SIZE = 65536;
12
+ function parseBody(req) {
13
+ return new Promise((resolve, reject) => {
14
+ const chunks = [];
15
+ let size = 0;
16
+ req.on('data', (chunk) => {
17
+ size += chunk.length;
18
+ if (size > MAX_BODY_SIZE) {
19
+ req.destroy();
20
+ reject(new ValidationError('Request body too large (max 64KB).'));
21
+ return;
22
+ }
23
+ chunks.push(chunk);
24
+ });
25
+ req.on('end', () => {
26
+ try {
27
+ const raw = Buffer.concat(chunks).toString('utf-8');
28
+ if (!raw.trim()) {
29
+ resolve({});
30
+ return;
31
+ }
32
+ const parsed = JSON.parse(raw);
33
+ if (typeof parsed !== 'object' || parsed === null || Array.isArray(parsed)) {
34
+ reject(new ValidationError('Request body must be a JSON object.'));
35
+ return;
36
+ }
37
+ resolve(parsed);
38
+ }
39
+ catch {
40
+ reject(new ValidationError('Invalid JSON in request body.'));
41
+ }
42
+ });
43
+ req.on('error', reject);
44
+ });
45
+ }
46
+ const __dirname = dirname(fileURLToPath(import.meta.url));
47
+ const MIME_TYPES = {
48
+ '.html': 'text/html; charset=utf-8',
49
+ '.css': 'text/css; charset=utf-8',
50
+ '.js': 'application/javascript; charset=utf-8',
51
+ '.json': 'application/json; charset=utf-8',
52
+ '.svg': 'image/svg+xml',
53
+ '.png': 'image/png',
54
+ '.ico': 'image/x-icon',
55
+ };
56
+ export function createRouter(ctx) {
57
+ const pkg = JSON.parse(readFileSync(join(__dirname, '..', '..', 'package.json'), 'utf8'));
58
+ const routes = [];
59
+ const uiDir = resolve(join(__dirname, '..', 'ui'));
60
+ function route(method, path, handler) {
61
+ const paramNames = [];
62
+ const pattern = path.replace(/:(\w+)/g, (_match, name) => {
63
+ paramNames.push(name);
64
+ return '([^/]+)';
65
+ });
66
+ routes.push({ method, pattern: new RegExp(`^${pattern}$`), paramNames, handler });
67
+ }
68
+ function json(res, data, status = 200) {
69
+ res.writeHead(status, {
70
+ 'Content-Type': 'application/json',
71
+ 'Access-Control-Allow-Origin': '*',
72
+ 'X-Content-Type-Options': 'nosniff',
73
+ });
74
+ res.end(JSON.stringify(data));
75
+ }
76
+ // -----------------------------------------------------------------------
77
+ // API routes
78
+ // -----------------------------------------------------------------------
79
+ route('GET', '/health', (_req, res) => {
80
+ json(res, {
81
+ status: 'ok',
82
+ version: pkg.version,
83
+ uptime: process.uptime(),
84
+ tasks: ctx.tasks.list().length,
85
+ });
86
+ });
87
+ route('GET', '/api/tasks', (req, res) => {
88
+ const url = new URL(req.url, `http://${req.headers.host}`);
89
+ json(res, ctx.tasks.list({
90
+ status: url.searchParams.get('status') ?? undefined,
91
+ assigned_to: url.searchParams.get('assigned_to') ?? undefined,
92
+ stage: url.searchParams.get('stage') ?? undefined,
93
+ project: url.searchParams.get('project') ?? undefined,
94
+ limit: url.searchParams.has('limit')
95
+ ? parseInt(url.searchParams.get('limit'), 10)
96
+ : undefined,
97
+ }));
98
+ });
99
+ route('GET', '/api/tasks/:id', (_req, res, params) => {
100
+ const task = ctx.tasks.getById(parseInt(params.id, 10));
101
+ if (!task) {
102
+ json(res, { error: 'Task not found' }, 404);
103
+ return;
104
+ }
105
+ json(res, task);
106
+ });
107
+ route('GET', '/api/tasks/:id/artifacts', (req, res, params) => {
108
+ const url = new URL(req.url, `http://${req.headers.host}`);
109
+ const stage = url.searchParams.get('stage') ?? undefined;
110
+ try {
111
+ json(res, ctx.tasks.getArtifacts(parseInt(params.id, 10), stage));
112
+ }
113
+ catch (err) {
114
+ if (err instanceof TasksError) {
115
+ json(res, { error: err.message }, err.statusCode);
116
+ }
117
+ else {
118
+ json(res, { error: 'Internal error' }, 500);
119
+ }
120
+ }
121
+ });
122
+ route('GET', '/api/tasks/:id/dependencies', (_req, res, params) => {
123
+ try {
124
+ json(res, ctx.tasks.getDependencies(parseInt(params.id, 10)));
125
+ }
126
+ catch (err) {
127
+ if (err instanceof TasksError) {
128
+ json(res, { error: err.message }, err.statusCode);
129
+ }
130
+ else {
131
+ json(res, { error: 'Internal error' }, 500);
132
+ }
133
+ }
134
+ });
135
+ route('GET', '/api/dependencies', (_req, res) => {
136
+ json(res, ctx.tasks.getAllDependencies());
137
+ });
138
+ route('GET', '/api/pipeline', (req, res) => {
139
+ const url = new URL(req.url, `http://${req.headers.host}`);
140
+ const project = url.searchParams.get('project') ?? undefined;
141
+ json(res, { stages: ctx.tasks.getPipelineStages(project) });
142
+ });
143
+ route('GET', '/api/overview', (_req, res) => {
144
+ json(res, {
145
+ tasks: ctx.tasks.list(),
146
+ dependencies: ctx.tasks.getAllDependencies(),
147
+ artifactCounts: ctx.tasks.getArtifactCounts(),
148
+ commentCounts: ctx.comments.countByTask(),
149
+ subtaskProgress: ctx.tasks.getAllSubtaskProgress(),
150
+ stages: ctx.tasks.getPipelineStages(),
151
+ });
152
+ });
153
+ route('POST', '/api/tasks', async (req, res) => {
154
+ try {
155
+ const body = await parseBody(req);
156
+ const task = ctx.tasks.create({
157
+ title: body.title,
158
+ description: body.description,
159
+ assign_to: body.assign_to,
160
+ stage: body.stage,
161
+ priority: body.priority,
162
+ project: body.project,
163
+ tags: body.tags,
164
+ parent_id: body.parent_id,
165
+ }, body.created_by || 'api');
166
+ json(res, task, 201);
167
+ }
168
+ catch (err) {
169
+ if (err instanceof TasksError) {
170
+ json(res, { error: err.message }, err.statusCode);
171
+ }
172
+ else {
173
+ json(res, { error: 'Internal error' }, 500);
174
+ }
175
+ }
176
+ });
177
+ route('PUT', '/api/tasks/:id/stage', async (req, res, params) => {
178
+ try {
179
+ const body = await parseBody(req);
180
+ const taskId = parseInt(params.id, 10);
181
+ const targetStage = body.stage;
182
+ const task = ctx.tasks.getById(taskId);
183
+ if (!task) {
184
+ json(res, { error: 'Task not found' }, 404);
185
+ return;
186
+ }
187
+ const stages = ctx.tasks.getPipelineStages(task.project ?? undefined);
188
+ const currentIdx = stages.indexOf(task.stage);
189
+ const targetIdx = stages.indexOf(targetStage);
190
+ if (targetIdx > currentIdx) {
191
+ json(res, ctx.tasks.advance(taskId, targetStage));
192
+ }
193
+ else if (targetIdx < currentIdx) {
194
+ json(res, ctx.tasks.regress(taskId, targetStage, body.reason));
195
+ }
196
+ else {
197
+ json(res, task);
198
+ }
199
+ }
200
+ catch (err) {
201
+ if (err instanceof TasksError) {
202
+ json(res, { error: err.message }, err.statusCode);
203
+ }
204
+ else {
205
+ json(res, { error: 'Internal error' }, 500);
206
+ }
207
+ }
208
+ });
209
+ route('GET', '/api/tasks/:id/subtasks', (_req, res, params) => {
210
+ try {
211
+ json(res, ctx.tasks.getSubtasks(parseInt(params.id, 10)));
212
+ }
213
+ catch (err) {
214
+ if (err instanceof TasksError) {
215
+ json(res, { error: err.message }, err.statusCode);
216
+ }
217
+ else {
218
+ json(res, { error: 'Internal error' }, 500);
219
+ }
220
+ }
221
+ });
222
+ route('GET', '/api/tasks/:id/comments', (_req, res, params) => {
223
+ try {
224
+ json(res, ctx.comments.list(parseInt(params.id, 10)));
225
+ }
226
+ catch (err) {
227
+ if (err instanceof TasksError) {
228
+ json(res, { error: err.message }, err.statusCode);
229
+ }
230
+ else {
231
+ json(res, { error: 'Internal error' }, 500);
232
+ }
233
+ }
234
+ });
235
+ route('POST', '/api/tasks/:id/comments', async (req, res, params) => {
236
+ try {
237
+ const body = await parseBody(req);
238
+ const comment = ctx.comments.add(parseInt(params.id, 10), body.agent_id || 'api', body.content, body.parent_comment_id);
239
+ json(res, comment, 201);
240
+ }
241
+ catch (err) {
242
+ if (err instanceof TasksError) {
243
+ json(res, { error: err.message }, err.statusCode);
244
+ }
245
+ else {
246
+ json(res, { error: 'Internal error' }, 500);
247
+ }
248
+ }
249
+ });
250
+ route('GET', '/api/search', (req, res) => {
251
+ const url = new URL(req.url, `http://${req.headers.host}`);
252
+ const query = url.searchParams.get('q') ?? '';
253
+ try {
254
+ json(res, ctx.tasks.search(query, {
255
+ project: url.searchParams.get('project') ?? undefined,
256
+ limit: url.searchParams.has('limit')
257
+ ? parseInt(url.searchParams.get('limit'), 10)
258
+ : undefined,
259
+ }));
260
+ }
261
+ catch (err) {
262
+ if (err instanceof TasksError) {
263
+ json(res, { error: err.message }, err.statusCode);
264
+ }
265
+ else {
266
+ json(res, { error: 'Internal error' }, 500);
267
+ }
268
+ }
269
+ });
270
+ // -----------------------------------------------------------------------
271
+ // Static file serving
272
+ // -----------------------------------------------------------------------
273
+ function serveStatic(req, res) {
274
+ let pathname = new URL(req.url, `http://${req.headers.host}`).pathname;
275
+ if (pathname === '/' || pathname === '')
276
+ pathname = '/index.html';
277
+ let decoded;
278
+ try {
279
+ decoded = decodeURIComponent(pathname);
280
+ }
281
+ catch {
282
+ res.writeHead(400);
283
+ res.end('Bad request');
284
+ return;
285
+ }
286
+ if (decoded.includes('\0') || decoded.includes('..')) {
287
+ res.writeHead(403);
288
+ res.end('Forbidden');
289
+ return;
290
+ }
291
+ const filePath = join(uiDir, decoded);
292
+ let realPath;
293
+ try {
294
+ realPath = realpathSync(filePath);
295
+ }
296
+ catch {
297
+ res.writeHead(404);
298
+ res.end('Not found');
299
+ return;
300
+ }
301
+ const realUiDir = realpathSync(uiDir);
302
+ if (!realPath.startsWith(realUiDir)) {
303
+ res.writeHead(403);
304
+ res.end('Forbidden');
305
+ return;
306
+ }
307
+ try {
308
+ const content = readFileSync(realPath);
309
+ const ext = extname(realPath);
310
+ const contentType = MIME_TYPES[ext] || 'application/octet-stream';
311
+ res.writeHead(200, {
312
+ 'Content-Type': contentType,
313
+ 'X-Content-Type-Options': 'nosniff',
314
+ });
315
+ res.end(content);
316
+ }
317
+ catch {
318
+ res.writeHead(500);
319
+ res.end('Internal error');
320
+ }
321
+ }
322
+ // -----------------------------------------------------------------------
323
+ // Request dispatcher
324
+ // -----------------------------------------------------------------------
325
+ return (req, res) => {
326
+ if (req.method === 'OPTIONS') {
327
+ res.writeHead(204, {
328
+ 'Access-Control-Allow-Origin': '*',
329
+ 'Access-Control-Allow-Methods': 'GET, POST, PUT, DELETE, OPTIONS',
330
+ 'Access-Control-Allow-Headers': 'Content-Type',
331
+ });
332
+ res.end();
333
+ return;
334
+ }
335
+ const url = new URL(req.url, `http://${req.headers.host}`);
336
+ const pathname = url.pathname;
337
+ for (const r of routes) {
338
+ if (r.method !== req.method)
339
+ continue;
340
+ const match = pathname.match(r.pattern);
341
+ if (!match)
342
+ continue;
343
+ const params = {};
344
+ r.paramNames.forEach((name, i) => {
345
+ params[name] = decodeURIComponent(match[i + 1]);
346
+ });
347
+ try {
348
+ const result = r.handler(req, res, params);
349
+ if (result instanceof Promise) {
350
+ result.catch((err) => {
351
+ if (!res.writableEnded) {
352
+ if (err instanceof TasksError) {
353
+ json(res, { error: err.message }, err.statusCode);
354
+ }
355
+ else {
356
+ json(res, { error: 'Internal error' }, 500);
357
+ }
358
+ }
359
+ });
360
+ }
361
+ }
362
+ catch (err) {
363
+ if (!res.writableEnded) {
364
+ if (err instanceof TasksError) {
365
+ json(res, { error: err.message }, err.statusCode);
366
+ }
367
+ else {
368
+ json(res, { error: 'Internal error' }, 500);
369
+ }
370
+ }
371
+ }
372
+ return;
373
+ }
374
+ if (req.method === 'GET' && existsSync(uiDir)) {
375
+ serveStatic(req, res);
376
+ }
377
+ else {
378
+ json(res, { error: 'Not found' }, 404);
379
+ }
380
+ };
381
+ }
382
+ //# sourceMappingURL=rest.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"rest.js","sourceRoot":"","sources":["../../src/transport/rest.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,10 @@
1
+ import { WebSocketServer } from 'ws';
2
+ import type { Server } from 'http';
3
+ import type { AppContext } from '../context.js';
4
+ export interface WebSocketHandle {
5
+ wss: WebSocketServer;
6
+ broadcast(message: string): void;
7
+ close(): void;
8
+ }
9
+ export declare function setupWebSocket(httpServer: Server, ctx: AppContext): WebSocketHandle;
10
+ //# sourceMappingURL=ws.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws.d.ts","sourceRoot":"","sources":["../../src/transport/ws.ts"],"names":[],"mappings":"AAOA,OAAO,EAAE,eAAe,EAAa,MAAM,IAAI,CAAC;AAChD,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,MAAM,CAAC;AAInC,OAAO,KAAK,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAShD,MAAM,WAAW,eAAe;IAC9B,GAAG,EAAE,eAAe,CAAC;IACrB,SAAS,CAAC,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;IACjC,KAAK,IAAI,IAAI,CAAC;CACf;AA+BD,wBAAgB,cAAc,CAAC,UAAU,EAAE,MAAM,EAAE,GAAG,EAAE,UAAU,GAAG,eAAe,CAwInF"}
@@ -0,0 +1,177 @@
1
+ // =============================================================================
2
+ // agent-tasks — WebSocket transport
3
+ //
4
+ // Real-time event streaming to connected UI clients.
5
+ // Full state sent on connect; individual events streamed after.
6
+ // =============================================================================
7
+ import { WebSocketServer, WebSocket } from 'ws';
8
+ import { readFileSync } from 'fs';
9
+ import { join, dirname } from 'path';
10
+ import { fileURLToPath } from 'url';
11
+ const __dirname_ws = dirname(fileURLToPath(import.meta.url));
12
+ const MAX_WS_MESSAGE_SIZE = 4096;
13
+ const MAX_WS_CONNECTIONS = 50;
14
+ const PING_INTERVAL_MS = 30_000;
15
+ const VALID_EVENT_TYPES = new Set([
16
+ '*',
17
+ 'task:created',
18
+ 'task:updated',
19
+ 'task:claimed',
20
+ 'task:advanced',
21
+ 'task:regressed',
22
+ 'task:completed',
23
+ 'task:failed',
24
+ 'task:cancelled',
25
+ 'task:deleted',
26
+ 'artifact:created',
27
+ 'dependency:added',
28
+ 'dependency:removed',
29
+ 'pipeline:configured',
30
+ 'comment:created',
31
+ 'collaborator:added',
32
+ 'collaborator:removed',
33
+ 'approval:requested',
34
+ 'approval:approved',
35
+ 'approval:rejected',
36
+ ]);
37
+ export function setupWebSocket(httpServer, ctx) {
38
+ const wss = new WebSocketServer({ server: httpServer, maxPayload: MAX_WS_MESSAGE_SIZE });
39
+ const clients = new Map();
40
+ wss.on('connection', (ws) => {
41
+ if (wss.clients.size > MAX_WS_CONNECTIONS) {
42
+ ws.close(1013, 'Too many connections');
43
+ return;
44
+ }
45
+ const state = {
46
+ alive: true,
47
+ subscribedEvents: new Set(),
48
+ unsub: ctx.events.on('*', (event) => {
49
+ if (ws.readyState !== WebSocket.OPEN)
50
+ return;
51
+ if (state.subscribedEvents.size > 0) {
52
+ if (!state.subscribedEvents.has('*') && !state.subscribedEvents.has(event.type)) {
53
+ return;
54
+ }
55
+ }
56
+ ws.send(JSON.stringify(event));
57
+ }),
58
+ };
59
+ clients.set(ws, state);
60
+ sendFullState(ws, ctx);
61
+ ws.on('pong', () => {
62
+ state.alive = true;
63
+ });
64
+ ws.on('message', (raw) => {
65
+ let parsed;
66
+ try {
67
+ parsed = JSON.parse(raw.toString());
68
+ }
69
+ catch {
70
+ ws.send(JSON.stringify({ type: 'error', message: 'Invalid JSON' }));
71
+ return;
72
+ }
73
+ if (typeof parsed !== 'object' || parsed === null || Array.isArray(parsed)) {
74
+ ws.send(JSON.stringify({ type: 'error', message: 'Message must be a JSON object' }));
75
+ return;
76
+ }
77
+ const msg = parsed;
78
+ if (typeof msg.type !== 'string') {
79
+ ws.send(JSON.stringify({ type: 'error', message: 'Missing type field' }));
80
+ return;
81
+ }
82
+ switch (msg.type) {
83
+ case 'refresh':
84
+ sendFullState(ws, ctx);
85
+ break;
86
+ case 'subscribe': {
87
+ const events = msg.events;
88
+ if (!Array.isArray(events)) {
89
+ ws.send(JSON.stringify({
90
+ type: 'error',
91
+ message: '"events" must be an array of event type strings',
92
+ }));
93
+ break;
94
+ }
95
+ state.subscribedEvents.clear();
96
+ for (const e of events) {
97
+ if (typeof e === 'string' && VALID_EVENT_TYPES.has(e)) {
98
+ state.subscribedEvents.add(e);
99
+ }
100
+ }
101
+ ws.send(JSON.stringify({ type: 'subscribed', events: [...state.subscribedEvents] }));
102
+ break;
103
+ }
104
+ default: {
105
+ const safeType = String(msg.type)
106
+ .slice(0, 64)
107
+ .replace(/[<>&"']/g, '');
108
+ ws.send(JSON.stringify({ type: 'error', message: `Unknown message type: ${safeType}` }));
109
+ }
110
+ }
111
+ });
112
+ ws.on('error', () => {
113
+ const s = clients.get(ws);
114
+ if (s) {
115
+ s.unsub();
116
+ clients.delete(ws);
117
+ }
118
+ });
119
+ ws.on('close', () => {
120
+ const s = clients.get(ws);
121
+ if (s) {
122
+ s.unsub();
123
+ clients.delete(ws);
124
+ }
125
+ });
126
+ });
127
+ const pingInterval = setInterval(() => {
128
+ for (const [ws, state] of clients) {
129
+ if (!state.alive) {
130
+ ws.terminate();
131
+ clients.delete(ws);
132
+ continue;
133
+ }
134
+ state.alive = false;
135
+ ws.ping();
136
+ }
137
+ }, PING_INTERVAL_MS);
138
+ pingInterval.unref();
139
+ return {
140
+ wss,
141
+ broadcast(message) {
142
+ for (const [ws] of clients) {
143
+ if (ws.readyState === WebSocket.OPEN) {
144
+ ws.send(message);
145
+ }
146
+ }
147
+ },
148
+ close() {
149
+ clearInterval(pingInterval);
150
+ for (const [ws, state] of clients) {
151
+ state.unsub();
152
+ ws.close(1001, 'Server shutting down');
153
+ }
154
+ clients.clear();
155
+ wss.close();
156
+ },
157
+ };
158
+ }
159
+ function sendFullState(ws, ctx) {
160
+ const pkg = JSON.parse(readFileSync(join(__dirname_ws, '..', '..', 'package.json'), 'utf8'));
161
+ try {
162
+ ws.send(JSON.stringify({
163
+ type: 'state',
164
+ version: pkg.version,
165
+ tasks: ctx.tasks.list(),
166
+ dependencies: ctx.tasks.getAllDependencies(),
167
+ artifactCounts: ctx.tasks.getArtifactCounts(),
168
+ commentCounts: ctx.comments.countByTask(),
169
+ subtaskProgress: ctx.tasks.getAllSubtaskProgress(),
170
+ stages: ctx.tasks.getPipelineStages(),
171
+ }));
172
+ }
173
+ catch {
174
+ /* ignore send errors on closed sockets */
175
+ }
176
+ }
177
+ //# sourceMappingURL=ws.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ws.js","sourceRoot":"","sources":["../../src/transport/ws.ts"],"names":[],"mappings":"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"}