@codeyam/codeyam-cli 0.1.30 → 0.1.32

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 (64) hide show
  1. package/analyzer-template/.build-info.json +7 -7
  2. package/analyzer-template/log.txt +3 -3
  3. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js +1 -0
  4. package/codeyam-cli/src/commands/__tests__/init.gitignore.test.js.map +1 -1
  5. package/codeyam-cli/src/commands/editor.js +39 -0
  6. package/codeyam-cli/src/commands/editor.js.map +1 -1
  7. package/codeyam-cli/src/commands/init.js +3 -2
  8. package/codeyam-cli/src/commands/init.js.map +1 -1
  9. package/codeyam-cli/src/utils/__tests__/devServerState.test.js +93 -1
  10. package/codeyam-cli/src/utils/__tests__/devServerState.test.js.map +1 -1
  11. package/codeyam-cli/src/utils/__tests__/editorRoadmap.test.js +712 -2
  12. package/codeyam-cli/src/utils/__tests__/editorRoadmap.test.js.map +1 -1
  13. package/codeyam-cli/src/utils/__tests__/envFile.test.js +125 -0
  14. package/codeyam-cli/src/utils/__tests__/envFile.test.js.map +1 -0
  15. package/codeyam-cli/src/utils/__tests__/handoffContext.test.js +500 -0
  16. package/codeyam-cli/src/utils/__tests__/handoffContext.test.js.map +1 -0
  17. package/codeyam-cli/src/utils/devServerState.js +32 -0
  18. package/codeyam-cli/src/utils/devServerState.js.map +1 -1
  19. package/codeyam-cli/src/utils/editorRoadmap.js +290 -17
  20. package/codeyam-cli/src/utils/editorRoadmap.js.map +1 -1
  21. package/codeyam-cli/src/utils/envFile.js +90 -0
  22. package/codeyam-cli/src/utils/envFile.js.map +1 -0
  23. package/codeyam-cli/src/utils/handoffContext.js +257 -0
  24. package/codeyam-cli/src/utils/handoffContext.js.map +1 -0
  25. package/codeyam-cli/src/utils/install-skills.js +9 -0
  26. package/codeyam-cli/src/utils/install-skills.js.map +1 -1
  27. package/codeyam-cli/src/utils/techStackConfig.js +38 -0
  28. package/codeyam-cli/src/utils/techStackConfig.js.map +1 -0
  29. package/codeyam-cli/src/utils/techStackConfig.test.js +85 -0
  30. package/codeyam-cli/src/utils/techStackConfig.test.js.map +1 -0
  31. package/codeyam-cli/src/webserver/__tests__/buildPtyEnv.test.js +119 -1
  32. package/codeyam-cli/src/webserver/__tests__/buildPtyEnv.test.js.map +1 -1
  33. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js +239 -1
  34. package/codeyam-cli/src/webserver/__tests__/editorProxy.test.js.map +1 -1
  35. package/codeyam-cli/src/webserver/app/lib/database.js.map +1 -1
  36. package/codeyam-cli/src/webserver/build/client/assets/{MiniClaudeChat-BusrvT2F.js → MiniClaudeChat-Bs2_Oua4.js} +1 -1
  37. package/codeyam-cli/src/webserver/build/client/assets/api.editor-database-verify-l0sNRNKZ.js +1 -0
  38. package/codeyam-cli/src/webserver/build/client/assets/api.editor-github-verify-l0sNRNKZ.js +1 -0
  39. package/codeyam-cli/src/webserver/build/client/assets/api.editor-handoff-l0sNRNKZ.js +1 -0
  40. package/codeyam-cli/src/webserver/build/client/assets/api.editor-hosting-verify-l0sNRNKZ.js +1 -0
  41. package/codeyam-cli/src/webserver/build/client/assets/editor.entity.(_sha)-DhtVC4aI.js +161 -0
  42. package/codeyam-cli/src/webserver/build/client/assets/{entity._sha._-Ce1s4OQ1.js → entity._sha._-pc-vc6wO.js} +1 -1
  43. package/codeyam-cli/src/webserver/build/client/assets/globals-L-aUIeux.css +1 -0
  44. package/codeyam-cli/src/webserver/build/client/assets/manifest-b0d69c06.js +1 -0
  45. package/codeyam-cli/src/webserver/build/client/assets/{root-Didv9PLi.js → root-CLedrjXQ.js} +3 -3
  46. package/codeyam-cli/src/webserver/build/server/assets/{analysisRunner-BKMsxwqe.js → analysisRunner-DPUEhrWo.js} +1 -1
  47. package/codeyam-cli/src/webserver/build/server/assets/{index-CvuvIPEn.js → index-oF2amaGI.js} +1 -1
  48. package/codeyam-cli/src/webserver/build/server/assets/init-C42BvUGp.js +14 -0
  49. package/codeyam-cli/src/webserver/build/server/assets/server-build-DiCdDL5d.js +853 -0
  50. package/codeyam-cli/src/webserver/build/server/index.js +1 -1
  51. package/codeyam-cli/src/webserver/build-info.json +5 -5
  52. package/codeyam-cli/src/webserver/editorProxy.js +178 -16
  53. package/codeyam-cli/src/webserver/editorProxy.js.map +1 -1
  54. package/codeyam-cli/src/webserver/terminalServer.js +81 -11
  55. package/codeyam-cli/src/webserver/terminalServer.js.map +1 -1
  56. package/codeyam-cli/templates/codeyam-editor-codex.md +61 -0
  57. package/codeyam-cli/templates/nextjs-prisma-sqlite/gitignore +1 -0
  58. package/codeyam-cli/templates/seed-adapters/supabase.ts +143 -31
  59. package/package.json +1 -1
  60. package/codeyam-cli/src/webserver/build/client/assets/editor.entity.(_sha)-CsYVRiNH.js +0 -147
  61. package/codeyam-cli/src/webserver/build/client/assets/globals-Gp2o-NMc.css +0 -1
  62. package/codeyam-cli/src/webserver/build/client/assets/manifest-ef0f624d.js +0 -1
  63. package/codeyam-cli/src/webserver/build/server/assets/init-B3gVLAAJ.js +0 -14
  64. package/codeyam-cli/src/webserver/build/server/assets/server-build-B4LxStYP.js +0 -741
