cadr-cli 2.0.0 → 2.0.2

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 (153) hide show
  1. package/dist/adr/adr.d.ts +17 -0
  2. package/dist/adr/adr.d.ts.map +1 -0
  3. package/dist/{adr.js → adr/adr.js} +4 -44
  4. package/dist/adr/adr.js.map +1 -0
  5. package/dist/adr/adr.test.d.ts +5 -0
  6. package/dist/{adr.test.d.ts.map → adr/adr.test.d.ts.map} +1 -1
  7. package/dist/{adr.test.js → adr/adr.test.js} +0 -14
  8. package/dist/adr/adr.test.js.map +1 -0
  9. package/dist/adr/index.d.ts +2 -0
  10. package/dist/adr/index.d.ts.map +1 -0
  11. package/dist/adr/index.js +18 -0
  12. package/dist/adr/index.js.map +1 -0
  13. package/dist/analysis/analysis.orchestrator.d.ts +14 -0
  14. package/dist/analysis/analysis.orchestrator.d.ts.map +1 -0
  15. package/dist/analysis/analysis.orchestrator.js +175 -0
  16. package/dist/analysis/analysis.orchestrator.js.map +1 -0
  17. package/dist/analysis/analysis.orchestrator.test.d.ts +2 -0
  18. package/dist/analysis/analysis.orchestrator.test.d.ts.map +1 -0
  19. package/dist/analysis/analysis.orchestrator.test.js +177 -0
  20. package/dist/analysis/analysis.orchestrator.test.js.map +1 -0
  21. package/dist/analysis/strategies/git-strategy.d.ts +22 -0
  22. package/dist/analysis/strategies/git-strategy.d.ts.map +1 -0
  23. package/dist/analysis/strategies/git-strategy.js +114 -0
  24. package/dist/analysis/strategies/git-strategy.js.map +1 -0
  25. package/dist/analysis/strategies/git-strategy.test.d.ts +2 -0
  26. package/dist/analysis/strategies/git-strategy.test.d.ts.map +1 -0
  27. package/dist/analysis/strategies/git-strategy.test.js +147 -0
  28. package/dist/analysis/strategies/git-strategy.test.js.map +1 -0
  29. package/dist/commands/analyze.js +3 -3
  30. package/dist/commands/analyze.js.map +1 -1
  31. package/dist/commands/analyze.test.d.ts +2 -0
  32. package/dist/commands/analyze.test.d.ts.map +1 -0
  33. package/dist/commands/analyze.test.js +70 -0
  34. package/dist/commands/analyze.test.js.map +1 -0
  35. package/dist/commands/init.test.js +128 -2
  36. package/dist/commands/init.test.js.map +1 -1
  37. package/dist/config.test.js +167 -0
  38. package/dist/config.test.js.map +1 -1
  39. package/dist/git/git.errors.d.ts +6 -0
  40. package/dist/git/git.errors.d.ts.map +1 -0
  41. package/dist/git/git.errors.js +15 -0
  42. package/dist/git/git.errors.js.map +1 -0
  43. package/dist/git/git.errors.test.d.ts +2 -0
  44. package/dist/git/git.errors.test.d.ts.map +1 -0
  45. package/dist/git/git.errors.test.js +34 -0
  46. package/dist/git/git.errors.test.js.map +1 -0
  47. package/dist/git/git.operations.d.ts +12 -0
  48. package/dist/git/git.operations.d.ts.map +1 -0
  49. package/dist/git/git.operations.js +64 -0
  50. package/dist/git/git.operations.js.map +1 -0
  51. package/dist/git/git.operations.test.d.ts +2 -0
  52. package/dist/git/git.operations.test.d.ts.map +1 -0
  53. package/dist/git/git.operations.test.js +164 -0
  54. package/dist/git/git.operations.test.js.map +1 -0
  55. package/dist/git/index.d.ts +4 -0
  56. package/dist/git/index.d.ts.map +1 -0
  57. package/dist/git/index.js +19 -0
  58. package/dist/git/index.js.map +1 -0
  59. package/dist/llm/index.d.ts +3 -0
  60. package/dist/llm/index.d.ts.map +1 -0
  61. package/dist/llm/index.js +19 -0
  62. package/dist/llm/index.js.map +1 -0
  63. package/dist/llm/llm.d.ts +35 -0
  64. package/dist/llm/llm.d.ts.map +1 -0
  65. package/dist/{llm.js → llm/llm.js} +16 -58
  66. package/dist/llm/llm.js.map +1 -0
  67. package/dist/{llm.test.d.ts.map → llm/llm.test.d.ts.map} +1 -1
  68. package/dist/llm/llm.test.js +224 -0
  69. package/dist/llm/llm.test.js.map +1 -0
  70. package/dist/{prompts.d.ts → llm/prompts.d.ts} +1 -38
  71. package/dist/llm/prompts.d.ts.map +1 -0
  72. package/dist/{prompts.js → llm/prompts.js} +9 -54
  73. package/dist/llm/prompts.js.map +1 -0
  74. package/dist/llm/response-parser.d.ts +9 -0
  75. package/dist/llm/response-parser.d.ts.map +1 -0
  76. package/dist/llm/response-parser.js +67 -0
  77. package/dist/llm/response-parser.js.map +1 -0
  78. package/dist/llm/response-parser.test.d.ts +2 -0
  79. package/dist/llm/response-parser.test.d.ts.map +1 -0
  80. package/dist/llm/response-parser.test.js +134 -0
  81. package/dist/llm/response-parser.test.js.map +1 -0
  82. package/dist/presenters/console-presenter.d.ts +35 -0
  83. package/dist/presenters/console-presenter.d.ts.map +1 -0
  84. package/dist/presenters/console-presenter.js +114 -0
  85. package/dist/presenters/console-presenter.js.map +1 -0
  86. package/dist/presenters/console-presenter.test.d.ts +2 -0
  87. package/dist/presenters/console-presenter.test.d.ts.map +1 -0
  88. package/dist/presenters/console-presenter.test.js +227 -0
  89. package/dist/presenters/console-presenter.test.js.map +1 -0
  90. package/dist/version.test.d.ts +1 -2
  91. package/dist/version.test.d.ts.map +1 -1
  92. package/dist/version.test.js +29 -16
  93. package/dist/version.test.js.map +1 -1
  94. package/package.json +1 -1
  95. package/src/{adr.test.ts → adr/adr.test.ts} +10 -23
  96. package/src/{adr.ts → adr/adr.ts} +7 -48
  97. package/src/adr/index.ts +1 -0
  98. package/src/analysis/analysis.orchestrator.test.ts +237 -0
  99. package/src/analysis/analysis.orchestrator.ts +175 -0
  100. package/src/analysis/strategies/git-strategy.test.ts +210 -0
  101. package/src/analysis/strategies/git-strategy.ts +106 -0
  102. package/src/commands/analyze.test.ts +91 -0
  103. package/src/commands/analyze.ts +8 -9
  104. package/src/commands/init.test.ts +200 -5
  105. package/src/config.test.ts +232 -2
  106. package/src/git/git.errors.test.ts +43 -0
  107. package/src/git/git.errors.ts +10 -0
  108. package/src/git/git.operations.test.ts +222 -0
  109. package/src/git/git.operations.ts +85 -0
  110. package/src/git/index.ts +3 -0
  111. package/src/llm/index.ts +2 -0
  112. package/src/llm/llm.test.ts +315 -0
  113. package/src/{llm.ts → llm/llm.ts} +46 -107
  114. package/src/{prompts.ts → llm/prompts.ts} +30 -72
  115. package/src/llm/response-parser.test.ts +170 -0
  116. package/src/llm/response-parser.ts +90 -0
  117. package/src/presenters/console-presenter.test.ts +259 -0
  118. package/src/presenters/console-presenter.ts +152 -0
  119. package/src/version.test.ts +30 -16
  120. package/dist/adr.d.ts +0 -50
  121. package/dist/adr.d.ts.map +0 -1
  122. package/dist/adr.js.map +0 -1
  123. package/dist/adr.test.d.ts +0 -8
  124. package/dist/adr.test.js.map +0 -1
  125. package/dist/analysis.d.ts +0 -24
  126. package/dist/analysis.d.ts.map +0 -1
  127. package/dist/analysis.js +0 -281
  128. package/dist/analysis.js.map +0 -1
  129. package/dist/analysis.test.d.ts +0 -8
  130. package/dist/analysis.test.d.ts.map +0 -1
  131. package/dist/analysis.test.js +0 -351
  132. package/dist/analysis.test.js.map +0 -1
  133. package/dist/git.d.ts +0 -54
  134. package/dist/git.d.ts.map +0 -1
  135. package/dist/git.js +0 -204
  136. package/dist/git.js.map +0 -1
  137. package/dist/llm.d.ts +0 -73
  138. package/dist/llm.d.ts.map +0 -1
  139. package/dist/llm.js.map +0 -1
  140. package/dist/llm.test.js +0 -592
  141. package/dist/llm.test.js.map +0 -1
  142. package/dist/prompts.d.ts.map +0 -1
  143. package/dist/prompts.js.map +0 -1
  144. package/dist/prompts.test.d.ts +0 -2
  145. package/dist/prompts.test.d.ts.map +0 -1
  146. package/dist/prompts.test.js +0 -427
  147. package/dist/prompts.test.js.map +0 -1
  148. package/src/analysis.test.ts +0 -396
  149. package/src/analysis.ts +0 -262
  150. package/src/git.ts +0 -300
  151. package/src/llm.test.ts +0 -701
  152. package/src/prompts.test.ts +0 -515
  153. /package/dist/{llm.test.d.ts → llm/llm.test.d.ts} +0 -0
