@su-record/vibe 2.6.16 → 2.6.18

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 (249) hide show
  1. package/CLAUDE.md +681 -681
  2. package/LICENSE +21 -21
  3. package/README.md +235 -203
  4. package/agents/architect-low.md +41 -41
  5. package/agents/architect-medium.md +59 -59
  6. package/agents/architect.md +80 -80
  7. package/agents/build-error-resolver.md +115 -115
  8. package/agents/compounder.md +261 -261
  9. package/agents/diagrammer.md +178 -178
  10. package/agents/e2e-tester.md +266 -266
  11. package/agents/explorer-low.md +42 -42
  12. package/agents/explorer-medium.md +59 -59
  13. package/agents/explorer.md +48 -48
  14. package/agents/implementer-low.md +43 -43
  15. package/agents/implementer-medium.md +52 -52
  16. package/agents/implementer.md +54 -54
  17. package/agents/refactor-cleaner.md +143 -143
  18. package/agents/research/best-practices-agent.md +199 -199
  19. package/agents/research/codebase-patterns-agent.md +157 -157
  20. package/agents/research/framework-docs-agent.md +188 -188
  21. package/agents/research/security-advisory-agent.md +213 -213
  22. package/agents/review/architecture-reviewer.md +107 -107
  23. package/agents/review/complexity-reviewer.md +116 -116
  24. package/agents/review/data-integrity-reviewer.md +88 -88
  25. package/agents/review/git-history-reviewer.md +103 -103
  26. package/agents/review/performance-reviewer.md +86 -86
  27. package/agents/review/python-reviewer.md +150 -150
  28. package/agents/review/rails-reviewer.md +139 -139
  29. package/agents/review/react-reviewer.md +144 -144
  30. package/agents/review/security-reviewer.md +80 -80
  31. package/agents/review/simplicity-reviewer.md +140 -140
  32. package/agents/review/test-coverage-reviewer.md +116 -116
  33. package/agents/review/typescript-reviewer.md +127 -127
  34. package/agents/searcher.md +54 -54
  35. package/agents/simplifier.md +120 -120
  36. package/agents/tester.md +49 -49
  37. package/agents/ui-previewer.md +268 -268
  38. package/commands/vibe.analyze.md +356 -356
  39. package/commands/vibe.reason.md +329 -329
  40. package/commands/vibe.review.md +412 -412
  41. package/commands/vibe.run.md +1266 -1266
  42. package/commands/vibe.spec.md +1054 -1054
  43. package/commands/vibe.spec.review.md +319 -319
  44. package/commands/vibe.trace.md +161 -161
  45. package/commands/vibe.utils.md +376 -376
  46. package/commands/vibe.verify.md +375 -375
  47. package/dist/cli/collaborator.js +52 -52
  48. package/dist/cli/detect.js +32 -32
  49. package/dist/cli/hud.js +20 -20
  50. package/dist/cli/index.d.ts.map +1 -1
  51. package/dist/cli/index.js +120 -118
  52. package/dist/cli/index.js.map +1 -1
  53. package/dist/cli/llm.js +144 -144
  54. package/dist/cli/postinstall.d.ts +1 -0
  55. package/dist/cli/postinstall.d.ts.map +1 -1
  56. package/dist/cli/postinstall.js +859 -859
  57. package/dist/cli/postinstall.js.map +1 -1
  58. package/dist/cli/setup/ProjectSetup.d.ts +2 -2
  59. package/dist/cli/setup/ProjectSetup.d.ts.map +1 -1
  60. package/dist/cli/setup/ProjectSetup.js +51 -17
  61. package/dist/cli/setup/ProjectSetup.js.map +1 -1
  62. package/dist/lib/DeepInit.js +24 -24
  63. package/dist/lib/IterationTracker.js +11 -11
  64. package/dist/lib/PythonParser.js +108 -108
  65. package/dist/lib/ReviewRace.js +96 -96
  66. package/dist/lib/SkillFrontmatter.js +28 -28
  67. package/dist/lib/SkillQualityGate.js +9 -9
  68. package/dist/lib/SkillRepository.js +159 -159
  69. package/dist/lib/UltraQA.js +77 -77
  70. package/dist/lib/gemini-api.js +5 -5
  71. package/dist/lib/gpt-api.js +4 -4
  72. package/dist/lib/memory/KnowledgeGraph.js +4 -4
  73. package/dist/lib/memory/MemorySearch.js +20 -20
  74. package/dist/lib/memory/MemoryStorage.js +64 -64
  75. package/dist/orchestrator/AgentManager.js +12 -12
  76. package/dist/orchestrator/MultiLlmResearch.js +8 -8
  77. package/dist/orchestrator/SmartRouter.js +11 -11
  78. package/dist/orchestrator/SwarmOrchestrator.d.ts +144 -0
  79. package/dist/orchestrator/SwarmOrchestrator.d.ts.map +1 -0
  80. package/dist/orchestrator/SwarmOrchestrator.js +361 -0
  81. package/dist/orchestrator/SwarmOrchestrator.js.map +1 -0
  82. package/dist/orchestrator/SwarmOrchestrator.test.d.ts +5 -0
  83. package/dist/orchestrator/SwarmOrchestrator.test.d.ts.map +1 -0
  84. package/dist/orchestrator/SwarmOrchestrator.test.js +95 -0
  85. package/dist/orchestrator/SwarmOrchestrator.test.js.map +1 -0
  86. package/dist/orchestrator/index.d.ts +2 -0
  87. package/dist/orchestrator/index.d.ts.map +1 -1
  88. package/dist/orchestrator/index.js +2 -0
  89. package/dist/orchestrator/index.js.map +1 -1
  90. package/dist/orchestrator/parallelResearch.js +24 -24
  91. package/dist/tools/convention/analyzeComplexity.test.js +115 -115
  92. package/dist/tools/convention/validateCodeQuality.test.js +104 -104
  93. package/dist/tools/spec/prdParser.test.js +171 -171
  94. package/dist/tools/spec/specGenerator.js +169 -169
  95. package/dist/tools/spec/traceabilityMatrix.js +64 -64
  96. package/dist/tools/spec/traceabilityMatrix.test.js +28 -28
  97. package/hooks/hooks.json +222 -222
  98. package/hooks/scripts/code-check.js +22 -22
  99. package/hooks/scripts/code-review.js +22 -22
  100. package/hooks/scripts/complexity.js +22 -22
  101. package/hooks/scripts/compound.js +23 -23
  102. package/hooks/scripts/context-save.js +33 -33
  103. package/hooks/scripts/gemini-ui-gen.js +281 -281
  104. package/hooks/scripts/generate-brand-assets.js +474 -474
  105. package/hooks/scripts/hud-multiline.js +262 -262
  106. package/hooks/scripts/hud-status.js +291 -291
  107. package/hooks/scripts/keyword-detector.js +214 -214
  108. package/hooks/scripts/llm-orchestrate.js +171 -171
  109. package/hooks/scripts/post-edit.js +97 -97
  110. package/hooks/scripts/post-tool-verify.js +210 -210
  111. package/hooks/scripts/pre-tool-guard.js +125 -125
  112. package/hooks/scripts/recall.js +22 -22
  113. package/hooks/scripts/session-start.js +30 -30
  114. package/hooks/scripts/skill-injector.js +191 -191
  115. package/hooks/scripts/utils.js +97 -97
  116. package/languages/csharp-unity.md +515 -515
  117. package/languages/gdscript-godot.md +470 -470
  118. package/languages/ruby-rails.md +489 -489
  119. package/languages/typescript-angular.md +433 -433
  120. package/languages/typescript-astro.md +416 -416
  121. package/languages/typescript-electron.md +406 -406
  122. package/languages/typescript-nestjs.md +524 -524
  123. package/languages/typescript-svelte.md +407 -407
  124. package/languages/typescript-tauri.md +365 -365
  125. package/package.json +84 -84
  126. package/skills/brand-assets.md +141 -141
  127. package/skills/commerce-patterns.md +361 -361
  128. package/skills/context7-usage.md +102 -102
  129. package/skills/e2e-commerce.md +304 -304
  130. package/skills/frontend-design.md +92 -92
  131. package/skills/git-worktree.md +181 -181
  132. package/skills/parallel-research.md +77 -77
  133. package/skills/priority-todos.md +239 -239
  134. package/skills/seo-checklist.md +244 -244
  135. package/skills/tool-fallback.md +190 -190
  136. package/skills/vibe-capabilities.md +161 -161
  137. package/vibe/constitution.md +227 -227
  138. package/vibe/rules/core/communication-guide.md +98 -98
  139. package/vibe/rules/core/development-philosophy.md +52 -52
  140. package/vibe/rules/core/quick-start.md +102 -102
  141. package/vibe/rules/quality/bdd-contract-testing.md +393 -393
  142. package/vibe/rules/quality/checklist.md +276 -276
  143. package/vibe/rules/quality/testing-strategy.md +440 -440
  144. package/vibe/rules/standards/anti-patterns.md +541 -541
  145. package/vibe/rules/standards/code-structure.md +291 -291
  146. package/vibe/rules/standards/complexity-metrics.md +313 -313
  147. package/vibe/rules/standards/naming-conventions.md +198 -198
  148. package/vibe/setup.sh +31 -31
  149. package/vibe/templates/constitution-template.md +252 -252
  150. package/vibe/templates/contract-backend-template.md +526 -526
  151. package/vibe/templates/contract-frontend-template.md +599 -599
  152. package/vibe/templates/feature-template.md +96 -96
  153. package/vibe/templates/spec-template.md +221 -221
  154. package/dist/cli/mcp.d.ts +0 -49
  155. package/dist/cli/mcp.d.ts.map +0 -1
  156. package/dist/cli/mcp.js +0 -169
  157. package/dist/cli/mcp.js.map +0 -1
  158. package/dist/lib/gemini-mcp.d.ts +0 -10
  159. package/dist/lib/gemini-mcp.d.ts.map +0 -1
  160. package/dist/lib/gemini-mcp.js +0 -353
  161. package/dist/lib/gemini-mcp.js.map +0 -1
  162. package/dist/lib/gpt-mcp.d.ts +0 -10
  163. package/dist/lib/gpt-mcp.d.ts.map +0 -1
  164. package/dist/lib/gpt-mcp.js +0 -352
  165. package/dist/lib/gpt-mcp.js.map +0 -1
  166. package/dist/tools/analytics/getUsageAnalytics.d.ts +0 -10
  167. package/dist/tools/analytics/getUsageAnalytics.d.ts.map +0 -1
  168. package/dist/tools/analytics/getUsageAnalytics.js +0 -246
  169. package/dist/tools/analytics/getUsageAnalytics.js.map +0 -1
  170. package/dist/tools/analytics/index.d.ts +0 -5
  171. package/dist/tools/analytics/index.d.ts.map +0 -1
  172. package/dist/tools/analytics/index.js +0 -5
  173. package/dist/tools/analytics/index.js.map +0 -1
  174. package/dist/tools/convention/getCodingGuide.d.ts +0 -7
  175. package/dist/tools/convention/getCodingGuide.d.ts.map +0 -1
  176. package/dist/tools/convention/getCodingGuide.js +0 -69
  177. package/dist/tools/convention/getCodingGuide.js.map +0 -1
  178. package/dist/tools/planning/analyzeRequirements.d.ts +0 -9
  179. package/dist/tools/planning/analyzeRequirements.d.ts.map +0 -1
  180. package/dist/tools/planning/analyzeRequirements.js +0 -171
  181. package/dist/tools/planning/analyzeRequirements.js.map +0 -1
  182. package/dist/tools/planning/createUserStories.d.ts +0 -9
  183. package/dist/tools/planning/createUserStories.d.ts.map +0 -1
  184. package/dist/tools/planning/createUserStories.js +0 -124
  185. package/dist/tools/planning/createUserStories.js.map +0 -1
  186. package/dist/tools/planning/featureRoadmap.d.ts +0 -10
  187. package/dist/tools/planning/featureRoadmap.d.ts.map +0 -1
  188. package/dist/tools/planning/featureRoadmap.js +0 -207
  189. package/dist/tools/planning/featureRoadmap.js.map +0 -1
  190. package/dist/tools/planning/generatePrd.d.ts +0 -11
  191. package/dist/tools/planning/generatePrd.d.ts.map +0 -1
  192. package/dist/tools/planning/generatePrd.js +0 -161
  193. package/dist/tools/planning/generatePrd.js.map +0 -1
  194. package/dist/tools/planning/index.d.ts +0 -8
  195. package/dist/tools/planning/index.d.ts.map +0 -1
  196. package/dist/tools/planning/index.js +0 -8
  197. package/dist/tools/planning/index.js.map +0 -1
  198. package/dist/tools/prompt/analyzePrompt.d.ts +0 -7
  199. package/dist/tools/prompt/analyzePrompt.d.ts.map +0 -1
  200. package/dist/tools/prompt/analyzePrompt.js +0 -150
  201. package/dist/tools/prompt/analyzePrompt.js.map +0 -1
  202. package/dist/tools/prompt/enhancePrompt.d.ts +0 -8
  203. package/dist/tools/prompt/enhancePrompt.d.ts.map +0 -1
  204. package/dist/tools/prompt/enhancePrompt.js +0 -110
  205. package/dist/tools/prompt/enhancePrompt.js.map +0 -1
  206. package/dist/tools/prompt/enhancePromptGemini.d.ts +0 -8
  207. package/dist/tools/prompt/enhancePromptGemini.d.ts.map +0 -1
  208. package/dist/tools/prompt/enhancePromptGemini.js +0 -332
  209. package/dist/tools/prompt/enhancePromptGemini.js.map +0 -1
  210. package/dist/tools/prompt/index.d.ts +0 -7
  211. package/dist/tools/prompt/index.d.ts.map +0 -1
  212. package/dist/tools/prompt/index.js +0 -7
  213. package/dist/tools/prompt/index.js.map +0 -1
  214. package/dist/tools/reasoning/applyReasoningFramework.d.ts +0 -8
  215. package/dist/tools/reasoning/applyReasoningFramework.d.ts.map +0 -1
  216. package/dist/tools/reasoning/applyReasoningFramework.js +0 -266
  217. package/dist/tools/reasoning/applyReasoningFramework.js.map +0 -1
  218. package/dist/tools/reasoning/index.d.ts +0 -5
  219. package/dist/tools/reasoning/index.d.ts.map +0 -1
  220. package/dist/tools/reasoning/index.js +0 -5
  221. package/dist/tools/reasoning/index.js.map +0 -1
  222. package/dist/tools/thinking/analyzeProblem.d.ts +0 -7
  223. package/dist/tools/thinking/analyzeProblem.d.ts.map +0 -1
  224. package/dist/tools/thinking/analyzeProblem.js +0 -55
  225. package/dist/tools/thinking/analyzeProblem.js.map +0 -1
  226. package/dist/tools/thinking/breakDownProblem.d.ts +0 -8
  227. package/dist/tools/thinking/breakDownProblem.d.ts.map +0 -1
  228. package/dist/tools/thinking/breakDownProblem.js +0 -145
  229. package/dist/tools/thinking/breakDownProblem.js.map +0 -1
  230. package/dist/tools/thinking/createThinkingChain.d.ts +0 -7
  231. package/dist/tools/thinking/createThinkingChain.d.ts.map +0 -1
  232. package/dist/tools/thinking/createThinkingChain.js +0 -44
  233. package/dist/tools/thinking/createThinkingChain.js.map +0 -1
  234. package/dist/tools/thinking/formatAsPlan.d.ts +0 -9
  235. package/dist/tools/thinking/formatAsPlan.d.ts.map +0 -1
  236. package/dist/tools/thinking/formatAsPlan.js +0 -78
  237. package/dist/tools/thinking/formatAsPlan.js.map +0 -1
  238. package/dist/tools/thinking/index.d.ts +0 -10
  239. package/dist/tools/thinking/index.d.ts.map +0 -1
  240. package/dist/tools/thinking/index.js +0 -10
  241. package/dist/tools/thinking/index.js.map +0 -1
  242. package/dist/tools/thinking/stepByStepAnalysis.d.ts +0 -8
  243. package/dist/tools/thinking/stepByStepAnalysis.d.ts.map +0 -1
  244. package/dist/tools/thinking/stepByStepAnalysis.js +0 -63
  245. package/dist/tools/thinking/stepByStepAnalysis.js.map +0 -1
  246. package/dist/tools/thinking/thinkAloudProcess.d.ts +0 -8
  247. package/dist/tools/thinking/thinkAloudProcess.d.ts.map +0 -1
  248. package/dist/tools/thinking/thinkAloudProcess.js +0 -80
  249. package/dist/tools/thinking/thinkAloudProcess.js.map +0 -1