@@ -0,0 +1,500 @@
1
+ import fs from 'fs';
2
+ import os from 'os';
3
+ import path from 'path';
4
+ import { execSync } from 'child_process';
5
+ import { captureStepSnapshot, updateHandoffProgress, generateHandoffContext, formatHandoffContextAsMarkdown, } from "../handoffContext.js";
6
+ jest.mock('child_process', () => ({
7
+ execSync: jest.fn(),
8
+ }));
9
+ const mockedExecSync = execSync;
10
+ function createTempDir() {
11
+ return fs.mkdtempSync(path.join(os.tmpdir(), 'handoff-test-'));
12
+ }
13
+ function setupCodeyamDir(root) {
14
+ const codeyamDir = path.join(root, '.codeyam');
15
+ fs.mkdirSync(codeyamDir, { recursive: true });
16
+ return codeyamDir;
17
+ }
18
+ describe('captureStepSnapshot', () => {
19
+ let root;
20
+ beforeEach(() => {
21
+ root = createTempDir();
22
+ setupCodeyamDir(root);
23
+ mockedExecSync.mockReturnValue('');
24
+ });
25
+ afterEach(() => {
26
+ fs.rmSync(root, { recursive: true, force: true });
27
+ jest.restoreAllMocks();
28
+ });
29
+ it('returns correct structure with all zero counts for an empty project', () => {
30
+ const snapshot = captureStepSnapshot(root, 3);
31
+ expect(snapshot.step).toBe(3);
32
+ expect(snapshot.label).toBe('Prototype');
33
+ expect(snapshot.scenarioCount).toBe(0);
34
+ expect(snapshot.glossaryComponentCount).toBe(0);
35
+ expect(snapshot.glossaryFunctionCount).toBe(0);
36
+ expect(snapshot.uncommittedFileCount).toBe(0);
37
+ expect(snapshot.journalEntryCount).toBe(0);
38
+ expect(typeof snapshot.timestamp).toBe('string');
39
+ });
40
+ it('returns correct step label from STEP_LABELS', () => {
41
+ expect(captureStepSnapshot(root, 1).label).toBe('Plan');
42
+ expect(captureStepSnapshot(root, 8).label).toBe('Glossary');
43
+ expect(captureStepSnapshot(root, 18).label).toBe('Push');
44
+ });
45
+ it('returns "Unknown" label for invalid step numbers', () => {
46
+ expect(captureStepSnapshot(root, 99).label).toBe('Unknown');
47
+ });
48
+ it('counts glossary components vs functions correctly', () => {
49
+ const glossaryPath = path.join(root, '.codeyam', 'glossary.json');
50
+ const glossary = [
51
+ { name: 'TaskCard', filePath: 'app/components/TaskCard.tsx' },
52
+ { name: 'EmptyState', filePath: 'app/components/EmptyState.tsx' },
53
+ {
54
+ name: 'formatDate',
55
+ filePath: 'app/lib/utils.ts',
56
+ testFile: 'app/lib/__tests__/utils.test.ts',
57
+ },
58
+ {
59
+ name: 'sortTasks',
60
+ filePath: 'app/lib/tasks.ts',
61
+ testFile: 'app/lib/__tests__/tasks.test.ts',
62
+ },
63
+ {
64
+ name: 'filterTasks',
65
+ filePath: 'app/lib/tasks.ts',
66
+ testFile: 'app/lib/__tests__/tasks.test.ts',
67
+ },
68
+ ];
69
+ fs.writeFileSync(glossaryPath, JSON.stringify(glossary));
70
+ const snapshot = captureStepSnapshot(root, 7);
71
+ expect(snapshot.glossaryComponentCount).toBe(2);
72
+ expect(snapshot.glossaryFunctionCount).toBe(3);
73
+ });
74
+ it('handles malformed glossary.json gracefully', () => {
75
+ const glossaryPath = path.join(root, '.codeyam', 'glossary.json');
76
+ fs.writeFileSync(glossaryPath, 'not valid json');
77
+ const snapshot = captureStepSnapshot(root, 7);
78
+ expect(snapshot.glossaryComponentCount).toBe(0);
79
+ expect(snapshot.glossaryFunctionCount).toBe(0);
80
+ });
81
+ it('handles object-wrapped glossary format', () => {
82
+ const glossaryPath = path.join(root, '.codeyam', 'glossary.json');
83
+ // LLMs sometimes write glossary as { components: [...] } instead of an array
84
+ const glossary = {
85
+ components: [
86
+ { name: 'TaskCard', filePath: 'app/components/TaskCard.tsx' },
87
+ ],
88
+ };
89
+ fs.writeFileSync(glossaryPath, JSON.stringify(glossary));
90
+ const snapshot = captureStepSnapshot(root, 7);
91
+ // Should handle gracefully — either extract the array or return 0
92
+ expect(snapshot.glossaryComponentCount).toBe(0);
93
+ expect(snapshot.glossaryFunctionCount).toBe(0);
94
+ });
95
+ it('counts scenario files from editor-scenarios directory', () => {
96
+ const scenariosDir = path.join(root, '.codeyam', 'editor-scenarios');
97
+ fs.mkdirSync(scenariosDir, { recursive: true });
98
+ fs.writeFileSync(path.join(scenariosDir, 'abc123.json'), JSON.stringify({ name: 'Default' }));
99
+ fs.writeFileSync(path.join(scenariosDir, 'def456.json'), JSON.stringify({ name: 'Empty' }));
100
+ fs.writeFileSync(path.join(scenariosDir, 'ghi789.json'), JSON.stringify({ name: 'Rich' }));
101
+ // seed files should not be counted
102
+ fs.writeFileSync(path.join(scenariosDir, 'abc123.seed.json'), JSON.stringify({}));
103
+ const snapshot = captureStepSnapshot(root, 10);
104
+ expect(snapshot.scenarioCount).toBe(3);
105
+ });
106
+ it('counts uncommitted files from git status, excluding .codeyam/', () => {
107
+ mockedExecSync.mockReturnValue([
108
+ ' M app/components/TaskCard.tsx',
109
+ '?? app/components/NewComponent.tsx',
110
+ ' M .codeyam/config.json',
111
+ ' M .codeyam/glossary.json',
112
+ 'A app/lib/utils.ts',
113
+ ].join('\n'));
114
+ const snapshot = captureStepSnapshot(root, 7);
115
+ // Should exclude the 2 .codeyam/ files
116
+ expect(snapshot.uncommittedFileCount).toBe(3);
117
+ });
118
+ it('handles git status failure gracefully', () => {
119
+ mockedExecSync.mockImplementation(() => {
120
+ throw new Error('Not a git repository');
121
+ });
122
+ const snapshot = captureStepSnapshot(root, 7);
123
+ expect(snapshot.uncommittedFileCount).toBe(0);
124
+ });
125
+ it('counts journal entries from journal directory', () => {
126
+ const journalDir = path.join(root, '.codeyam', 'journal');
127
+ fs.mkdirSync(journalDir, { recursive: true });
128
+ fs.writeFileSync(path.join(journalDir, 'entry-001.json'), JSON.stringify({ title: 'Initial setup' }));
129
+ fs.writeFileSync(path.join(journalDir, 'entry-002.json'), JSON.stringify({ title: 'Added components' }));
130
+ // Non-json files should not be counted
131
+ fs.writeFileSync(path.join(journalDir, 'index.md'), '# Journal');
132
+ const snapshot = captureStepSnapshot(root, 13);
133
+ expect(snapshot.journalEntryCount).toBe(2);
134
+ });
135
+ it('returns a valid ISO timestamp', () => {
136
+ const snapshot = captureStepSnapshot(root, 1);
137
+ const parsed = new Date(snapshot.timestamp);
138
+ expect(parsed.toISOString()).toBe(snapshot.timestamp);
139
+ });
140
+ });
141
+ describe('updateHandoffProgress', () => {
142
+ let root;
143
+ beforeEach(() => {
144
+ root = createTempDir();
145
+ setupCodeyamDir(root);
146
+ mockedExecSync.mockReturnValue('');
147
+ });
148
+ afterEach(() => {
149
+ fs.rmSync(root, { recursive: true, force: true });
150
+ jest.restoreAllMocks();
151
+ });
152
+ it('creates handoff object in config if it does not exist', () => {
153
+ const configPath = path.join(root, '.codeyam', 'config.json');
154
+ fs.writeFileSync(configPath, JSON.stringify({ provider: 'claude', projectSlug: 'test' }));
155
+ updateHandoffProgress(root, 3);
156
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
157
+ expect(config.handoff).toBeDefined();
158
+ expect(config.handoff.lastStep).toBe(3);
159
+ expect(config.handoff.lastProvider).toBe('claude');
160
+ expect(config.handoff.stepHistory).toHaveLength(1);
161
+ expect(config.handoff.stepHistory[0].step).toBe(3);
162
+ expect(config.handoff.stepHistory[0].label).toBe('Prototype');
163
+ });
164
+ it('appends snapshot to existing stepHistory array', () => {
165
+ const configPath = path.join(root, '.codeyam', 'config.json');
166
+ const existingHistory = [
167
+ {
168
+ step: 1,
169
+ label: 'Plan',
170
+ timestamp: '2026-03-30T10:00:00.000Z',
171
+ scenarioCount: 0,
172
+ glossaryComponentCount: 0,
173
+ glossaryFunctionCount: 0,
174
+ uncommittedFileCount: 0,
175
+ journalEntryCount: 0,
176
+ },
177
+ {
178
+ step: 2,
179
+ label: 'Prepare',
180
+ timestamp: '2026-03-30T10:05:00.000Z',
181
+ scenarioCount: 0,
182
+ glossaryComponentCount: 0,
183
+ glossaryFunctionCount: 0,
184
+ uncommittedFileCount: 5,
185
+ journalEntryCount: 0,
186
+ },
187
+ ];
188
+ fs.writeFileSync(configPath, JSON.stringify({
189
+ provider: 'claude',
190
+ handoff: {
191
+ stepHistory: existingHistory,
192
+ lastStep: 2,
193
+ lastProvider: 'claude',
194
+ },
195
+ }));
196
+ updateHandoffProgress(root, 3);
197
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
198
+ expect(config.handoff.stepHistory).toHaveLength(3);
199
+ expect(config.handoff.stepHistory[2].step).toBe(3);
200
+ expect(config.handoff.lastStep).toBe(3);
201
+ });
202
+ it('creates stepHistory array if missing from existing handoff', () => {
203
+ const configPath = path.join(root, '.codeyam', 'config.json');
204
+ fs.writeFileSync(configPath, JSON.stringify({
205
+ provider: 'gemini',
206
+ handoff: {
207
+ summary: 'Previous work',
208
+ lastStep: 5,
209
+ lastProvider: 'claude',
210
+ },
211
+ }));
212
+ updateHandoffProgress(root, 6);
213
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
214
+ expect(config.handoff.stepHistory).toHaveLength(1);
215
+ expect(config.handoff.stepHistory[0].step).toBe(6);
216
+ });
217
+ it('updates lastProvider, lastStep, lastUpdated alongside step history', () => {
218
+ const configPath = path.join(root, '.codeyam', 'config.json');
219
+ fs.writeFileSync(configPath, JSON.stringify({ provider: 'gemini', projectSlug: 'test' }));
220
+ updateHandoffProgress(root, 7);
221
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
222
+ expect(config.handoff.lastProvider).toBe('gemini');
223
+ expect(config.handoff.lastStep).toBe(7);
224
+ expect(typeof config.handoff.lastUpdated).toBe('string');
225
+ // Verify it's a valid ISO timestamp
226
+ expect(new Date(config.handoff.lastUpdated).toISOString()).toBe(config.handoff.lastUpdated);
227
+ });
228
+ it('preserves existing summary field when updating progress', () => {
229
+ const configPath = path.join(root, '.codeyam', 'config.json');
230
+ fs.writeFileSync(configPath, JSON.stringify({
231
+ provider: 'claude',
232
+ handoff: {
233
+ summary: 'Built the TaskCard',
234
+ lastStep: 5,
235
+ lastProvider: 'claude',
236
+ },
237
+ }));
238
+ updateHandoffProgress(root, 6);
239
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
240
+ expect(config.handoff.summary).toBe('Built the TaskCard');
241
+ expect(config.handoff.lastStep).toBe(6);
242
+ });
243
+ it('handles missing config.json gracefully (non-fatal)', () => {
244
+ // No config.json written — should not throw
245
+ expect(() => updateHandoffProgress(root, 3)).not.toThrow();
246
+ });
247
+ it('defaults provider to claude when not specified in config', () => {
248
+ const configPath = path.join(root, '.codeyam', 'config.json');
249
+ fs.writeFileSync(configPath, JSON.stringify({ projectSlug: 'test' }));
250
+ updateHandoffProgress(root, 1);
251
+ const config = JSON.parse(fs.readFileSync(configPath, 'utf8'));
252
+ expect(config.handoff.lastProvider).toBe('claude');
253
+ });
254
+ });
255
+ describe('generateHandoffContext', () => {
256
+ let root;
257
+ beforeEach(() => {
258
+ root = createTempDir();
259
+ setupCodeyamDir(root);
260
+ mockedExecSync.mockReturnValue('');
261
+ });
262
+ afterEach(() => {
263
+ fs.rmSync(root, { recursive: true, force: true });
264
+ jest.restoreAllMocks();
265
+ });
266
+ it('includes step history from handoff in the context', () => {
267
+ const stepHistory = [
268
+ {
269
+ step: 1,
270
+ label: 'Plan',
271
+ timestamp: '2026-03-30T10:00:00.000Z',
272
+ scenarioCount: 0,
273
+ glossaryComponentCount: 0,
274
+ glossaryFunctionCount: 0,
275
+ uncommittedFileCount: 0,
276
+ journalEntryCount: 0,
277
+ },
278
+ {
279
+ step: 3,
280
+ label: 'Prototype',
281
+ timestamp: '2026-03-30T10:15:00.000Z',
282
+ scenarioCount: 0,
283
+ glossaryComponentCount: 3,
284
+ glossaryFunctionCount: 2,
285
+ uncommittedFileCount: 12,
286
+ journalEntryCount: 0,
287
+ },
288
+ ];
289
+ const context = generateHandoffContext({
290
+ projectRoot: root,
291
+ previousProvider: 'claude',
292
+ newProvider: 'gemini',
293
+ handoff: { summary: 'Built stuff', lastStep: 3, stepHistory },
294
+ });
295
+ expect(context.previousProvider).toBe('claude');
296
+ expect(context.newProvider).toBe('gemini');
297
+ expect(context.stepHistory).toEqual(stepHistory);
298
+ expect(context.lastStep).toBe(3);
299
+ expect(context.lastStepLabel).toBe('Prototype');
300
+ });
301
+ it('includes uncommitted files from git status', () => {
302
+ mockedExecSync.mockReturnValue([
303
+ ' M app/components/Card.tsx',
304
+ '?? app/components/New.tsx',
305
+ ' M .codeyam/config.json',
306
+ ].join('\n'));
307
+ const context = generateHandoffContext({
308
+ projectRoot: root,
309
+ previousProvider: 'claude',
310
+ newProvider: 'codex',
311
+ handoff: null,
312
+ });
313
+ expect(context.uncommittedChanges.files).toHaveLength(2);
314
+ expect(context.uncommittedChanges.files[0].path).toBe('app/components/Card.tsx');
315
+ expect(context.uncommittedChanges.files[0].status).toBe('modified');
316
+ expect(context.uncommittedChanges.files[1].path).toBe('app/components/New.tsx');
317
+ expect(context.uncommittedChanges.files[1].status).toBe('untracked');
318
+ });
319
+ it('includes existing handoff summary if present', () => {
320
+ const context = generateHandoffContext({
321
+ projectRoot: root,
322
+ previousProvider: 'claude',
323
+ newProvider: 'gemini',
324
+ handoff: { summary: 'Built the TaskCard and registered scenarios' },
325
+ });
326
+ expect(context.handoffSummary).toBe('Built the TaskCard and registered scenarios');
327
+ });
328
+ it('handles null handoff gracefully', () => {
329
+ const context = generateHandoffContext({
330
+ projectRoot: root,
331
+ previousProvider: 'claude',
332
+ newProvider: 'gemini',
333
+ handoff: null,
334
+ });
335
+ expect(context.stepHistory).toEqual([]);
336
+ expect(context.handoffSummary).toBeNull();
337
+ expect(context.lastStep).toBeNull();
338
+ expect(context.lastStepLabel).toBeNull();
339
+ });
340
+ it('handles no uncommitted changes', () => {
341
+ mockedExecSync.mockReturnValue('');
342
+ const context = generateHandoffContext({
343
+ projectRoot: root,
344
+ previousProvider: 'gemini',
345
+ newProvider: 'claude',
346
+ handoff: { lastStep: 5 },
347
+ });
348
+ expect(context.uncommittedChanges.files).toHaveLength(0);
349
+ expect(context.uncommittedChanges.summary).toBe('0 files changed');
350
+ });
351
+ it('generates correct summary string for multiple changes', () => {
352
+ mockedExecSync.mockReturnValue([
353
+ ' M file1.tsx',
354
+ ' M file2.tsx',
355
+ '?? file3.tsx',
356
+ 'A file4.tsx',
357
+ ' D file5.tsx',
358
+ ].join('\n'));
359
+ const context = generateHandoffContext({
360
+ projectRoot: root,
361
+ previousProvider: 'claude',
362
+ newProvider: 'gemini',
363
+ handoff: null,
364
+ });
365
+ expect(context.uncommittedChanges.summary).toBe('5 files changed');
366
+ });
367
+ it('sets generatedAt to a valid ISO timestamp', () => {
368
+ const context = generateHandoffContext({
369
+ projectRoot: root,
370
+ previousProvider: 'claude',
371
+ newProvider: 'gemini',
372
+ handoff: null,
373
+ });
374
+ expect(new Date(context.generatedAt).toISOString()).toBe(context.generatedAt);
375
+ });
376
+ });
377
+ describe('formatHandoffContextAsMarkdown', () => {
378
+ function makeContext(overrides = {}) {
379
+ return {
380
+ previousProvider: 'claude',
381
+ newProvider: 'gemini',
382
+ lastStep: 8,
383
+ lastStepLabel: 'Glossary',
384
+ handoffSummary: null,
385
+ stepHistory: [],
386
+ uncommittedChanges: { summary: '0 files changed', files: [] },
387
+ generatedAt: '2026-03-31T12:00:00.000Z',
388
+ ...overrides,
389
+ };
390
+ }
391
+ it('includes provider names and step info in the header', () => {
392
+ const md = formatHandoffContextAsMarkdown(makeContext());
393
+ expect(md).toContain('Claude');
394
+ expect(md).toContain('Gemini');
395
+ expect(md).toContain('Step 8');
396
+ expect(md).toContain('Glossary');
397
+ });
398
+ it('generates a progress timeline table from step history', () => {
399
+ const stepHistory = [
400
+ {
401
+ step: 1,
402
+ label: 'Plan',
403
+ timestamp: '2026-03-30T10:00:00.000Z',
404
+ scenarioCount: 0,
405
+ glossaryComponentCount: 0,
406
+ glossaryFunctionCount: 0,
407
+ uncommittedFileCount: 0,
408
+ journalEntryCount: 0,
409
+ },
410
+ {
411
+ step: 3,
412
+ label: 'Prototype',
413
+ timestamp: '2026-03-30T10:15:00.000Z',
414
+ scenarioCount: 0,
415
+ glossaryComponentCount: 3,
416
+ glossaryFunctionCount: 2,
417
+ uncommittedFileCount: 12,
418
+ journalEntryCount: 0,
419
+ },
420
+ {
421
+ step: 7,
422
+ label: 'Extract',
423
+ timestamp: '2026-03-30T11:00:00.000Z',
424
+ scenarioCount: 4,
425
+ glossaryComponentCount: 6,
426
+ glossaryFunctionCount: 5,
427
+ uncommittedFileCount: 22,
428
+ journalEntryCount: 0,
429
+ },
430
+ ];
431
+ const md = formatHandoffContextAsMarkdown(makeContext({ stepHistory, lastStep: 7, lastStepLabel: 'Extract' }));
432
+ expect(md).toContain('Progress Timeline');
433
+ expect(md).toContain('| Step | Label | Scenarios | Components | Functions | Files Changed |');
434
+ expect(md).toContain('| 1 | Plan | 0 | 0 | 0 | 0 |');
435
+ expect(md).toContain('| 3 | Prototype | 0 | 3 | 2 | 12 |');
436
+ expect(md).toContain('| 7 | Extract | 4 | 6 | 5 | 22 |');
437
+ });
438
+ it('includes previous provider notes when present', () => {
439
+ const md = formatHandoffContextAsMarkdown(makeContext({
440
+ handoffSummary: 'Built the TaskCard component with 3 scenarios',
441
+ }));
442
+ expect(md).toContain("Previous Provider's Notes");
443
+ expect(md).toContain('Built the TaskCard component with 3 scenarios');
444
+ });
445
+ it('omits notes section when no summary is present', () => {
446
+ const md = formatHandoffContextAsMarkdown(makeContext({ handoffSummary: null }));
447
+ expect(md).not.toContain("Previous Provider's Notes");
448
+ });
449
+ it('lists uncommitted files with their status', () => {
450
+ const files = [
451
+ { path: 'app/components/TaskCard.tsx', status: 'modified' },
452
+ { path: 'app/components/NewFile.tsx', status: 'added' },
453
+ { path: 'app/lib/old.ts', status: 'deleted' },
454
+ ];
455
+ const md = formatHandoffContextAsMarkdown(makeContext({
456
+ uncommittedChanges: { summary: '3 files changed', files },
457
+ }));
458
+ expect(md).toContain('Uncommitted Changes');
459
+ expect(md).toContain('`app/components/TaskCard.tsx`');
460
+ expect(md).toContain('modified');
461
+ expect(md).toContain('`app/components/NewFile.tsx`');
462
+ expect(md).toContain('added');
463
+ });
464
+ it('truncates file list at 30 entries with a summary', () => {
465
+ const files = Array.from({ length: 40 }, (_, i) => ({
466
+ path: `app/file${i}.tsx`,
467
+ status: 'modified',
468
+ }));
469
+ const md = formatHandoffContextAsMarkdown(makeContext({
470
+ uncommittedChanges: { summary: '40 files changed', files },
471
+ }));
472
+ // Should show first 30 files
473
+ expect(md).toContain('`app/file0.tsx`');
474
+ expect(md).toContain('`app/file29.tsx`');
475
+ // Should NOT show file30+
476
+ expect(md).not.toContain('`app/file30.tsx`');
477
+ // Should have a "... and N more" message
478
+ expect(md).toContain('... and 10 more');
479
+ });
480
+ it('omits uncommitted changes section when no files changed', () => {
481
+ const md = formatHandoffContextAsMarkdown(makeContext({
482
+ uncommittedChanges: { summary: '0 files changed', files: [] },
483
+ }));
484
+ expect(md).not.toContain('Uncommitted Changes');
485
+ });
486
+ it('omits progress timeline section when step history is empty', () => {
487
+ const md = formatHandoffContextAsMarkdown(makeContext({ stepHistory: [] }));
488
+ expect(md).not.toContain('Progress Timeline');
489
+ });
490
+ it('includes continuation instructions referencing the correct step', () => {
491
+ const md = formatHandoffContextAsMarkdown(makeContext({
492
+ lastStep: 8,
493
+ lastStepLabel: 'Glossary',
494
+ }));
495
+ expect(md).toContain('Instructions');
496
+ expect(md).toContain('Step 8');
497
+ expect(md).toContain('Glossary');
498
+ });
499
+ });
500
+ //# sourceMappingURL=handoffContext.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"handoffContext.test.js","sourceRoot":"","sources":["../../../../../src/utils/__tests__/handoffContext.test.ts"],"names":[],"mappings":"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"}
@@ -68,4 +68,36 @@ export function computeDevServerTransition(current, response) {
68
68
  // Unknown status — no changes
69
69
  return { ...current, shouldAutoStart: false };
70
70
  }