@@ -1,515 +0,0 @@
1
- import {
2
- ANALYSIS_PROMPT_V1,
3
- formatPrompt,
4
- GENERATION_PROMPT_V1,
5
- formatGenerationPrompt,
6
- promptForGeneration
7
- } from './prompts';
8
- import * as readline from 'readline';
9
-
10
- // Mock readline
11
- jest.mock('readline');
12
-
13
- describe('Prompts Module', () => {
14
- describe('ANALYSIS_PROMPT_V1', () => {
15
- test('prompt template includes required placeholders', () => {
16
- expect(ANALYSIS_PROMPT_V1).toContain('{file_paths}');
17
- expect(ANALYSIS_PROMPT_V1).toContain('{diff_content}');
18
- });
19
-
20
- test('prompt template includes architectural significance instructions', () => {
21
- expect(ANALYSIS_PROMPT_V1).toContain('architectural');
22
- expect(ANALYSIS_PROMPT_V1).toContain('significant');
23
- });
24
-
25
- test('prompt template specifies JSON response format', () => {
26
- expect(ANALYSIS_PROMPT_V1).toContain('JSON');
27
- expect(ANALYSIS_PROMPT_V1).toContain('is_significant');
28
- expect(ANALYSIS_PROMPT_V1).toContain('reason');
29
- });
30
-
31
- test('prompt template includes analysis criteria', () => {
32
- const prompt = ANALYSIS_PROMPT_V1.toLowerCase();
33
-
34
- // Should mention various architectural concerns
35
- expect(
36
- prompt.includes('pattern') ||
37
- prompt.includes('data model') ||
38
- prompt.includes('api') ||
39
- prompt.includes('security')
40
- ).toBe(true);
41
- });
42
-
43
- test('prompt includes all specific architectural significance criteria', () => {
44
- const prompt = ANALYSIS_PROMPT_V1.toLowerCase();
45
-
46
- // Must include all key criteria
47
- expect(prompt).toContain('dependency');
48
- expect(prompt).toContain('infrastructure');
49
- expect(prompt).toContain('api contract');
50
- expect(prompt).toContain('data schema');
51
- expect(prompt).toContain('authentication');
52
- expect(prompt).toContain('authorization');
53
- expect(prompt).toContain('cross-cutting');
54
- });
55
-
56
- test('prompt specifies empty string for non-significant changes', () => {
57
- expect(ANALYSIS_PROMPT_V1).toContain('empty string');
58
- });
59
-
60
- test('prompt emphasizes strict JSON output format', () => {
61
- expect(ANALYSIS_PROMPT_V1).toContain('minified');
62
- expect(ANALYSIS_PROMPT_V1).toContain('no preamble');
63
- expect(ANALYSIS_PROMPT_V1).toContain('no markdown');
64
- });
65
- });
66
-
67
- describe('formatPrompt', () => {
68
- const mockData = {
69
- file_paths: ['src/auth.ts', 'src/user.ts', 'src/middleware/auth.ts'],
70
- diff_content: `
71
- diff --git a/src/auth.ts b/src/auth.ts
72
- index 1234567..abcdefg 100644
73
- --- a/src/auth.ts
74
- +++ b/src/auth.ts
75
- @@ -1,5 +1,10 @@
76
- +import jwt from 'jsonwebtoken';
77
- +
78
- +export function authenticateUser(token: string) {
79
- + return jwt.verify(token, process.env.JWT_SECRET);
80
- +}
81
- `
82
- };
83
-
84
- test('replaces file_paths placeholder with formatted list', () => {
85
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, mockData);
86
-
87
- expect(formatted).not.toContain('{file_paths}');
88
- expect(formatted).toContain('src/auth.ts');
89
- expect(formatted).toContain('src/user.ts');
90
- expect(formatted).toContain('src/middleware/auth.ts');
91
- });
92
-
93
- test('replaces diff_content placeholder with actual diff', () => {
94
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, mockData);
95
-
96
- expect(formatted).not.toContain('{diff_content}');
97
- expect(formatted).toContain('diff --git');
98
- expect(formatted).toContain('authenticateUser');
99
- expect(formatted).toContain('jwt.verify');
100
- });
101
-
102
- test('handles empty file paths array', () => {
103
- const emptyData = {
104
- file_paths: [],
105
- diff_content: mockData.diff_content
106
- };
107
-
108
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, emptyData);
109
-
110
- expect(formatted).not.toContain('{file_paths}');
111
- expect(formatted).toBeDefined();
112
- });
113
-
114
- test('handles empty diff content', () => {
115
- const emptyDiff = {
116
- file_paths: mockData.file_paths,
117
- diff_content: ''
118
- };
119
-
120
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, emptyDiff);
121
-
122
- expect(formatted).not.toContain('{diff_content}');
123
- expect(formatted).toBeDefined();
124
- });
125
-
126
- test('preserves prompt structure and instructions', () => {
127
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, mockData);
128
-
129
- expect(formatted).toContain('architectural');
130
- expect(formatted).toContain('significant');
131
- expect(formatted).toContain('JSON');
132
- });
133
-
134
- test('formats multiple file paths readably', () => {
135
- const manyFiles = {
136
- file_paths: [
137
- 'src/auth/login.ts',
138
- 'src/auth/logout.ts',
139
- 'src/auth/session.ts',
140
- 'src/middleware/auth.ts',
141
- 'src/models/user.ts'
142
- ],
143
- diff_content: mockData.diff_content
144
- };
145
-
146
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, manyFiles);
147
-
148
- // Should list all files in a readable format
149
- manyFiles.file_paths.forEach(filePath => {
150
- expect(formatted).toContain(filePath);
151
- });
152
- });
153
-
154
- test('handles special characters in diff content', () => {
155
- const specialChars = {
156
- file_paths: ['src/test.ts'],
157
- diff_content: `
158
- diff --git a/src/test.ts b/src/test.ts
159
- +const regex = /[a-z]+/gi;
160
- +const template = \`Hello \${name}\`;
161
- +const quote = "He said: \\"Hello\\"";
162
- `
163
- };
164
-
165
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, specialChars);
166
-
167
- expect(formatted).toContain('regex');
168
- expect(formatted).toContain('template');
169
- expect(formatted).toContain('quote');
170
- });
171
-
172
- test('returns valid prompt with all placeholders replaced', () => {
173
- const formatted = formatPrompt(ANALYSIS_PROMPT_V1, mockData);
174
-
175
- // Should not contain any unreplaced placeholders
176
- expect(formatted).not.toMatch(/\{[a-z_]+\}/);
177
- });
178
- });
179
-
180
- describe('Prompt Versioning', () => {
181
- test('ANALYSIS_PROMPT_V1 is exported and accessible', () => {
182
- expect(ANALYSIS_PROMPT_V1).toBeDefined();
183
- expect(typeof ANALYSIS_PROMPT_V1).toBe('string');
184
- expect(ANALYSIS_PROMPT_V1.length).toBeGreaterThan(0);
185
- });
186
-
187
- test('prompt version is clearly identifiable', () => {
188
- // Future proofing: when V2 is added, we should be able to distinguish versions
189
- expect(ANALYSIS_PROMPT_V1).toBeDefined();
190
- });
191
- });
192
-
193
- describe('Prompt Quality', () => {
194
- test('prompt is not empty or too short', () => {
195
- expect(ANALYSIS_PROMPT_V1.length).toBeGreaterThan(50);
196
- });
197
-
198
- test('prompt provides clear instructions', () => {
199
- const prompt = ANALYSIS_PROMPT_V1.toLowerCase();
200
-
201
- // Should have clear action words
202
- expect(
203
- prompt.includes('analyze') ||
204
- prompt.includes('determine') ||
205
- prompt.includes('evaluate')
206
- ).toBe(true);
207
- });
208
-
209
- test('prompt specifies expected output format', () => {
210
- expect(ANALYSIS_PROMPT_V1).toContain('is_significant');
211
- expect(ANALYSIS_PROMPT_V1).toContain('reason');
212
-
213
- // Should specify it's a boolean
214
- const hasBoolean =
215
- ANALYSIS_PROMPT_V1.includes('boolean') ||
216
- ANALYSIS_PROMPT_V1.includes('true') ||
217
- ANALYSIS_PROMPT_V1.includes('false');
218
-
219
- expect(hasBoolean).toBe(true);
220
- });
221
- });
222
-
223
- describe('GENERATION_PROMPT_V1', () => {
224
- test('prompt template includes required placeholders', () => {
225
- expect(GENERATION_PROMPT_V1).toContain('{file_paths}');
226
- expect(GENERATION_PROMPT_V1).toContain('{diff_content}');
227
- expect(GENERATION_PROMPT_V1).toContain('{current_date}');
228
- });
229
-
230
- test('prompt includes MADR template structure', () => {
231
- const prompt = GENERATION_PROMPT_V1.toLowerCase();
232
-
233
- // Must mention MADR sections
234
- expect(prompt).toContain('context and problem statement');
235
- expect(prompt).toContain('decision drivers');
236
- expect(prompt).toContain('considered options');
237
- expect(prompt).toContain('decision outcome');
238
- expect(prompt).toContain('consequences');
239
- });
240
-
241
- test('prompt specifies MADR format explicitly', () => {
242
- expect(GENERATION_PROMPT_V1).toContain('MADR');
243
- });
244
-
245
- test('prompt includes all required MADR sections in template', () => {
246
- const prompt = GENERATION_PROMPT_V1;
247
-
248
- // Check for MADR section headings
249
- expect(prompt).toContain('# ['); // Title format
250
- expect(prompt).toContain('* Status:');
251
- expect(prompt).toContain('* Date:');
252
- expect(prompt).toContain('## Context and Problem Statement');
253
- expect(prompt).toContain('## Decision Drivers');
254
- expect(prompt).toContain('## Considered Options');
255
- expect(prompt).toContain('## Decision Outcome');
256
- expect(prompt).toContain('### Consequences');
257
- expect(prompt).toContain('## More Information');
258
- });
259
-
260
- test('prompt instructs to use EXACT markdown structure', () => {
261
- const prompt = GENERATION_PROMPT_V1.toUpperCase();
262
- expect(prompt).toContain('EXACT');
263
- });
264
-
265
- test('prompt specifies response should be markdown only', () => {
266
- const prompt = GENERATION_PROMPT_V1.toLowerCase();
267
- expect(prompt).toContain('respond only');
268
- expect(prompt).toContain('markdown');
269
- });
270
-
271
- test('prompt version is clearly identified', () => {
272
- expect(GENERATION_PROMPT_V1).toBeDefined();
273
- expect(typeof GENERATION_PROMPT_V1).toBe('string');
274
- });
275
- });
276
-
277
- describe('formatGenerationPrompt', () => {
278
- const mockData = {
279
- file_paths: ['src/database.ts', 'src/config.ts'],
280
- diff_content: `
281
- diff --git a/src/database.ts b/src/database.ts
282
- +import pg from 'pg';
283
- +export const database = new pg.Pool();
284
- `
285
- };
286
-
287
- test('replaces file_paths placeholder with formatted list', () => {
288
- const formatted = formatGenerationPrompt(mockData);
289
-
290
- expect(formatted).not.toContain('{file_paths}');
291
- expect(formatted).toContain('src/database.ts');
292
- expect(formatted).toContain('src/config.ts');
293
- });
294
-
295
- test('replaces diff_content placeholder with actual diff', () => {
296
- const formatted = formatGenerationPrompt(mockData);
297
-
298
- expect(formatted).not.toContain('{diff_content}');
299
- expect(formatted).toContain('diff --git');
300
- expect(formatted).toContain('pg.Pool');
301
- });
302
-
303
- test('replaces current_date placeholder with valid date', () => {
304
- const formatted = formatGenerationPrompt(mockData);
305
-
306
- expect(formatted).not.toContain('{current_date}');
307
- // Check for YYYY-MM-DD format
308
- expect(formatted).toMatch(/\d{4}-\d{2}-\d{2}/);
309
- });
310
-
311
- test('current date is today\'s date', () => {
312
- const formatted = formatGenerationPrompt(mockData);
313
- const today = new Date().toISOString().split('T')[0];
314
-
315
- expect(formatted).toContain(today);
316
- });
317
-
318
- test('handles empty file paths array', () => {
319
- const emptyData = {
320
- file_paths: [],
321
- diff_content: mockData.diff_content
322
- };
323
-
324
- const formatted = formatGenerationPrompt(emptyData);
325
- expect(formatted).not.toContain('{file_paths}');
326
- expect(formatted).toBeDefined();
327
- });
328
-
329
- test('preserves MADR template structure', () => {
330
- const formatted = formatGenerationPrompt(mockData);
331
-
332
- expect(formatted).toContain('Context and Problem Statement');
333
- expect(formatted).toContain('Decision Drivers');
334
- expect(formatted).toContain('Considered Options');
335
- expect(formatted).toContain('Decision Outcome');
336
- expect(formatted).toContain('Consequences');
337
- });
338
-
339
- test('returns valid prompt with all placeholders replaced', () => {
340
- const formatted = formatGenerationPrompt(mockData);
341
-
342
- // Should not contain any unreplaced placeholders
343
- expect(formatted).not.toMatch(/\{[a-z_]+\}/);
344
- });
345
- });
346
-
347
- describe('promptForGeneration', () => {
348
- let mockReadlineInterface: {
349
- question: jest.Mock;
350
- close: jest.Mock;
351
- };
352
-
353
- beforeEach(() => {
354
- // Reset mocks
355
- jest.clearAllMocks();
356
-
357
- // Setup mock readline interface
358
- mockReadlineInterface = {
359
- question: jest.fn(),
360
- close: jest.fn()
361
- };
362
-
363
- (readline.createInterface as jest.Mock).mockReturnValue(mockReadlineInterface);
364
- });
365
-
366
- test('returns true for empty input (ENTER pressed)', async () => {
367
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
368
- callback(''); // Simulate pressing ENTER
369
- });
370
-
371
- const result = await promptForGeneration('Test reason');
372
-
373
- expect(result).toBe(true);
374
- expect(mockReadlineInterface.close).toHaveBeenCalled();
375
- });
376
-
377
- test('returns true for "yes" input', async () => {
378
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
379
- callback('yes');
380
- });
381
-
382
- const result = await promptForGeneration('Test reason');
383
-
384
- expect(result).toBe(true);
385
- });
386
-
387
- test('returns true for "y" input', async () => {
388
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
389
- callback('y');
390
- });
391
-
392
- const result = await promptForGeneration('Test reason');
393
-
394
- expect(result).toBe(true);
395
- });
396
-
397
- test('returns true for "YES" input (case insensitive)', async () => {
398
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
399
- callback('YES');
400
- });
401
-
402
- const result = await promptForGeneration('Test reason');
403
-
404
- expect(result).toBe(true);
405
- });
406
-
407
- test('returns true for "Y" input (case insensitive)', async () => {
408
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
409
- callback('Y');
410
- });
411
-
412
- const result = await promptForGeneration('Test reason');
413
-
414
- expect(result).toBe(true);
415
- });
416
-
417
- test('returns false for "no" input', async () => {
418
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
419
- callback('no');
420
- });
421
-
422
- const result = await promptForGeneration('Test reason');
423
-
424
- expect(result).toBe(false);
425
- });
426
-
427
- test('returns false for "n" input', async () => {
428
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
429
- callback('n');
430
- });
431
-
432
- const result = await promptForGeneration('Test reason');
433
-
434
- expect(result).toBe(false);
435
- });
436
-
437
- test('returns false for any other input', async () => {
438
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
439
- callback('maybe');
440
- });
441
-
442
- const result = await promptForGeneration('Test reason');
443
-
444
- expect(result).toBe(false);
445
- });
446
-
447
- test('handles whitespace in input', async () => {
448
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
449
- callback(' yes ');
450
- });
451
-
452
- const result = await promptForGeneration('Test reason');
453
-
454
- expect(result).toBe(true);
455
- });
456
-
457
- test('displays the reason in the prompt', async () => {
458
- const reason = 'Introduces PostgreSQL as primary datastore';
459
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
460
- callback('yes');
461
- });
462
-
463
- await promptForGeneration(reason);
464
-
465
- // Check that question was called
466
- expect(mockReadlineInterface.question).toHaveBeenCalled();
467
- });
468
-
469
- test('closes readline interface after response', async () => {
470
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
471
- callback('yes');
472
- });
473
-
474
- await promptForGeneration('Test reason');
475
-
476
- expect(mockReadlineInterface.close).toHaveBeenCalledTimes(1);
477
- });
478
-
479
- test('creates readline interface with stdin and stdout', async () => {
480
- mockReadlineInterface.question.mockImplementation((question: string, callback: (answer: string) => void) => {
481
- callback('yes');
482
- });
483
-
484
- await promptForGeneration('Test reason');
485
-
486
- expect(readline.createInterface).toHaveBeenCalledWith({
487
- input: process.stdin,
488
- output: process.stdout
489
- });
490
- });
491
- });
492
-
493
- describe('Generation Prompt Quality', () => {
494
- test('generation prompt is substantial', () => {
495
- expect(GENERATION_PROMPT_V1.length).toBeGreaterThan(200);
496
- });
497
-
498
- test('generation prompt provides clear instructions', () => {
499
- const prompt = GENERATION_PROMPT_V1.toLowerCase();
500
-
501
- expect(
502
- prompt.includes('generate') ||
503
- prompt.includes('write') ||
504
- prompt.includes('create')
505
- ).toBe(true);
506
- });
507
-
508
- test('both prompts are exported and distinct', () => {
509
- expect(ANALYSIS_PROMPT_V1).toBeDefined();
510
- expect(GENERATION_PROMPT_V1).toBeDefined();
511
- expect(ANALYSIS_PROMPT_V1).not.toBe(GENERATION_PROMPT_V1);
512
- });
513
- });
514
- });
515
-
File without changes