hzl-web 1.8.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.
@@ -0,0 +1,2 @@
1
+ export { createWebServer, type ServerOptions, type ServerHandle } from './server.js';
2
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,KAAK,aAAa,EAAE,KAAK,YAAY,EAAE,MAAM,aAAa,CAAC"}
package/dist/index.js ADDED
@@ -0,0 +1,2 @@
1
+ export { createWebServer } from './server.js';
2
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAyC,MAAM,aAAa,CAAC"}
@@ -0,0 +1,15 @@
1
+ import { TaskService, EventStore } from 'hzl-core';
2
+ export interface ServerOptions {
3
+ port: number;
4
+ host?: string;
5
+ taskService: TaskService;
6
+ eventStore: EventStore;
7
+ }
8
+ export interface ServerHandle {
9
+ close: () => Promise<void>;
10
+ port: number;
11
+ host: string;
12
+ url: string;
13
+ }
14
+ export declare function createWebServer(options: ServerOptions): ServerHandle;
15
+ //# sourceMappingURL=server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AACA,OAAO,EACL,WAAW,EACX,UAAU,EAEX,MAAM,UAAU,CAAC;AAGlB,MAAM,WAAW,aAAa;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,EAAE,WAAW,CAAC;IACzB,UAAU,EAAE,UAAU,CAAC;CACxB;AAED,MAAM,WAAW,YAAY;IAC3B,KAAK,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,GAAG,EAAE,MAAM,CAAC;CACb;AAmFD,wBAAgB,eAAe,CAAC,OAAO,EAAE,aAAa,GAAG,YAAY,CAoMpE"}
package/dist/server.js ADDED
@@ -0,0 +1,201 @@
1
+ import { createServer } from 'http';
2
+ import { DASHBOARD_HTML } from './ui-embed.js';
3
+ // Date filter presets in days
4
+ const DATE_PRESETS = {
5
+ '1d': 1,
6
+ '3d': 3,
7
+ '7d': 7,
8
+ '14d': 14,
9
+ '30d': 30,
10
+ };
11
+ function parseUrl(url) {
12
+ const idx = url.indexOf('?');
13
+ if (idx === -1) {
14
+ return { pathname: url, params: new URLSearchParams() };
15
+ }
16
+ return {
17
+ pathname: url.slice(0, idx),
18
+ params: new URLSearchParams(url.slice(idx + 1)),
19
+ };
20
+ }
21
+ function json(res, data, status = 200) {
22
+ res.writeHead(status, {
23
+ 'Content-Type': 'application/json',
24
+ // No CORS header - dashboard is served from same origin
25
+ });
26
+ res.end(JSON.stringify(data));
27
+ }
28
+ function notFound(res, message = 'Not Found') {
29
+ json(res, { error: message }, 404);
30
+ }
31
+ function serverError(res, error) {
32
+ const message = error instanceof Error ? error.message : 'Internal Server Error';
33
+ json(res, { error: message }, 500);
34
+ }
35
+ export function createWebServer(options) {
36
+ const { port, host = '0.0.0.0', taskService, eventStore } = options;
37
+ // Route handlers
38
+ function handleTasks(params, res) {
39
+ const since = params.get('since') || '3d';
40
+ const project = params.get('project');
41
+ const days = DATE_PRESETS[since] ?? 3;
42
+ // Get tasks from service
43
+ const rows = taskService.listTasks({
44
+ sinceDays: days,
45
+ project: project ?? undefined,
46
+ });
47
+ // Get blocked tasks map from service
48
+ const blockedMap = taskService.getBlockedByMap();
49
+ // Merge blocked info into tasks
50
+ const tasks = rows.map((row) => ({
51
+ ...row,
52
+ blocked_by: blockedMap.get(row.task_id) ?? null,
53
+ }));
54
+ json(res, { tasks, since, project });
55
+ }
56
+ function handleTaskDetail(taskId, res) {
57
+ const task = taskService.getTaskById(taskId);
58
+ if (!task) {
59
+ notFound(res, `Task not found: ${taskId}`);
60
+ return;
61
+ }
62
+ // Get blocking dependencies from service
63
+ const blocked_by = taskService.getBlockingDependencies(taskId);
64
+ const taskDetail = {
65
+ task_id: task.task_id,
66
+ title: task.title,
67
+ project: task.project,
68
+ status: task.status,
69
+ priority: task.priority,
70
+ parent_id: task.parent_id,
71
+ description: task.description,
72
+ links: task.links,
73
+ tags: task.tags,
74
+ due_at: task.due_at,
75
+ metadata: task.metadata,
76
+ claimed_at: task.claimed_at,
77
+ claimed_by_author: task.claimed_by_author,
78
+ claimed_by_agent_id: task.claimed_by_agent_id,
79
+ lease_until: task.lease_until,
80
+ created_at: task.created_at,
81
+ updated_at: task.updated_at,
82
+ blocked_by,
83
+ };
84
+ json(res, { task: taskDetail });
85
+ }
86
+ function handleComments(taskId, res) {
87
+ const comments = taskService.getComments(taskId);
88
+ // Map to response format (convert undefined to null for JSON)
89
+ const response = comments.map((c) => ({
90
+ event_rowid: c.event_rowid,
91
+ task_id: c.task_id,
92
+ author: c.author ?? null,
93
+ agent_id: c.agent_id ?? null,
94
+ text: c.text,
95
+ timestamp: c.timestamp,
96
+ }));
97
+ json(res, { comments: response });
98
+ }
99
+ function handleCheckpoints(taskId, res) {
100
+ const checkpoints = taskService.getCheckpoints(taskId);
101
+ json(res, { checkpoints });
102
+ }
103
+ function handleEvents(params, res) {
104
+ const sinceId = parseInt(params.get('since') || '0', 10);
105
+ // Get events from EventStore
106
+ const rawEvents = eventStore.getRecentEvents({ sinceId, limit: 50 });
107
+ // Get task titles for these events (batched query to avoid N+1)
108
+ const taskIds = [...new Set(rawEvents.map((e) => e.task_id).filter(Boolean))];
109
+ const titleMap = taskService.getTaskTitlesByIds(taskIds);
110
+ const events = rawEvents.map((e) => ({
111
+ id: e.rowid,
112
+ event_id: e.event_id,
113
+ task_id: e.task_id,
114
+ type: e.type,
115
+ data: e.data,
116
+ author: e.author ?? null,
117
+ agent_id: e.agent_id ?? null,
118
+ timestamp: e.timestamp,
119
+ task_title: titleMap.get(e.task_id) ?? null,
120
+ }));
121
+ json(res, { events });
122
+ }
123
+ function handleStats(res) {
124
+ const stats = taskService.getStats();
125
+ const response = {
126
+ total: stats.total,
127
+ by_status: stats.byStatus,
128
+ projects: stats.projects,
129
+ };
130
+ json(res, response);
131
+ }
132
+ function handleRoot(res) {
133
+ res.writeHead(200, {
134
+ 'Content-Type': 'text/html; charset=utf-8',
135
+ 'Content-Security-Policy': "default-src 'self'; script-src 'unsafe-inline'; style-src 'unsafe-inline'",
136
+ 'X-Content-Type-Options': 'nosniff',
137
+ 'X-Frame-Options': 'DENY',
138
+ 'Referrer-Policy': 'no-referrer',
139
+ });
140
+ res.end(DASHBOARD_HTML);
141
+ }
142
+ // Request handler
143
+ function handleRequest(req, res) {
144
+ const { pathname, params } = parseUrl(req.url || '/');
145
+ try {
146
+ // Route matching
147
+ if (pathname === '/') {
148
+ handleRoot(res);
149
+ return;
150
+ }
151
+ if (pathname === '/api/tasks') {
152
+ handleTasks(params, res);
153
+ return;
154
+ }
155
+ if (pathname === '/api/events') {
156
+ handleEvents(params, res);
157
+ return;
158
+ }
159
+ if (pathname === '/api/stats') {
160
+ handleStats(res);
161
+ return;
162
+ }
163
+ // /api/tasks/:id routes
164
+ const taskMatch = pathname.match(/^\/api\/tasks\/([^/]+)$/);
165
+ if (taskMatch) {
166
+ handleTaskDetail(taskMatch[1], res);
167
+ return;
168
+ }
169
+ const commentsMatch = pathname.match(/^\/api\/tasks\/([^/]+)\/comments$/);
170
+ if (commentsMatch) {
171
+ handleComments(commentsMatch[1], res);
172
+ return;
173
+ }
174
+ const checkpointsMatch = pathname.match(/^\/api\/tasks\/([^/]+)\/checkpoints$/);
175
+ if (checkpointsMatch) {
176
+ handleCheckpoints(checkpointsMatch[1], res);
177
+ return;
178
+ }
179
+ notFound(res);
180
+ }
181
+ catch (error) {
182
+ serverError(res, error);
183
+ }
184
+ }
185
+ const server = createServer(handleRequest);
186
+ server.listen(port, host);
187
+ return {
188
+ close: () => new Promise((resolve, reject) => {
189
+ server.close((err) => {
190
+ if (err)
191
+ reject(err);
192
+ else
193
+ resolve();
194
+ });
195
+ }),
196
+ port,
197
+ host,
198
+ url: `http://${host === '0.0.0.0' ? 'localhost' : host}:${port}`,
199
+ };
200
+ }
201
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=server.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.d.ts","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,297 @@
1
+ // packages/hzl-web/src/server.test.ts
2
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
3
+ import { createWebServer } from './server.js';
4
+ import { createTestDb } from 'hzl-core/db/test-utils';
5
+ import { EventStore } from 'hzl-core/events/store';
6
+ import { ProjectionEngine } from 'hzl-core/projections/engine';
7
+ import { TasksCurrentProjector } from 'hzl-core/projections/tasks-current';
8
+ import { DependenciesProjector } from 'hzl-core/projections/dependencies';
9
+ import { CommentsCheckpointsProjector } from 'hzl-core/projections/comments-checkpoints';
10
+ import { ProjectsProjector } from 'hzl-core/projections/projects';
11
+ import { TaskService } from 'hzl-core/services/task-service';
12
+ import { ProjectService } from 'hzl-core/services/project-service';
13
+ import { TaskStatus } from 'hzl-core/events/types';
14
+ describe('hzl-web server', () => {
15
+ let db;
16
+ let eventStore;
17
+ let projectionEngine;
18
+ let taskService;
19
+ let projectService;
20
+ let server;
21
+ beforeEach(() => {
22
+ db = createTestDb();
23
+ eventStore = new EventStore(db);
24
+ projectionEngine = new ProjectionEngine(db);
25
+ projectionEngine.register(new TasksCurrentProjector());
26
+ projectionEngine.register(new DependenciesProjector());
27
+ projectionEngine.register(new CommentsCheckpointsProjector());
28
+ projectionEngine.register(new ProjectsProjector());
29
+ projectService = new ProjectService(db, eventStore, projectionEngine);
30
+ projectService.ensureInboxExists();
31
+ projectService.createProject('test-project');
32
+ taskService = new TaskService(db, eventStore, projectionEngine, projectService);
33
+ });
34
+ afterEach(async () => {
35
+ if (server) {
36
+ await server.close();
37
+ }
38
+ db.close();
39
+ });
40
+ function createServer(port, host = '127.0.0.1') {
41
+ server = createWebServer({ port, host, taskService, eventStore });
42
+ return server;
43
+ }
44
+ async function fetchJson(path) {
45
+ const res = await globalThis.fetch(`${server.url}${path}`);
46
+ const data = await res.json().catch(() => null);
47
+ return { status: res.status, data };
48
+ }
49
+ describe('server configuration', () => {
50
+ it('starts on specified port', async () => {
51
+ const s = createServer(4500);
52
+ await new Promise((r) => setTimeout(r, 20));
53
+ expect(s.port).toBe(4500);
54
+ expect(s.url).toContain('4500');
55
+ });
56
+ it('binds to specified host', async () => {
57
+ const s = createServer(4501, '127.0.0.1');
58
+ await new Promise((r) => setTimeout(r, 20));
59
+ expect(s.host).toBe('127.0.0.1');
60
+ });
61
+ it('defaults host to localhost in URL when using 0.0.0.0', async () => {
62
+ const s = createServer(4502, '0.0.0.0');
63
+ await new Promise((r) => setTimeout(r, 20));
64
+ expect(s.url).toBe('http://localhost:4502');
65
+ });
66
+ it('closes gracefully', async () => {
67
+ const s = createServer(4503);
68
+ // Wait until server responds to prove it's ready (retry on connection refused)
69
+ for (let i = 0; i < 20; i++) {
70
+ try {
71
+ await globalThis.fetch(s.url);
72
+ break;
73
+ }
74
+ catch {
75
+ await new Promise((r) => setTimeout(r, 10));
76
+ }
77
+ }
78
+ await expect(s.close()).resolves.not.toThrow();
79
+ // Prevent afterEach from closing again
80
+ server = undefined;
81
+ });
82
+ });
83
+ describe('GET /api/tasks', () => {
84
+ it('returns empty task list initially', async () => {
85
+ createServer(4510);
86
+ const { status, data } = await fetchJson('/api/tasks');
87
+ expect(status).toBe(200);
88
+ expect(data).toMatchObject({
89
+ tasks: [],
90
+ since: '3d',
91
+ project: null,
92
+ });
93
+ });
94
+ it('returns tasks after creation', async () => {
95
+ taskService.createTask({
96
+ title: 'Test Task',
97
+ project: 'test-project',
98
+ priority: 3,
99
+ });
100
+ createServer(4511);
101
+ const { status, data } = await fetchJson('/api/tasks');
102
+ expect(status).toBe(200);
103
+ const tasks = data.tasks;
104
+ expect(tasks).toHaveLength(1);
105
+ expect(tasks[0]).toMatchObject({
106
+ title: 'Test Task',
107
+ project: 'test-project',
108
+ });
109
+ });
110
+ it('filters by project', async () => {
111
+ projectService.createProject('other-project');
112
+ taskService.createTask({ title: 'Task 1', project: 'test-project' });
113
+ taskService.createTask({ title: 'Task 2', project: 'other-project' });
114
+ createServer(4512);
115
+ const { data } = await fetchJson('/api/tasks?project=test-project');
116
+ const tasks = data.tasks;
117
+ expect(tasks).toHaveLength(1);
118
+ expect(tasks[0].title).toBe('Task 1');
119
+ });
120
+ it('respects since parameter', async () => {
121
+ createServer(4513);
122
+ const { data } = await fetchJson('/api/tasks?since=7d');
123
+ expect(data.since).toBe('7d');
124
+ });
125
+ it('marks blocked tasks', async () => {
126
+ // Blocker task stays in backlog (not done)
127
+ const blocker = taskService.createTask({
128
+ title: 'Blocker',
129
+ project: 'test-project',
130
+ });
131
+ // Blocked task depends on blocker and is set to ready
132
+ const blocked = taskService.createTask({
133
+ title: 'Blocked Task',
134
+ project: 'test-project',
135
+ depends_on: [blocker.task_id],
136
+ });
137
+ // Set blocked task to ready so it appears as "blocked" in the UI
138
+ taskService.setStatus(blocked.task_id, TaskStatus.Ready);
139
+ createServer(4514);
140
+ const { data } = await fetchJson('/api/tasks');
141
+ const tasks = data.tasks;
142
+ const blockedTask = tasks.find((t) => t.title === 'Blocked Task');
143
+ expect(blockedTask?.blocked_by).toContain(blocker.task_id);
144
+ });
145
+ });
146
+ describe('GET /api/tasks/:id', () => {
147
+ it('returns task detail', async () => {
148
+ const task = taskService.createTask({
149
+ title: 'Detailed Task',
150
+ project: 'test-project',
151
+ priority: 3,
152
+ description: 'A detailed description',
153
+ });
154
+ createServer(4520);
155
+ const { status, data } = await fetchJson(`/api/tasks/${task.task_id}`);
156
+ expect(status).toBe(200);
157
+ expect(data.task).toMatchObject({
158
+ task_id: task.task_id,
159
+ title: 'Detailed Task',
160
+ description: 'A detailed description',
161
+ priority: 3,
162
+ });
163
+ });
164
+ it('returns 404 for nonexistent task', async () => {
165
+ createServer(4521);
166
+ const { status, data } = await fetchJson('/api/tasks/nonexistent');
167
+ expect(status).toBe(404);
168
+ expect(data.error).toContain('not found');
169
+ });
170
+ });
171
+ describe('GET /api/tasks/:id/comments', () => {
172
+ it('returns task comments', async () => {
173
+ const task = taskService.createTask({
174
+ title: 'Task with comments',
175
+ project: 'test-project',
176
+ });
177
+ taskService.addComment(task.task_id, 'First comment', { author: 'tester' });
178
+ createServer(4530);
179
+ const { status, data } = await fetchJson(`/api/tasks/${task.task_id}/comments`);
180
+ expect(status).toBe(200);
181
+ const comments = data.comments;
182
+ expect(comments).toHaveLength(1);
183
+ expect(comments[0].text).toBe('First comment');
184
+ });
185
+ it('returns empty array for task with no comments', async () => {
186
+ const task = taskService.createTask({
187
+ title: 'Task without comments',
188
+ project: 'test-project',
189
+ });
190
+ createServer(4531);
191
+ const { status, data } = await fetchJson(`/api/tasks/${task.task_id}/comments`);
192
+ expect(status).toBe(200);
193
+ expect(data.comments).toHaveLength(0);
194
+ });
195
+ });
196
+ describe('GET /api/tasks/:id/checkpoints', () => {
197
+ it('returns task checkpoints', async () => {
198
+ const task = taskService.createTask({
199
+ title: 'Task with checkpoints',
200
+ project: 'test-project',
201
+ });
202
+ // Task must be ready before it can be claimed
203
+ taskService.setStatus(task.task_id, TaskStatus.Ready);
204
+ taskService.claimTask(task.task_id, { author: 'worker' });
205
+ taskService.addCheckpoint(task.task_id, 'progress-1', { progress: 50 });
206
+ createServer(4535);
207
+ const { status, data } = await fetchJson(`/api/tasks/${task.task_id}/checkpoints`);
208
+ expect(status).toBe(200);
209
+ const checkpoints = data.checkpoints;
210
+ expect(checkpoints).toHaveLength(1);
211
+ expect(checkpoints[0].name).toBe('progress-1');
212
+ });
213
+ });
214
+ describe('GET /api/stats', () => {
215
+ it('returns task statistics', async () => {
216
+ taskService.createTask({ title: 'Task 1', project: 'test-project' });
217
+ taskService.createTask({ title: 'Task 2', project: 'test-project' });
218
+ createServer(4540);
219
+ const { status, data } = await fetchJson('/api/stats');
220
+ expect(status).toBe(200);
221
+ expect(data).toMatchObject({
222
+ total: 2,
223
+ projects: expect.arrayContaining(['test-project']),
224
+ });
225
+ });
226
+ it('groups by status', async () => {
227
+ const task = taskService.createTask({ title: 'Ready Task', project: 'test-project' });
228
+ taskService.setStatus(task.task_id, TaskStatus.Ready);
229
+ createServer(4541);
230
+ const { data } = await fetchJson('/api/stats');
231
+ expect(data.by_status.ready).toBe(1);
232
+ });
233
+ });
234
+ describe('GET /api/events', () => {
235
+ it('returns recent events', async () => {
236
+ taskService.createTask({ title: 'New Task', project: 'test-project' });
237
+ createServer(4550);
238
+ const { status, data } = await fetchJson('/api/events?since=0');
239
+ expect(status).toBe(200);
240
+ const events = data.events;
241
+ expect(events.length).toBeGreaterThan(0);
242
+ expect(events.some((e) => e.type === 'task_created')).toBe(true);
243
+ });
244
+ it('filters events by since parameter', async () => {
245
+ taskService.createTask({ title: 'Task', project: 'test-project' });
246
+ createServer(4551);
247
+ // Get events since a high ID (should return nothing)
248
+ const { data } = await fetchJson('/api/events?since=99999');
249
+ expect(data.events).toHaveLength(0);
250
+ });
251
+ it('includes task titles in events', async () => {
252
+ taskService.createTask({ title: 'Named Task', project: 'test-project' });
253
+ createServer(4552);
254
+ const { data } = await fetchJson('/api/events?since=0');
255
+ const events = data.events;
256
+ const createEvent = events.find((e) => e.type === 'task_created');
257
+ expect(createEvent?.task_title).toBe('Named Task');
258
+ });
259
+ });
260
+ describe('GET / (dashboard HTML)', () => {
261
+ it('serves HTML at root', async () => {
262
+ server = createServer(4560);
263
+ const res = await globalThis.fetch(server.url);
264
+ expect(res.status).toBe(200);
265
+ expect(res.headers.get('content-type')).toContain('text/html');
266
+ });
267
+ it('includes security headers', async () => {
268
+ server = createServer(4561);
269
+ const res = await globalThis.fetch(server.url);
270
+ expect(res.headers.get('x-frame-options')).toBe('DENY');
271
+ expect(res.headers.get('x-content-type-options')).toBe('nosniff');
272
+ expect(res.headers.get('content-security-policy')).toBeTruthy();
273
+ expect(res.headers.get('referrer-policy')).toBe('no-referrer');
274
+ });
275
+ });
276
+ describe('404 handling', () => {
277
+ it('returns 404 for unknown routes', async () => {
278
+ createServer(4570);
279
+ const { status } = await fetchJson('/api/unknown');
280
+ expect(status).toBe(404);
281
+ });
282
+ it('returns JSON error for 404', async () => {
283
+ createServer(4571);
284
+ const { data } = await fetchJson('/api/unknown');
285
+ expect(data.error).toBe('Not Found');
286
+ });
287
+ });
288
+ describe('JSON API response format', () => {
289
+ it('does not include CORS headers', async () => {
290
+ createServer(4580);
291
+ const res = await globalThis.fetch(`${server.url}/api/tasks`);
292
+ // CORS header should NOT be present (removed for security)
293
+ expect(res.headers.get('access-control-allow-origin')).toBeNull();
294
+ });
295
+ });
296
+ });
297
+ //# sourceMappingURL=server.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.test.js","sourceRoot":"","sources":["../src/server.test.ts"],"names":[],"mappings":"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"}