71
+ // ─── Server Error Auto-Restart Logic ────────────────────────────────
72
+ /** Max consecutive auto-restarts before requiring manual intervention. */
73
+ export const MAX_AUTO_RESTARTS = 2;
74
+ /** Minimum interval (ms) between auto-restart attempts. */
75
+ export const AUTO_RESTART_COOLDOWN_MS = 30000;
76
+ /**
77
+ * Decide whether to auto-restart the dev server after an error page
78
+ * is shown in the preview iframe. Pure function for testability.
79
+ *
80
+ * Prevents restart loops by enforcing:
81
+ * - A 30-second cooldown between auto-restarts
82
+ * - A maximum of 2 consecutive auto-restarts (reset when server comes up)
83
+ * - No restart if the server is already starting
84
+ */
85
+ export function shouldAutoRestartOnError(state, isStarting, now) {
86
+ if (isStarting) {
87
+ return { shouldRestart: false, nextState: state };
88
+ }
89
+ if (now - state.lastAutoRestartAt < AUTO_RESTART_COOLDOWN_MS) {
90
+ return { shouldRestart: false, nextState: state };
91
+ }
92
+ if (state.autoRestartCount >= MAX_AUTO_RESTARTS) {
93
+ return { shouldRestart: false, nextState: state };
94
+ }
95
+ return {
96
+ shouldRestart: true,
97
+ nextState: {
98
+ lastAutoRestartAt: now,
99
+ autoRestartCount: state.autoRestartCount + 1,
100
+ },
101
+ };
102
+ }
71
103
  //# sourceMappingURL=devServerState.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"devServerState.js","sourceRoot":"","sources":["../../../../src/utils/devServerState.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAuBH;;;GAGG;AACH,MAAM,UAAU,0BAA0B,CACxC,OAAuB,EACvB,QAA8B;IAE9B,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,EAAE,CAAC;QAChC,OAAO;YACL,GAAG,EAAE,IAAI;YACT,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,KAAK;YACjB,KAAK,EAAE,QAAQ,CAAC,YAAY,IAAI,oBAAoB;YACpD,cAAc,EAAE,OAAO,CAAC,cAAc;YACtC,kBAAkB,EAAE,OAAO,CAAC,kBAAkB;YAC9C,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,GAAG,EAAE,CAAC;QACjB,OAAO;YACL,GAAG,EAAE,QAAQ,CAAC,GAAG;YACjB,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,IAAI;YACnC,UAAU,EAAE,KAAK;YACjB,KAAK,EAAE,IAAI;YACX,cAAc,EAAE,IAAI;YACpB,kBAAkB,EAAE,OAAO,CAAC,kBAAkB;YAC9C,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;QACnC,OAAO;YACL,GAAG,OAAO;YACV,UAAU,EAAE,IAAI;YAChB,KAAK,EAAE,IAAI;YACX,cAAc,EAAE,IAAI;YACpB,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;QAClC,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;YAChB,kDAAkD;YAClD,OAAO;gBACL,GAAG,OAAO;gBACV,GAAG,EAAE,IAAI;gBACT,UAAU,EAAE,KAAK;gBACjB,eAAe,EAAE,KAAK;aACvB,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,kBAAkB,EAAE,CAAC;YAChC,wDAAwD;YACxD,OAAO;gBACL,GAAG,OAAO;gBACV,kBAAkB,EAAE,IAAI;gBACxB,eAAe,EAAE,IAAI;aACtB,CAAC;QACJ,CAAC;QAED,4CAA4C;QAC5C,OAAO;YACL,GAAG,OAAO;YACV,UAAU,EAAE,KAAK;YACjB,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,8BAA8B;IAC9B,OAAO,EAAE,GAAG,OAAO,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC;AAChD,CAAC"}