@@ -12,20 +12,20 @@ describe('PRD Parser', () => {
12
12
  });
13
13
  describe('parsePRD()', () => {
14
14
  it('should parse basic markdown PRD', () => {
15
- const content = `# Login Feature
16
-
17
- A user authentication feature.
18
-
19
- ## Requirements
20
-
21
- - User must be able to login with email and password
22
- - User must be able to reset password
23
- - User must be able to logout
24
-
25
- ## Acceptance Criteria
26
-
27
- - Login form validates email format
28
- - Password must be at least 8 characters
15
+ const content = `# Login Feature
16
+
17
+ A user authentication feature.
18
+
19
+ ## Requirements
20
+
21
+ - User must be able to login with email and password
22
+ - User must be able to reset password
23
+ - User must be able to logout
24
+
25
+ ## Acceptance Criteria
26
+
27
+ - Login form validates email format
28
+ - Password must be at least 8 characters
29
29
  `;
30
30
  const result = parsePRD(content, 'login');
31
31
  expect(result.title).toBe('Login Feature');
@@ -34,33 +34,33 @@ A user authentication feature.
34
34
  expect(result.metadata.hasYamlFrontmatter).toBe(false);
35
35
  });
36
36
  it('should extract title from first heading', () => {
37
- const content = `# My Feature Title
38
-
39
- Some description here.
37
+ const content = `# My Feature Title
38
+
39
+ Some description here.
40
40
  `;
41
41
  const result = parsePRD(content, 'feature');
42
42
  expect(result.title).toBe('My Feature Title');
43
43
  });
44
44
  it('should extract description', () => {
45
- const content = `# Feature
46
-
47
- This is the feature description.
48
- It spans multiple lines.
49
-
50
- ## Requirements
51
-
52
- - Requirement 1
45
+ const content = `# Feature
46
+
47
+ This is the feature description.
48
+ It spans multiple lines.
49
+
50
+ ## Requirements
51
+
52
+ - Requirement 1
53
53
  `;
54
54
  const result = parsePRD(content, 'feature');
55
55
  expect(result.description).toContain('feature description');
56
56
  });
57
57
  it('should assign requirement IDs', () => {
58
- const content = `# Feature
59
-
60
- ## Requirements
61
-
62
- - First requirement
63
- - Second requirement
58
+ const content = `# Feature
59
+
60
+ ## Requirements
61
+
62
+ - First requirement
63
+ - Second requirement
64
64
  `;
65
65
  const result = parsePRD(content, 'test');
66
66
  expect(result.requirements[0].id).toMatch(/^REQ-test-\d{3}$/);
@@ -70,30 +70,30 @@ It spans multiple lines.
70
70
  });
71
71
  describe('YAML frontmatter parsing', () => {
72
72
  it('should parse YAML frontmatter', () => {
73
- const content = `---
74
- title: YAML Feature
75
- version: 1.0.0
76
- requirements:
77
- - First YAML requirement
78
- - Second YAML requirement
79
- ---
80
-
81
- # YAML Feature
82
-
83
- Content here.
73
+ const content = `---
74
+ title: YAML Feature
75
+ version: 1.0.0
76
+ requirements:
77
+ - First YAML requirement
78
+ - Second YAML requirement
79
+ ---
80
+
81
+ # YAML Feature
82
+
83
+ Content here.
84
84
  `;
85
85
  const result = parsePRD(content, 'yaml-feature');
86
86
  expect(result.metadata.hasYamlFrontmatter).toBe(true);
87
87
  expect(result.metadata.format).toBe('mixed');
88
88
  });
89
89
  it('should extract requirements from YAML', () => {
90
- const content = `---
91
- requirements:
92
- - YAML requirement one
93
- - YAML requirement two
94
- ---
95
-
96
- # Feature
90
+ const content = `---
91
+ requirements:
92
+ - YAML requirement one
93
+ - YAML requirement two
94
+ ---
95
+
96
+ # Feature
97
97
  `;
98
98
  const result = parsePRD(content, 'yaml');
99
99
  const yamlReqs = result.requirements.filter(r => r.source === 'YAML frontmatter');
@@ -102,42 +102,42 @@ requirements:
102
102
  });
103
103
  describe('Priority inference', () => {
104
104
  it('should infer high priority from "must"', () => {
105
- const content = `# Feature
106
-
107
- ## Requirements
108
-
109
- - User must be authenticated
105
+ const content = `# Feature
106
+
107
+ ## Requirements
108
+
109
+ - User must be authenticated
110
110
  `;
111
111
  const result = parsePRD(content, 'priority');
112
112
  const req = result.requirements.find(r => r.description.includes('must'));
113
113
  expect(req?.priority).toBe('high');
114
114
  });
115
115
  it('should infer high priority from "critical"', () => {
116
- const content = `# Feature
117
-
118
- ## Requirements
119
-
120
- - Critical security feature required
116
+ const content = `# Feature
117
+
118
+ ## Requirements
119
+
120
+ - Critical security feature required
121
121
  `;
122
122
  const result = parsePRD(content, 'priority');
123
123
  expect(result.requirements[0].priority).toBe('high');
124
124
  });
125
125
  it('should infer low priority from "nice to have"', () => {
126
- const content = `# Feature
127
-
128
- ## Requirements
129
-
130
- - Nice to have dark mode support
126
+ const content = `# Feature
127
+
128
+ ## Requirements
129
+
130
+ - Nice to have dark mode support
131
131
  `;
132
132
  const result = parsePRD(content, 'priority');
133
133
  expect(result.requirements[0].priority).toBe('low');
134
134
  });
135
135
  it('should default to medium priority', () => {
136
- const content = `# Feature
137
-
138
- ## Requirements
139
-
140
- - Regular feature without priority keywords
136
+ const content = `# Feature
137
+
138
+ ## Requirements
139
+
140
+ - Regular feature without priority keywords
141
141
  `;
142
142
  const result = parsePRD(content, 'priority');
143
143
  expect(result.requirements[0].priority).toBe('medium');
@@ -145,42 +145,42 @@ requirements:
145
145
  });
146
146
  describe('Section extraction', () => {
147
147
  it('should extract from Requirements section', () => {
148
- const content = `# Feature
149
-
150
- ## Overview
151
-
152
- Some overview text.
153
-
154
- ## Requirements
155
-
156
- - Requirement from requirements section
157
-
158
- ## Other Section
159
-
160
- Other content.
148
+ const content = `# Feature
149
+
150
+ ## Overview
151
+
152
+ Some overview text.
153
+
154
+ ## Requirements
155
+
156
+ - Requirement from requirements section
157
+
158
+ ## Other Section
159
+
160
+ Other content.
161
161
  `;
162
162
  const result = parsePRD(content, 'section');
163
163
  const fromReqs = result.requirements.filter(r => r.source?.includes('Requirements'));
164
164
  expect(fromReqs.length).toBeGreaterThan(0);
165
165
  });
166
166
  it('should extract from User Stories section', () => {
167
- const content = `# Feature
168
-
169
- ## User Stories
170
-
171
- As a user, I want to login so that I can access my account.
167
+ const content = `# Feature
168
+
169
+ ## User Stories
170
+
171
+ As a user, I want to login so that I can access my account.
172
172
  `;
173
173
  const result = parsePRD(content, 'story');
174
174
  const fromStories = result.requirements.filter(r => r.source?.includes('User Stories'));
175
175
  expect(fromStories.length).toBe(1);
176
176
  });
177
177
  it('should handle Korean section names', () => {
178
- const content = `# 기능
179
-
180
- ## 요구사항
181
-
182
- - 한글 요구사항 1
183
- - 한글 요구사항 2
178
+ const content = `# 기능
179
+
180
+ ## 요구사항
181
+
182
+ - 한글 요구사항 1
183
+ - 한글 요구사항 2
184
184
  `;
185
185
  const result = parsePRD(content, 'korean');
186
186
  expect(result.requirements.length).toBe(2);
@@ -188,47 +188,47 @@ As a user, I want to login so that I can access my account.
188
188
  });
189
189
  describe('List parsing', () => {
190
190
  it('should parse bullet lists with -', () => {
191
- const content = `# Feature
192
-
193
- ## Requirements
194
-
195
- - Item one
196
- - Item two
197
- - Item three
191
+ const content = `# Feature
192
+
193
+ ## Requirements
194
+
195
+ - Item one
196
+ - Item two
197
+ - Item three
198
198
  `;
199
199
  const result = parsePRD(content, 'bullet');
200
200
  expect(result.requirements.length).toBe(3);
201
201
  });
202
202
  it('should parse bullet lists with *', () => {
203
- const content = `# Feature
204
-
205
- ## Requirements
206
-
207
- * Star item one
208
- * Star item two
203
+ const content = `# Feature
204
+
205
+ ## Requirements
206
+
207
+ * Star item one
208
+ * Star item two
209
209
  `;
210
210
  const result = parsePRD(content, 'star');
211
211
  expect(result.requirements.length).toBe(2);
212
212
  });
213
213
  it('should parse numbered lists', () => {
214
- const content = `# Feature
215
-
216
- ## Requirements
217
-
218
- 1. First item
219
- 2. Second item
220
- 3. Third item
214
+ const content = `# Feature
215
+
216
+ ## Requirements
217
+
218
+ 1. First item
219
+ 2. Second item
220
+ 3. Third item
221
221
  `;
222
222
  const result = parsePRD(content, 'numbered');
223
223
  expect(result.requirements.length).toBe(3);
224
224
  });
225
225
  it('should parse numbered lists with parentheses', () => {
226
- const content = `# Feature
227
-
228
- ## Requirements
229
-
230
- 1) First item
231
- 2) Second item
226
+ const content = `# Feature
227
+
228
+ ## Requirements
229
+
230
+ 1) First item
231
+ 2) Second item
232
232
  `;
233
233
  const result = parsePRD(content, 'paren');
234
234
  expect(result.requirements.length).toBe(2);
@@ -236,16 +236,16 @@ As a user, I want to login so that I can access my account.
236
236
  });
237
237
  describe('Acceptance Criteria', () => {
238
238
  it('should extract AC and associate with requirements', () => {
239
- const content = `# Feature
240
-
241
- ## Requirements
242
-
243
- - User login functionality
244
-
245
- ## Acceptance Criteria
246
-
247
- - Login form displays correctly
248
- - Validation errors show properly
239
+ const content = `# Feature
240
+
241
+ ## Requirements
242
+
243
+ - User login functionality
244
+
245
+ ## Acceptance Criteria
246
+
247
+ - Login form displays correctly
248
+ - Validation errors show properly
249
249
  `;
250
250
  const result = parsePRD(content, 'ac');
251
251
  expect(result.requirements.length).toBeGreaterThan(0);
@@ -253,16 +253,16 @@ As a user, I want to login so that I can access my account.
253
253
  });
254
254
  describe('Deduplication', () => {
255
255
  it('should deduplicate similar requirements', () => {
256
- const content = `# Feature
257
-
258
- ## Requirements
259
-
260
- - User must login with email
261
- - User must login with email and password
262
-
263
- ## Acceptance Criteria
264
-
265
- - User must login with email validation
256
+ const content = `# Feature
257
+
258
+ ## Requirements
259
+
260
+ - User must login with email
261
+ - User must login with email and password
262
+
263
+ ## Acceptance Criteria
264
+
265
+ - User must login with email validation
266
266
  `;
267
267
  const result = parsePRD(content, 'dedup');
268
268
  // First 50 chars are used for deduplication
@@ -272,15 +272,15 @@ As a user, I want to login so that I can access my account.
272
272
  });
273
273
  describe('Fallback extraction', () => {
274
274
  it('should use fallback when no standard sections found', () => {
275
- const content = `# Feature
276
-
277
- Some text here.
278
-
279
- - This is a long enough item to be captured
280
- - Another sufficiently long item
281
- - Short
282
-
283
- Random paragraph.
275
+ const content = `# Feature
276
+
277
+ Some text here.
278
+
279
+ - This is a long enough item to be captured
280
+ - Another sufficiently long item
281
+ - Short
282
+
283
+ Random paragraph.
284
284
  `;
285
285
  const result = parsePRD(content, 'fallback');
286
286
  // Fallback only captures items > 20 chars
@@ -290,39 +290,39 @@ Random paragraph.
290
290
  });
291
291
  describe('Metadata', () => {
292
292
  it('should count sections correctly', () => {
293
- const content = `# Feature
294
-
295
- ## Section One
296
-
297
- Content.
298
-
299
- ## Section Two
300
-
301
- More content.
302
-
303
- ## Section Three
304
-
305
- Even more.
293
+ const content = `# Feature
294
+
295
+ ## Section One
296
+
297
+ Content.
298
+
299
+ ## Section Two
300
+
301
+ More content.
302
+
303
+ ## Section Three
304
+
305
+ Even more.
306
306
  `;
307
307
  const result = parsePRD(content, 'meta');
308
308
  expect(result.metadata.sectionCount).toBe(3);
309
309
  });
310
310
  it('should track requirement count', () => {
311
- const content = `# Feature
312
-
313
- ## Requirements
314
-
315
- - Req 1
316
- - Req 2
317
- - Req 3
311
+ const content = `# Feature
312
+
313
+ ## Requirements
314
+
315
+ - Req 1
316
+ - Req 2
317
+ - Req 3
318
318
  `;
319
319
  const result = parsePRD(content, 'count');
320
320
  expect(result.metadata.requirementCount).toBe(3);
321
321
  });
322
322
  it('should preserve raw content', () => {
323
- const content = `# Feature
324
-
325
- Original content preserved.
323
+ const content = `# Feature
324
+
325
+ Original content preserved.
326
326
  `;
327
327
  const result = parsePRD(content, 'raw');
328
328
  expect(result.raw).toBe(content);