reviewflow 3.16.0 → 3.17.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 (92) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/dist/dashboard/index.html +132 -0
  3. package/dist/dashboard/modules/i18n.d.ts.map +1 -1
  4. package/dist/dashboard/modules/i18n.js +30 -0
  5. package/dist/dashboard/modules/i18n.js.map +1 -1
  6. package/dist/dashboard/modules/worktreePanel.d.ts +111 -0
  7. package/dist/dashboard/modules/worktreePanel.d.ts.map +1 -0
  8. package/dist/dashboard/modules/worktreePanel.js +373 -0
  9. package/dist/dashboard/modules/worktreePanel.js.map +1 -0
  10. package/dist/dashboard/styles.css +312 -0
  11. package/dist/dashboard/vendor/anime.esm.min.js +7 -0
  12. package/dist/frameworks/scheduler/worktreeSweepScheduler.d.ts +8 -2
  13. package/dist/frameworks/scheduler/worktreeSweepScheduler.d.ts.map +1 -1
  14. package/dist/frameworks/scheduler/worktreeSweepScheduler.js +38 -3
  15. package/dist/frameworks/scheduler/worktreeSweepScheduler.js.map +1 -1
  16. package/dist/main/dependencies.d.ts +6 -0
  17. package/dist/main/dependencies.d.ts.map +1 -1
  18. package/dist/main/dependencies.js +9 -0
  19. package/dist/main/dependencies.js.map +1 -1
  20. package/dist/main/routes.d.ts.map +1 -1
  21. package/dist/main/routes.js +7 -0
  22. package/dist/main/routes.js.map +1 -1
  23. package/dist/main/server.d.ts.map +1 -1
  24. package/dist/main/server.js +7 -2
  25. package/dist/main/server.js.map +1 -1
  26. package/dist/modules/review-execution/entities/progress/agentDefinition.type.d.ts.map +1 -1
  27. package/dist/modules/review-execution/entities/progress/agentDefinition.type.js +1 -0
  28. package/dist/modules/review-execution/entities/progress/agentDefinition.type.js.map +1 -1
  29. package/dist/modules/worktree-management/entities/sweep/lastSweepSummary.schema.d.ts +9 -0
  30. package/dist/modules/worktree-management/entities/sweep/lastSweepSummary.schema.d.ts.map +1 -0
  31. package/dist/modules/worktree-management/entities/sweep/lastSweepSummary.schema.js +8 -0
  32. package/dist/modules/worktree-management/entities/sweep/lastSweepSummary.schema.js.map +1 -0
  33. package/dist/modules/worktree-management/entities/sweep/runSweepResult.d.ts +12 -0
  34. package/dist/modules/worktree-management/entities/sweep/runSweepResult.d.ts.map +1 -0
  35. package/dist/modules/worktree-management/entities/sweep/runSweepResult.js +2 -0
  36. package/dist/modules/worktree-management/entities/sweep/runSweepResult.js.map +1 -0
  37. package/dist/modules/worktree-management/entities/worktree/worktreeSizeProbe.gateway.d.ts +4 -0
  38. package/dist/modules/worktree-management/entities/worktree/worktreeSizeProbe.gateway.d.ts.map +1 -0
  39. package/dist/modules/worktree-management/entities/worktree/worktreeSizeProbe.gateway.js +2 -0
  40. package/dist/modules/worktree-management/entities/worktree/worktreeSizeProbe.gateway.js.map +1 -0
  41. package/dist/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.d.ts +19 -0
  42. package/dist/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.d.ts.map +1 -0
  43. package/dist/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.js +39 -0
  44. package/dist/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.js.map +1 -0
  45. package/dist/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.d.ts +17 -0
  46. package/dist/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.d.ts.map +1 -0
  47. package/dist/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.js +46 -0
  48. package/dist/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.js.map +1 -0
  49. package/dist/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.d.ts +53 -0
  50. package/dist/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.d.ts.map +1 -0
  51. package/dist/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.js +102 -0
  52. package/dist/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.js.map +1 -0
  53. package/dist/modules/worktree-management/usecases/sweepStaleWorktrees.usecase.d.ts +1 -1
  54. package/dist/modules/worktree-management/usecases/sweepStaleWorktrees.usecase.d.ts.map +1 -1
  55. package/dist/modules/worktree-management/usecases/sweepStaleWorktrees.usecase.js +1 -1
  56. package/dist/modules/worktree-management/usecases/sweepStaleWorktrees.usecase.js.map +1 -1
  57. package/dist/tests/acceptance/173-dashboard-worktree-panel.acceptance.test.d.ts +10 -0
  58. package/dist/tests/acceptance/173-dashboard-worktree-panel.acceptance.test.d.ts.map +1 -0
  59. package/dist/tests/acceptance/173-dashboard-worktree-panel.acceptance.test.js +192 -0
  60. package/dist/tests/acceptance/173-dashboard-worktree-panel.acceptance.test.js.map +1 -0
  61. package/dist/tests/factories/lastSweepSummary.factory.d.ts +5 -0
  62. package/dist/tests/factories/lastSweepSummary.factory.d.ts.map +1 -0
  63. package/dist/tests/factories/lastSweepSummary.factory.js +12 -0
  64. package/dist/tests/factories/lastSweepSummary.factory.js.map +1 -0
  65. package/dist/tests/stubs/worktreeSizeProbe.stub.d.ts +11 -0
  66. package/dist/tests/stubs/worktreeSizeProbe.stub.d.ts.map +1 -0
  67. package/dist/tests/stubs/worktreeSizeProbe.stub.js +22 -0
  68. package/dist/tests/stubs/worktreeSizeProbe.stub.js.map +1 -0
  69. package/dist/tests/units/dashboard/modules/worktreePanel.test.d.ts +2 -0
  70. package/dist/tests/units/dashboard/modules/worktreePanel.test.d.ts.map +1 -0
  71. package/dist/tests/units/dashboard/modules/worktreePanel.test.js +274 -0
  72. package/dist/tests/units/dashboard/modules/worktreePanel.test.js.map +1 -0
  73. package/dist/tests/units/frameworks/scheduler/worktreeSweepScheduler.test.js +117 -0
  74. package/dist/tests/units/frameworks/scheduler/worktreeSweepScheduler.test.js.map +1 -1
  75. package/dist/tests/units/modules/worktree-management/entities/sweep/lastSweepSummary.schema.test.d.ts +2 -0
  76. package/dist/tests/units/modules/worktree-management/entities/sweep/lastSweepSummary.schema.test.d.ts.map +1 -0
  77. package/dist/tests/units/modules/worktree-management/entities/sweep/lastSweepSummary.schema.test.js +44 -0
  78. package/dist/tests/units/modules/worktree-management/entities/sweep/lastSweepSummary.schema.test.js.map +1 -0
  79. package/dist/tests/units/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.test.d.ts +2 -0
  80. package/dist/tests/units/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.test.d.ts.map +1 -0
  81. package/dist/tests/units/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.test.js +160 -0
  82. package/dist/tests/units/modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.test.js.map +1 -0
  83. package/dist/tests/units/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.test.d.ts +2 -0
  84. package/dist/tests/units/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.test.d.ts.map +1 -0
  85. package/dist/tests/units/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.test.js +54 -0
  86. package/dist/tests/units/modules/worktree-management/interface-adapters/gateways/worktreeSizeProbe.cli.gateway.test.js.map +1 -0
  87. package/dist/tests/units/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.test.d.ts +2 -0
  88. package/dist/tests/units/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.test.d.ts.map +1 -0
  89. package/dist/tests/units/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.test.js +259 -0
  90. package/dist/tests/units/modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.test.js.map +1 -0
  91. package/package.json +2 -1
  92. package/scripts/copyAssets.mjs +8 -0