1
+ {"version":3,"file":"devServerState.js","sourceRoot":"","sources":["../../../../src/utils/devServerState.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAuBH;;;GAGG;AACH,MAAM,UAAU,0BAA0B,CACxC,OAAuB,EACvB,QAA8B;IAE9B,IAAI,QAAQ,CAAC,MAAM,KAAK,OAAO,EAAE,CAAC;QAChC,OAAO;YACL,GAAG,EAAE,IAAI;YACT,QAAQ,EAAE,IAAI;YACd,UAAU,EAAE,KAAK;YACjB,KAAK,EAAE,QAAQ,CAAC,YAAY,IAAI,oBAAoB;YACpD,cAAc,EAAE,OAAO,CAAC,cAAc;YACtC,kBAAkB,EAAE,OAAO,CAAC,kBAAkB;YAC9C,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,GAAG,EAAE,CAAC;QACjB,OAAO;YACL,GAAG,EAAE,QAAQ,CAAC,GAAG;YACjB,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,IAAI;YACnC,UAAU,EAAE,KAAK;YACjB,KAAK,EAAE,IAAI;YACX,cAAc,EAAE,IAAI;YACpB,kBAAkB,EAAE,OAAO,CAAC,kBAAkB;YAC9C,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,UAAU,EAAE,CAAC;QACnC,OAAO;YACL,GAAG,OAAO;YACV,UAAU,EAAE,IAAI;YAChB,KAAK,EAAE,IAAI;YACX,cAAc,EAAE,IAAI;YACpB,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,IAAI,QAAQ,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;QAClC,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;YAChB,kDAAkD;YAClD,OAAO;gBACL,GAAG,OAAO;gBACV,GAAG,EAAE,IAAI;gBACT,UAAU,EAAE,KAAK;gBACjB,eAAe,EAAE,KAAK;aACvB,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,OAAO,CAAC,kBAAkB,EAAE,CAAC;YAChC,wDAAwD;YACxD,OAAO;gBACL,GAAG,OAAO;gBACV,kBAAkB,EAAE,IAAI;gBACxB,eAAe,EAAE,IAAI;aACtB,CAAC;QACJ,CAAC;QAED,4CAA4C;QAC5C,OAAO;YACL,GAAG,OAAO;YACV,UAAU,EAAE,KAAK;YACjB,eAAe,EAAE,KAAK;SACvB,CAAC;IACJ,CAAC;IAED,8BAA8B;IAC9B,OAAO,EAAE,GAAG,OAAO,EAAE,eAAe,EAAE,KAAK,EAAE,CAAC;AAChD,CAAC;AAED,uEAAuE;AAEvE,0EAA0E;AAC1E,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,CAAC;AACnC,2DAA2D;AAC3D,MAAM,CAAC,MAAM,wBAAwB,GAAG,KAAM,CAAC;AAY/C;;;;;;;;GAQG;AACH,MAAM,UAAU,wBAAwB,CACtC,KAA8B,EAC9B,UAAmB,EACnB,GAAW;IAEX,IAAI,UAAU,EAAE,CAAC;QACf,OAAO,EAAE,aAAa,EAAE,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IACpD,CAAC;IACD,IAAI,GAAG,GAAG,KAAK,CAAC,iBAAiB,GAAG,wBAAwB,EAAE,CAAC;QAC7D,OAAO,EAAE,aAAa,EAAE,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IACpD,CAAC;IACD,IAAI,KAAK,CAAC,gBAAgB,IAAI,iBAAiB,EAAE,CAAC;QAChD,OAAO,EAAE,aAAa,EAAE,KAAK,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IACpD,CAAC;IAED,OAAO;QACL,aAAa,EAAE,IAAI;QACnB,SAAS,EAAE;YACT,iBAAiB,EAAE,GAAG;YACtB,gBAAgB,EAAE,KAAK,CAAC,gBAAgB,GAAG,CAAC;SAC7C;KACF,CAAC;AACJ,CAAC"}