@@ -0,0 +1,192 @@
1
+ /**
2
+ * SPEC-173 — Dashboard Worktree Panel
3
+ *
4
+ * Outer-loop acceptance test (SDD): exercises the GET /api/worktrees and
5
+ * POST /api/worktrees/sweep endpoints end-to-end through a Fastify instance.
6
+ * No real scheduler timer, no real disk I/O, no real du — stubs only.
7
+ * Covers Scenarios 1, 2, 5, 6, 7 per docs/specs/173-dashboard-worktree-panel.md.
8
+ */
9
+ import { describe, it, expect } from 'vitest';
10
+ import Fastify from 'fastify';
11
+ import { worktreeOverviewRoutes } from '../../modules/worktree-management/interface-adapters/controllers/http/worktreeOverview.routes.js';
12
+ import { WorktreePanelPresenter } from '../../modules/worktree-management/interface-adapters/presenters/worktreePanel.presenter.js';
13
+ import { StubWorktreeSizeProbeGateway } from '../../tests/stubs/worktreeSizeProbe.stub.js';
14
+ import { LastSweepSummaryFactory } from '../../tests/factories/lastSweepSummary.factory.js';
15
+ import { createStubLogger } from '../../tests/stubs/logger.stub.js';
16
+ import { createWorktreePath } from '../../modules/worktree-management/entities/worktree/worktree.js';
17
+ import { startWorktreeSweepScheduler } from '../../frameworks/scheduler/worktreeSweepScheduler.js';
18
+ const NOW = new Date('2026-05-23T12:00:00.000Z');
19
+ const ONE_HOUR_MS = 60 * 60 * 1000;
20
+ const ONE_DAY_MS = 24 * ONE_HOUR_MS;
21
+ function buildEntry(identity, mtime, path) {
22
+ return {
23
+ identity,
24
+ path: createWorktreePath(path),
25
+ mtime,
26
+ };
27
+ }
28
+ class ConfigurableWorktreeGateway {
29
+ entries;
30
+ removed = [];
31
+ constructor(initial) {
32
+ this.entries = [...initial];
33
+ }
34
+ async list() {
35
+ return [...this.entries];
36
+ }
37
+ async ensure() {
38
+ return { status: 'failed', reason: 'not-implemented' };
39
+ }
40
+ async remove({ identity }) {
41
+ this.removed.push(identity);
42
+ this.entries = this.entries.filter(e => e.identity.platform !== identity.platform
43
+ || e.identity.projectPath !== identity.projectPath
44
+ || e.identity.mrNumber !== identity.mrNumber);
45
+ return { status: 'removed' };
46
+ }
47
+ async exists() {
48
+ return false;
49
+ }
50
+ }
51
+ async function buildAcceptanceApp(options) {
52
+ const app = Fastify();
53
+ const sizeProbe = new StubWorktreeSizeProbeGateway();
54
+ if (options.sizeMap) {
55
+ for (const [path, size] of Object.entries(options.sizeMap)) {
56
+ sizeProbe.setSize(path, size);
57
+ }
58
+ }
59
+ else {
60
+ sizeProbe.setDefault(100);
61
+ }
62
+ const presenter = new WorktreePanelPresenter({
63
+ sizeProbe,
64
+ cacheTtlMs: 30_000,
65
+ now: () => NOW,
66
+ });
67
+ let lastSweep = options.lastSweep ?? null;
68
+ const nextSweepAt = options.nextSweepAt ?? NOW;
69
+ const defaultRunSweep = async () => {
70
+ const summary = LastSweepSummaryFactory.create({ ranAt: NOW, removed: 4, failures: 0, scanned: 4 });
71
+ lastSweep = summary;
72
+ return { status: 'ok', summary };
73
+ };
74
+ await app.register(worktreeOverviewRoutes, {
75
+ worktreeGateway: new ConfigurableWorktreeGateway(options.worktrees),
76
+ presenter,
77
+ schedulerControls: {
78
+ getLastSweep: () => lastSweep,
79
+ getNextSweepEta: () => nextSweepAt,
80
+ runSweepNow: options.runSweepNow ?? defaultRunSweep,
81
+ },
82
+ logger: createStubLogger(),
83
+ });
84
+ return app;
85
+ }
86
+ describe('Acceptance — SPEC-173: Dashboard Worktree Panel', () => {
87
+ describe('Scenario 1 — list worktrees with active + idle + stale', () => {
88
+ it('returns 3 rows with statuses [active, idle, stale]', async () => {
89
+ const activeIdentity = { platform: 'gitlab', projectPath: 'group/a', mrNumber: 1 };
90
+ const idleIdentity = { platform: 'gitlab', projectPath: 'group/b', mrNumber: 2 };
91
+ const staleIdentity = { platform: 'gitlab', projectPath: 'group/c', mrNumber: 3 };
92
+ const activePath = '/tmp/worktrees/gitlab-group-a-1';
93
+ const idlePath = '/tmp/worktrees/gitlab-group-b-2';
94
+ const stalePath = '/tmp/worktrees/gitlab-group-c-3';
95
+ const app = await buildAcceptanceApp({
96
+ worktrees: [
97
+ buildEntry(activeIdentity, new Date(NOW.getTime() - ONE_HOUR_MS), activePath),
98
+ buildEntry(idleIdentity, new Date(NOW.getTime() - 36 * ONE_HOUR_MS), idlePath),
99
+ buildEntry(staleIdentity, new Date(NOW.getTime() - 8 * ONE_DAY_MS), stalePath),
100
+ ],
101
+ sizeMap: { [activePath]: 100, [idlePath]: 200, [stalePath]: 300 },
102
+ });
103
+ const response = await app.inject({ method: 'GET', url: '/api/worktrees' });
104
+ expect(response.statusCode).toBe(200);
105
+ const body = response.json();
106
+ expect(body.totalCount).toBe(3);
107
+ expect(body.totalSizeBytes).toBe(600);
108
+ const statuses = body.groups.flatMap(g => g.worktrees.map(w => w.status));
109
+ expect(statuses).toContain('active');
110
+ expect(statuses).toContain('idle');
111
+ expect(statuses).toContain('stale');
112
+ await app.close();
113
+ });
114
+ });
115
+ describe('Scenario 2 — empty worktree pool', () => {
116
+ it('renders empty groups and zero counters', async () => {
117
+ const app = await buildAcceptanceApp({ worktrees: [] });
118
+ const response = await app.inject({ method: 'GET', url: '/api/worktrees' });
119
+ expect(response.statusCode).toBe(200);
120
+ const body = response.json();
121
+ expect(body.totalCount).toBe(0);
122
+ expect(body.totalSizeBytes).toBe(0);
123
+ expect(body.groups).toEqual([]);
124
+ await app.close();
125
+ });
126
+ });
127
+ describe('Scenario 5 — manual sweep success', () => {
128
+ it('POST /api/worktrees/sweep returns 200 with the new summary', async () => {
129
+ const summary = LastSweepSummaryFactory.create({ removed: 4, failures: 0, scanned: 4 });
130
+ const app = await buildAcceptanceApp({
131
+ worktrees: [],
132
+ runSweepNow: async () => ({ status: 'ok', summary }),
133
+ });
134
+ const response = await app.inject({ method: 'POST', url: '/api/worktrees/sweep' });
135
+ expect(response.statusCode).toBe(200);
136
+ const body = response.json();
137
+ expect(body.removed).toBe(4);
138
+ expect(body.failures).toBe(0);
139
+ expect(body.scanned).toBe(4);
140
+ await app.close();
141
+ });
142
+ });
143
+ describe('Scenario 6 — manual sweep conflict', () => {
144
+ it('POST /api/worktrees/sweep returns 409 with startedAt when a sweep is already running', async () => {
145
+ const startedAt = new Date('2026-05-23T11:59:57.000Z');
146
+ const app = await buildAcceptanceApp({
147
+ worktrees: [],
148
+ runSweepNow: async () => ({ status: 'conflict', startedAt }),
149
+ });
150
+ const response = await app.inject({ method: 'POST', url: '/api/worktrees/sweep' });
151
+ expect(response.statusCode).toBe(409);
152
+ const body = response.json();
153
+ expect(body.error).toBe('sweep-in-progress');
154
+ expect(body.startedAt).toBe(startedAt.toISOString());
155
+ await app.close();
156
+ });
157
+ });
158
+ describe('Scenario 7 — lastSweep null on cold start', () => {
159
+ it('GET /api/worktrees returns lastSweep: null right after server start', async () => {
160
+ const app = await buildAcceptanceApp({ worktrees: [], lastSweep: null });
161
+ const response = await app.inject({ method: 'GET', url: '/api/worktrees' });
162
+ expect(response.statusCode).toBe(200);
163
+ const body = response.json();
164
+ expect(body.lastSweep).toBeNull();
165
+ await app.close();
166
+ });
167
+ });
168
+ describe('Scheduler integration (FR-3) — scheduler exposes controls consumed by the routes', () => {
169
+ it('startWorktreeSweepScheduler exposes getLastSweep, getNextSweepEta, runSweepNow', () => {
170
+ const stubGateway = {
171
+ list: async () => [],
172
+ remove: async () => ({ status: 'removed' }),
173
+ ensure: async () => ({ status: 'failed', reason: 'not-implemented' }),
174
+ exists: async () => false,
175
+ };
176
+ const trackingGateway = { getById: (_p, _id) => null };
177
+ const scheduler = startWorktreeSweepScheduler({
178
+ worktreeGateway: stubGateway,
179
+ trackingGateway,
180
+ getRepositories: () => [],
181
+ logger: createStubLogger(),
182
+ now: () => NOW,
183
+ });
184
+ expect(typeof scheduler.stop).toBe('function');
185
+ expect(typeof scheduler.getLastSweep).toBe('function');
186
+ expect(typeof scheduler.getNextSweepEta).toBe('function');
187
+ expect(typeof scheduler.runSweepNow).toBe('function');
188
+ scheduler.stop();
189
+ });
190
+ });
191
+ });
192
+ //# sourceMappingURL=173-dashboard-worktree-panel.acceptance.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"173-dashboard-worktree-panel.acceptance.test.js","sourceRoot":"","sources":["../../../src/tests/acceptance/173-dashboard-worktree-panel.acceptance.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,5 @@
1
+ import type { LastSweepSummary } from '../../modules/worktree-management/entities/sweep/lastSweepSummary.schema.js';
2
+ export declare class LastSweepSummaryFactory {
3
+ static create(overrides?: Partial<LastSweepSummary>): LastSweepSummary;
4
+ }
5
+ //# sourceMappingURL=lastSweepSummary.factory.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lastSweepSummary.factory.d.ts","sourceRoot":"","sources":["../../../src/tests/factories/lastSweepSummary.factory.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,yEAAyE,CAAC;AAEhH,qBAAa,uBAAuB;IAClC,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,OAAO,CAAC,gBAAgB,CAAC,GAAG,gBAAgB;CASvE"}
@@ -0,0 +1,12 @@
1
+ export class LastSweepSummaryFactory {
2
+ static create(overrides) {
3
+ return {
4
+ ranAt: new Date('2026-05-23T03:00:00.000Z'),
5
+ removed: 0,
6
+ failures: 0,
7
+ scanned: 0,
8
+ ...overrides,
9
+ };
10
+ }
11
+ }
12
+ //# sourceMappingURL=lastSweepSummary.factory.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"lastSweepSummary.factory.js","sourceRoot":"","sources":["../../../src/tests/factories/lastSweepSummary.factory.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,uBAAuB;IAClC,MAAM,CAAC,MAAM,CAAC,SAAqC;QACjD,OAAO;YACL,KAAK,EAAE,IAAI,IAAI,CAAC,0BAA0B,CAAC;YAC3C,OAAO,EAAE,CAAC;YACV,QAAQ,EAAE,CAAC;YACX,OAAO,EAAE,CAAC;YACV,GAAG,SAAS;SACb,CAAC;IACJ,CAAC;CACF"}
@@ -0,0 +1,11 @@
1
+ import type { WorktreeSizeProbeGateway } from '../../modules/worktree-management/entities/worktree/worktreeSizeProbe.gateway.js';
2
+ export declare class StubWorktreeSizeProbeGateway implements WorktreeSizeProbeGateway {
3
+ private readonly sizesByPath;
4
+ private defaultSize;
5
+ readonly calls: string[];
6
+ setSize(path: string, sizeBytes: number | null): void;
7
+ setDefault(sizeBytes: number | null): void;
8
+ probe(path: string): Promise<number | null>;
9
+ clearCalls(): void;
10
+ }
11
+ //# sourceMappingURL=worktreeSizeProbe.stub.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"worktreeSizeProbe.stub.d.ts","sourceRoot":"","sources":["../../../src/tests/stubs/worktreeSizeProbe.stub.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,8EAA8E,CAAC;AAE7H,qBAAa,4BAA6B,YAAW,wBAAwB;IAC3E,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAyC;IACrE,OAAO,CAAC,WAAW,CAAoB;IACvC,QAAQ,CAAC,KAAK,EAAE,MAAM,EAAE,CAAM;IAE9B,OAAO,CAAC,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI;IAIrD,UAAU,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI,GAAG,IAAI;IAIpC,KAAK,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;IAQjD,UAAU,IAAI,IAAI;CAGnB"}
@@ -0,0 +1,22 @@
1
+ export class StubWorktreeSizeProbeGateway {
2
+ sizesByPath = new Map();
3
+ defaultSize = 0;
4
+ calls = [];
5
+ setSize(path, sizeBytes) {
6
+ this.sizesByPath.set(path, sizeBytes);
7
+ }
8
+ setDefault(sizeBytes) {
9
+ this.defaultSize = sizeBytes;
10
+ }
11
+ async probe(path) {
12
+ this.calls.push(path);
13
+ if (this.sizesByPath.has(path)) {
14
+ return this.sizesByPath.get(path) ?? null;
15
+ }
16
+ return this.defaultSize;
17
+ }
18
+ clearCalls() {
19
+ this.calls.length = 0;
20
+ }
21
+ }
22
+ //# sourceMappingURL=worktreeSizeProbe.stub.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"worktreeSizeProbe.stub.js","sourceRoot":"","sources":["../../../src/tests/stubs/worktreeSizeProbe.stub.ts"],"names":[],"mappings":"AAEA,MAAM,OAAO,4BAA4B;IACtB,WAAW,GAA+B,IAAI,GAAG,EAAE,CAAC;IAC7D,WAAW,GAAkB,CAAC,CAAC;IAC9B,KAAK,GAAa,EAAE,CAAC;IAE9B,OAAO,CAAC,IAAY,EAAE,SAAwB;QAC5C,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;IACxC,CAAC;IAED,UAAU,CAAC,SAAwB;QACjC,IAAI,CAAC,WAAW,GAAG,SAAS,CAAC;IAC/B,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,IAAY;QACtB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QACtB,IAAI,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC;YAC/B,OAAO,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,IAAI,CAAC;QAC5C,CAAC;QACD,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED,UAAU;QACR,IAAI,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC;IACxB,CAAC;CACF"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=worktreePanel.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"worktreePanel.test.d.ts","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/worktreePanel.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,274 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { renderWorktreeSection, renderWorktreeEmptyState, renderWorktreeStatusBadge, fetchWorktreeOverview, triggerManualSweep, formatBytes, formatRelativeAge, snapshotTotals, computeChangedMetricKeys, } from '../../../../dashboard/modules/worktreePanel.js';
3
+ const NOW_ISO = '2026-05-23T12:00:00.000Z';
4
+ function buildViewModel(overrides = {}) {
5
+ return {
6
+ totalCount: 0,
7
+ totalSizeBytes: 0,
8
+ activeCount: 0,
9
+ idleCount: 0,
10
+ staleCount: 0,
11
+ nextSweepAt: '2026-05-24T03:00:00.000Z',
12
+ lastSweep: null,
13
+ groups: [],
14
+ ...overrides,
15
+ };
16
+ }
17
+ describe('renderWorktreeSection', () => {
18
+ it('includes the // WORKTREE POOL header prefix with the total count', () => {
19
+ const html = renderWorktreeSection(buildViewModel({ totalCount: 7 }));
20
+ expect(html).toContain('// WORKTREE POOL');
21
+ expect(html).toContain('7');
22
+ });
23
+ it('renders the empty state when totalCount is zero', () => {
24
+ const html = renderWorktreeSection(buildViewModel({ totalCount: 0 }));
25
+ expect(html).toContain('worktree-empty');
26
+ });
27
+ it('renders a row per worktree when groups are populated', () => {
28
+ const viewModel = buildViewModel({
29
+ totalCount: 2,
30
+ totalSizeBytes: 1024,
31
+ groups: [
32
+ {
33
+ platform: 'gitlab',
34
+ projectPath: 'group/project',
35
+ worktrees: [
36
+ {
37
+ mrNumber: 11,
38
+ path: '/tmp/worktrees/gitlab-group-project-11',
39
+ mtime: NOW_ISO,
40
+ ageSeconds: 60,
41
+ sizeBytes: 512,
42
+ status: 'active',
43
+ },
44
+ {
45
+ mrNumber: 12,
46
+ path: '/tmp/worktrees/gitlab-group-project-12',
47
+ mtime: '2026-05-22T12:00:00.000Z',
48
+ ageSeconds: 86_400,
49
+ sizeBytes: 512,
50
+ status: 'idle',
51
+ },
52
+ ],
53
+ },
54
+ ],
55
+ });
56
+ const html = renderWorktreeSection(viewModel);
57
+ expect(html).toContain('group/project');
58
+ expect(html).toContain('11');
59
+ expect(html).toContain('12');
60
+ expect(html).toContain('ACTIVE');
61
+ expect(html).toContain('IDLE');
62
+ });
63
+ it('renders the sweep button labelled SWEEP NOW', () => {
64
+ const html = renderWorktreeSection(buildViewModel());
65
+ expect(html).toContain('SWEEP NOW');
66
+ expect(html).toMatch(/data-action="sweep"/);
67
+ });
68
+ it('renders lastSweep summary when provided', () => {
69
+ const html = renderWorktreeSection(buildViewModel({
70
+ lastSweep: {
71
+ ranAt: '2026-05-23T03:00:00.000Z',
72
+ removed: 2,
73
+ failures: 0,
74
+ scanned: 9,
75
+ },
76
+ }));
77
+ expect(html).toContain('removed 2');
78
+ expect(html).toContain('failures 0');
79
+ expect(html).toContain('scanned 9');
80
+ });
81
+ it('renders "never" when lastSweep is null', () => {
82
+ const html = renderWorktreeSection(buildViewModel({ lastSweep: null }));
83
+ expect(html.toLowerCase()).toContain('never');
84
+ });
85
+ it('escapes HTML special characters in project path', () => {
86
+ const html = renderWorktreeSection(buildViewModel({
87
+ totalCount: 1,
88
+ groups: [
89
+ {
90
+ platform: 'gitlab',
91
+ projectPath: '<script>alert(1)</script>',
92
+ worktrees: [
93
+ {
94
+ mrNumber: 1,
95
+ path: '/tmp/x',
96
+ mtime: NOW_ISO,
97
+ ageSeconds: 0,
98
+ sizeBytes: null,
99
+ status: 'active',
100
+ },
101
+ ],
102
+ },
103
+ ],
104
+ }));
105
+ expect(html).not.toContain('<script>alert(1)</script>');
106
+ expect(html).toContain('&lt;script&gt;');
107
+ });
108
+ it('renders sizeBytes as "—" when null', () => {
109
+ const html = renderWorktreeSection(buildViewModel({
110
+ totalCount: 1,
111
+ groups: [
112
+ {
113
+ platform: 'gitlab',
114
+ projectPath: 'group/project',
115
+ worktrees: [
116
+ {
117
+ mrNumber: 1,
118
+ path: '/tmp/x',
119
+ mtime: NOW_ISO,
120
+ ageSeconds: 0,
121
+ sizeBytes: null,
122
+ status: 'active',
123
+ },
124
+ ],
125
+ },
126
+ ],
127
+ }));
128
+ expect(html).toContain('—');
129
+ });
130
+ });
131
+ describe('renderWorktreeEmptyState', () => {
132
+ it('returns HTML containing an inline SVG illustration', () => {
133
+ const html = renderWorktreeEmptyState();
134
+ expect(html).toContain('<svg');
135
+ expect(html).toContain('worktree-empty');
136
+ });
137
+ });
138
+ describe('renderWorktreeStatusBadge', () => {
139
+ it('renders an ACTIVE filled dot glyph', () => {
140
+ const html = renderWorktreeStatusBadge('active');
141
+ expect(html).toContain('ACTIVE');
142
+ expect(html).toContain('worktree-status-active');
143
+ });
144
+ it('renders an IDLE outline dot glyph', () => {
145
+ const html = renderWorktreeStatusBadge('idle');
146
+ expect(html).toContain('IDLE');
147
+ expect(html).toContain('worktree-status-idle');
148
+ });
149
+ it('renders a STALE diamond glyph', () => {
150
+ const html = renderWorktreeStatusBadge('stale');
151
+ expect(html).toContain('STALE');
152
+ expect(html).toContain('worktree-status-stale');
153
+ });
154
+ });
155
+ describe('formatBytes', () => {
156
+ it('formats null as a placeholder dash', () => {
157
+ expect(formatBytes(null)).toBe('—');
158
+ });
159
+ it('formats bytes under 1024 as B', () => {
160
+ expect(formatBytes(512)).toBe('512 B');
161
+ });
162
+ it('formats kilobytes', () => {
163
+ expect(formatBytes(2048)).toBe('2.0 KB');
164
+ });
165
+ it('formats megabytes with one decimal', () => {
166
+ expect(formatBytes(5 * 1024 * 1024)).toBe('5.0 MB');
167
+ });
168
+ it('formats gigabytes with two decimals', () => {
169
+ expect(formatBytes(1.34 * 1024 * 1024 * 1024)).toBe('1.34 GB');
170
+ });
171
+ });
172
+ describe('formatRelativeAge', () => {
173
+ it('formats seconds when under a minute', () => {
174
+ expect(formatRelativeAge(45)).toBe('45s');
175
+ });
176
+ it('formats minutes when under an hour', () => {
177
+ expect(formatRelativeAge(8 * 60)).toBe('8m');
178
+ });
179
+ it('formats hours when under a day', () => {
180
+ expect(formatRelativeAge(2 * 60 * 60)).toBe('2h');
181
+ });
182
+ it('formats days when over 24h', () => {
183
+ expect(formatRelativeAge(8 * 24 * 60 * 60)).toBe('8d');
184
+ });
185
+ });
186
+ describe('fetchWorktreeOverview', () => {
187
+ it('GETs /api/worktrees and returns the parsed JSON body', async () => {
188
+ const fakeBody = buildViewModel({ totalCount: 1 });
189
+ const fetchImpl = vi.fn().mockResolvedValue({
190
+ ok: true,
191
+ status: 200,
192
+ json: async () => fakeBody,
193
+ });
194
+ const result = await fetchWorktreeOverview(fetchImpl);
195
+ expect(fetchImpl).toHaveBeenCalledWith('/api/worktrees');
196
+ expect(result).toEqual(fakeBody);
197
+ });
198
+ it('throws when the response is not ok', async () => {
199
+ const fetchImpl = vi.fn().mockResolvedValue({
200
+ ok: false,
201
+ status: 503,
202
+ json: async () => ({}),
203
+ });
204
+ await expect(fetchWorktreeOverview(fetchImpl)).rejects.toThrow(/503/);
205
+ });
206
+ });
207
+ describe('triggerManualSweep', () => {
208
+ it('POSTs /api/worktrees/sweep and returns ok with payload on 200', async () => {
209
+ const fetchImpl = vi.fn().mockResolvedValue({
210
+ ok: true,
211
+ status: 200,
212
+ json: async () => ({ ranAt: NOW_ISO, removed: 2, failures: 0, scanned: 4 }),
213
+ });
214
+ const result = await triggerManualSweep(fetchImpl);
215
+ expect(fetchImpl).toHaveBeenCalledWith('/api/worktrees/sweep', { method: 'POST' });
216
+ expect(result.status).toBe('ok');
217
+ if (result.status === 'ok') {
218
+ expect(result.payload.removed).toBe(2);
219
+ }
220
+ });
221
+ it('returns conflict with startedAt on 409', async () => {
222
+ const fetchImpl = vi.fn().mockResolvedValue({
223
+ ok: false,
224
+ status: 409,
225
+ json: async () => ({ error: 'sweep-in-progress', startedAt: NOW_ISO }),
226
+ });
227
+ const result = await triggerManualSweep(fetchImpl);
228
+ expect(result.status).toBe('conflict');
229
+ if (result.status === 'conflict') {
230
+ expect(result.startedAt).toBe(NOW_ISO);
231
+ }
232
+ });
233
+ it('returns error on 500', async () => {
234
+ const fetchImpl = vi.fn().mockResolvedValue({
235
+ ok: false,
236
+ status: 500,
237
+ json: async () => ({ error: 'sweep-failed' }),
238
+ });
239
+ const result = await triggerManualSweep(fetchImpl);
240
+ expect(result.status).toBe('error');
241
+ });
242
+ });
243
+ describe('snapshotTotals', () => {
244
+ it('flattens the view model status counts into a single record', () => {
245
+ const totals = snapshotTotals(buildViewModel({
246
+ totalCount: 4,
247
+ activeCount: 2,
248
+ idleCount: 1,
249
+ staleCount: 1,
250
+ }));
251
+ expect(totals).toEqual({ total: 4, active: 2, idle: 1, stale: 1 });
252
+ });
253
+ it('returns zeros for an empty pool view model', () => {
254
+ const totals = snapshotTotals(buildViewModel());
255
+ expect(totals).toEqual({ total: 0, active: 0, idle: 0, stale: 0 });
256
+ });
257
+ });
258
+ describe('computeChangedMetricKeys', () => {
259
+ it('returns the keys whose values differ between previous and next', () => {
260
+ const previous = { total: 3, active: 2, idle: 1, stale: 0 };
261
+ const next = { total: 4, active: 3, idle: 1, stale: 0 };
262
+ expect(computeChangedMetricKeys(previous, next)).toEqual(['total', 'active']);
263
+ });
264
+ it('returns an empty array when nothing changed', () => {
265
+ const totals = { total: 1, active: 1, idle: 0, stale: 0 };
266
+ expect(computeChangedMetricKeys(totals, totals)).toEqual([]);
267
+ });
268
+ it('treats null previous values as no change (cold start)', () => {
269
+ const previous = { total: null, active: null, idle: null, stale: null };
270
+ const next = { total: 5, active: 5, idle: 0, stale: 0 };
271
+ expect(computeChangedMetricKeys(previous, next)).toEqual([]);
272
+ });
273
+ });
274
+ //# sourceMappingURL=worktreePanel.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"worktreePanel.test.js","sourceRoot":"","sources":["../../../../../src/tests/units/dashboard/modules/worktreePanel.test.ts"],"names":[],"mappings":"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"}