@tuan_son.dinh/gsd 2.6.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 (227) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +453 -0
  3. package/dist/app-paths.d.ts +4 -0
  4. package/dist/app-paths.js +6 -0
  5. package/dist/cli.d.ts +1 -0
  6. package/dist/cli.js +269 -0
  7. package/dist/loader.d.ts +2 -0
  8. package/dist/loader.js +70 -0
  9. package/dist/logo.d.ts +16 -0
  10. package/dist/logo.js +25 -0
  11. package/dist/onboarding.d.ts +43 -0
  12. package/dist/onboarding.js +418 -0
  13. package/dist/pi-migration.d.ts +14 -0
  14. package/dist/pi-migration.js +57 -0
  15. package/dist/resource-loader.d.ts +22 -0
  16. package/dist/resource-loader.js +60 -0
  17. package/dist/tool-bootstrap.d.ts +4 -0
  18. package/dist/tool-bootstrap.js +74 -0
  19. package/dist/wizard.d.ts +7 -0
  20. package/dist/wizard.js +25 -0
  21. package/package.json +60 -0
  22. package/patches/@mariozechner+pi-coding-agent+0.57.1.patch +108 -0
  23. package/patches/@mariozechner+pi-tui+0.57.1.patch +47 -0
  24. package/pkg/dist/modes/interactive/theme/dark.json +85 -0
  25. package/pkg/dist/modes/interactive/theme/light.json +84 -0
  26. package/pkg/dist/modes/interactive/theme/theme-schema.json +335 -0
  27. package/pkg/dist/modes/interactive/theme/theme.d.ts +78 -0
  28. package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -0
  29. package/pkg/dist/modes/interactive/theme/theme.js +949 -0
  30. package/pkg/dist/modes/interactive/theme/theme.js.map +1 -0
  31. package/pkg/package.json +8 -0
  32. package/scripts/postinstall.js +127 -0
  33. package/src/resources/GSD-WORKFLOW.md +661 -0
  34. package/src/resources/agents/researcher.md +29 -0
  35. package/src/resources/agents/scout.md +56 -0
  36. package/src/resources/agents/worker.md +31 -0
  37. package/src/resources/extensions/ask-user-questions.ts +249 -0
  38. package/src/resources/extensions/bg-shell/index.ts +2808 -0
  39. package/src/resources/extensions/browser-tools/BROWSER-TOOLS-V2-PROPOSAL.md +1277 -0
  40. package/src/resources/extensions/browser-tools/core.js +1057 -0
  41. package/src/resources/extensions/browser-tools/index.ts +4989 -0
  42. package/src/resources/extensions/browser-tools/package.json +20 -0
  43. package/src/resources/extensions/context7/index.ts +428 -0
  44. package/src/resources/extensions/context7/package.json +11 -0
  45. package/src/resources/extensions/get-secrets-from-user.ts +352 -0
  46. package/src/resources/extensions/google-search/index.ts +323 -0
  47. package/src/resources/extensions/google-search/package.json +9 -0
  48. package/src/resources/extensions/gsd/activity-log.ts +69 -0
  49. package/src/resources/extensions/gsd/auto.ts +2744 -0
  50. package/src/resources/extensions/gsd/commands.ts +313 -0
  51. package/src/resources/extensions/gsd/crash-recovery.ts +85 -0
  52. package/src/resources/extensions/gsd/dashboard-overlay.ts +521 -0
  53. package/src/resources/extensions/gsd/docs/preferences-reference.md +176 -0
  54. package/src/resources/extensions/gsd/doctor.ts +690 -0
  55. package/src/resources/extensions/gsd/files.ts +732 -0
  56. package/src/resources/extensions/gsd/git-service.ts +597 -0
  57. package/src/resources/extensions/gsd/gitignore.ts +168 -0
  58. package/src/resources/extensions/gsd/guided-flow.ts +817 -0
  59. package/src/resources/extensions/gsd/index.ts +558 -0
  60. package/src/resources/extensions/gsd/metrics.ts +374 -0
  61. package/src/resources/extensions/gsd/migrate/command.ts +218 -0
  62. package/src/resources/extensions/gsd/migrate/index.ts +42 -0
  63. package/src/resources/extensions/gsd/migrate/parser.ts +323 -0
  64. package/src/resources/extensions/gsd/migrate/parsers.ts +624 -0
  65. package/src/resources/extensions/gsd/migrate/preview.ts +48 -0
  66. package/src/resources/extensions/gsd/migrate/transformer.ts +346 -0
  67. package/src/resources/extensions/gsd/migrate/types.ts +370 -0
  68. package/src/resources/extensions/gsd/migrate/validator.ts +55 -0
  69. package/src/resources/extensions/gsd/migrate/writer.ts +539 -0
  70. package/src/resources/extensions/gsd/observability-validator.ts +408 -0
  71. package/src/resources/extensions/gsd/package.json +11 -0
  72. package/src/resources/extensions/gsd/paths.ts +308 -0
  73. package/src/resources/extensions/gsd/preferences.ts +757 -0
  74. package/src/resources/extensions/gsd/prompt-loader.ts +50 -0
  75. package/src/resources/extensions/gsd/prompts/complete-milestone.md +25 -0
  76. package/src/resources/extensions/gsd/prompts/complete-slice.md +29 -0
  77. package/src/resources/extensions/gsd/prompts/discuss.md +189 -0
  78. package/src/resources/extensions/gsd/prompts/doctor-heal.md +29 -0
  79. package/src/resources/extensions/gsd/prompts/execute-task.md +61 -0
  80. package/src/resources/extensions/gsd/prompts/guided-complete-slice.md +1 -0
  81. package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +3 -0
  82. package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +59 -0
  83. package/src/resources/extensions/gsd/prompts/guided-execute-task.md +1 -0
  84. package/src/resources/extensions/gsd/prompts/guided-plan-milestone.md +23 -0
  85. package/src/resources/extensions/gsd/prompts/guided-plan-slice.md +1 -0
  86. package/src/resources/extensions/gsd/prompts/guided-research-slice.md +11 -0
  87. package/src/resources/extensions/gsd/prompts/guided-resume-task.md +1 -0
  88. package/src/resources/extensions/gsd/prompts/plan-milestone.md +65 -0
  89. package/src/resources/extensions/gsd/prompts/plan-slice.md +51 -0
  90. package/src/resources/extensions/gsd/prompts/queue.md +85 -0
  91. package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +48 -0
  92. package/src/resources/extensions/gsd/prompts/replan-slice.md +39 -0
  93. package/src/resources/extensions/gsd/prompts/research-milestone.md +37 -0
  94. package/src/resources/extensions/gsd/prompts/research-slice.md +28 -0
  95. package/src/resources/extensions/gsd/prompts/review-migration.md +66 -0
  96. package/src/resources/extensions/gsd/prompts/run-uat.md +109 -0
  97. package/src/resources/extensions/gsd/prompts/system.md +187 -0
  98. package/src/resources/extensions/gsd/prompts/worktree-merge.md +123 -0
  99. package/src/resources/extensions/gsd/session-forensics.ts +487 -0
  100. package/src/resources/extensions/gsd/skill-discovery.ts +137 -0
  101. package/src/resources/extensions/gsd/state.ts +460 -0
  102. package/src/resources/extensions/gsd/templates/context.md +76 -0
  103. package/src/resources/extensions/gsd/templates/decisions.md +8 -0
  104. package/src/resources/extensions/gsd/templates/milestone-summary.md +73 -0
  105. package/src/resources/extensions/gsd/templates/plan.md +131 -0
  106. package/src/resources/extensions/gsd/templates/preferences.md +24 -0
  107. package/src/resources/extensions/gsd/templates/project.md +31 -0
  108. package/src/resources/extensions/gsd/templates/reassessment.md +28 -0
  109. package/src/resources/extensions/gsd/templates/requirements.md +81 -0
  110. package/src/resources/extensions/gsd/templates/research.md +46 -0
  111. package/src/resources/extensions/gsd/templates/roadmap.md +118 -0
  112. package/src/resources/extensions/gsd/templates/slice-context.md +58 -0
  113. package/src/resources/extensions/gsd/templates/slice-summary.md +99 -0
  114. package/src/resources/extensions/gsd/templates/state.md +19 -0
  115. package/src/resources/extensions/gsd/templates/task-plan.md +52 -0
  116. package/src/resources/extensions/gsd/templates/task-summary.md +57 -0
  117. package/src/resources/extensions/gsd/templates/uat.md +54 -0
  118. package/src/resources/extensions/gsd/tests/activity-log-prune.test.ts +327 -0
  119. package/src/resources/extensions/gsd/tests/auto-preflight.test.ts +56 -0
  120. package/src/resources/extensions/gsd/tests/auto-supervisor.test.mjs +53 -0
  121. package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +225 -0
  122. package/src/resources/extensions/gsd/tests/cost-projection.test.ts +160 -0
  123. package/src/resources/extensions/gsd/tests/derive-state-deps.test.ts +341 -0
  124. package/src/resources/extensions/gsd/tests/derive-state.test.ts +689 -0
  125. package/src/resources/extensions/gsd/tests/discuss-prompt.test.ts +38 -0
  126. package/src/resources/extensions/gsd/tests/doctor.test.ts +505 -0
  127. package/src/resources/extensions/gsd/tests/git-service.test.ts +1313 -0
  128. package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +308 -0
  129. package/src/resources/extensions/gsd/tests/metrics-io.test.ts +201 -0
  130. package/src/resources/extensions/gsd/tests/metrics.test.ts +217 -0
  131. package/src/resources/extensions/gsd/tests/migrate-command.test.ts +390 -0
  132. package/src/resources/extensions/gsd/tests/migrate-parser.test.ts +786 -0
  133. package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +657 -0
  134. package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +443 -0
  135. package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +318 -0
  136. package/src/resources/extensions/gsd/tests/migrate-writer.test.ts +420 -0
  137. package/src/resources/extensions/gsd/tests/must-have-parser.test.ts +309 -0
  138. package/src/resources/extensions/gsd/tests/parsers.test.ts +1351 -0
  139. package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +163 -0
  140. package/src/resources/extensions/gsd/tests/plan-quality-validator.test.ts +386 -0
  141. package/src/resources/extensions/gsd/tests/reassess-prompt.test.ts +171 -0
  142. package/src/resources/extensions/gsd/tests/remote-questions.test.ts +155 -0
  143. package/src/resources/extensions/gsd/tests/remote-status.test.ts +99 -0
  144. package/src/resources/extensions/gsd/tests/replan-slice.test.ts +521 -0
  145. package/src/resources/extensions/gsd/tests/requirements.test.ts +125 -0
  146. package/src/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +34 -0
  147. package/src/resources/extensions/gsd/tests/resolve-ts.mjs +11 -0
  148. package/src/resources/extensions/gsd/tests/run-uat.test.ts +348 -0
  149. package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +247 -0
  150. package/src/resources/extensions/gsd/tests/workflow-config.test.mjs +53 -0
  151. package/src/resources/extensions/gsd/tests/workspace-index.test.ts +94 -0
  152. package/src/resources/extensions/gsd/tests/worktree-integration.test.ts +253 -0
  153. package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +160 -0
  154. package/src/resources/extensions/gsd/tests/worktree.test.ts +264 -0
  155. package/src/resources/extensions/gsd/types.ts +159 -0
  156. package/src/resources/extensions/gsd/unit-runtime.ts +184 -0
  157. package/src/resources/extensions/gsd/workspace-index.ts +203 -0
  158. package/src/resources/extensions/gsd/worktree-command.ts +845 -0
  159. package/src/resources/extensions/gsd/worktree-manager.ts +392 -0
  160. package/src/resources/extensions/gsd/worktree.ts +183 -0
  161. package/src/resources/extensions/mac-tools/index.ts +852 -0
  162. package/src/resources/extensions/mac-tools/swift-cli/Package.swift +22 -0
  163. package/src/resources/extensions/mac-tools/swift-cli/Sources/main.swift +1318 -0
  164. package/src/resources/extensions/mcporter/index.ts +429 -0
  165. package/src/resources/extensions/remote-questions/config.ts +81 -0
  166. package/src/resources/extensions/remote-questions/discord-adapter.ts +128 -0
  167. package/src/resources/extensions/remote-questions/format.ts +163 -0
  168. package/src/resources/extensions/remote-questions/manager.ts +192 -0
  169. package/src/resources/extensions/remote-questions/remote-command.ts +307 -0
  170. package/src/resources/extensions/remote-questions/slack-adapter.ts +92 -0
  171. package/src/resources/extensions/remote-questions/status.ts +31 -0
  172. package/src/resources/extensions/remote-questions/store.ts +77 -0
  173. package/src/resources/extensions/remote-questions/types.ts +75 -0
  174. package/src/resources/extensions/search-the-web/cache.ts +78 -0
  175. package/src/resources/extensions/search-the-web/command-search-provider.ts +95 -0
  176. package/src/resources/extensions/search-the-web/format.ts +258 -0
  177. package/src/resources/extensions/search-the-web/http.ts +238 -0
  178. package/src/resources/extensions/search-the-web/index.ts +65 -0
  179. package/src/resources/extensions/search-the-web/native-search.ts +157 -0
  180. package/src/resources/extensions/search-the-web/provider.ts +118 -0
  181. package/src/resources/extensions/search-the-web/tavily.ts +116 -0
  182. package/src/resources/extensions/search-the-web/tool-fetch-page.ts +519 -0
  183. package/src/resources/extensions/search-the-web/tool-llm-context.ts +561 -0
  184. package/src/resources/extensions/search-the-web/tool-search.ts +576 -0
  185. package/src/resources/extensions/search-the-web/url-utils.ts +91 -0
  186. package/src/resources/extensions/shared/confirm-ui.ts +126 -0
  187. package/src/resources/extensions/shared/interview-ui.ts +613 -0
  188. package/src/resources/extensions/shared/next-action-ui.ts +197 -0
  189. package/src/resources/extensions/shared/progress-widget.ts +282 -0
  190. package/src/resources/extensions/shared/terminal.ts +23 -0
  191. package/src/resources/extensions/shared/thinking-widget.ts +107 -0
  192. package/src/resources/extensions/shared/ui.ts +400 -0
  193. package/src/resources/extensions/shared/wizard-ui.ts +551 -0
  194. package/src/resources/extensions/slash-commands/audit.ts +88 -0
  195. package/src/resources/extensions/slash-commands/clear.ts +10 -0
  196. package/src/resources/extensions/slash-commands/create-extension.ts +297 -0
  197. package/src/resources/extensions/slash-commands/create-slash-command.ts +234 -0
  198. package/src/resources/extensions/slash-commands/index.ts +12 -0
  199. package/src/resources/extensions/subagent/agents.ts +126 -0
  200. package/src/resources/extensions/subagent/index.ts +1020 -0
  201. package/src/resources/extensions/voice/index.ts +195 -0
  202. package/src/resources/extensions/voice/speech-recognizer.swift +154 -0
  203. package/src/resources/skills/debug-like-expert/SKILL.md +231 -0
  204. package/src/resources/skills/debug-like-expert/references/debugging-mindset.md +253 -0
  205. package/src/resources/skills/debug-like-expert/references/hypothesis-testing.md +373 -0
  206. package/src/resources/skills/debug-like-expert/references/investigation-techniques.md +337 -0
  207. package/src/resources/skills/debug-like-expert/references/verification-patterns.md +425 -0
  208. package/src/resources/skills/debug-like-expert/references/when-to-research.md +361 -0
  209. package/src/resources/skills/frontend-design/SKILL.md +45 -0
  210. package/src/resources/skills/swiftui/SKILL.md +208 -0
  211. package/src/resources/skills/swiftui/references/animations.md +921 -0
  212. package/src/resources/skills/swiftui/references/architecture.md +1561 -0
  213. package/src/resources/skills/swiftui/references/layout-system.md +1186 -0
  214. package/src/resources/skills/swiftui/references/navigation.md +1492 -0
  215. package/src/resources/skills/swiftui/references/networking-async.md +214 -0
  216. package/src/resources/skills/swiftui/references/performance.md +1706 -0
  217. package/src/resources/skills/swiftui/references/platform-integration.md +204 -0
  218. package/src/resources/skills/swiftui/references/state-management.md +1443 -0
  219. package/src/resources/skills/swiftui/references/swiftdata.md +297 -0
  220. package/src/resources/skills/swiftui/references/testing-debugging.md +247 -0
  221. package/src/resources/skills/swiftui/references/uikit-appkit-interop.md +218 -0
  222. package/src/resources/skills/swiftui/workflows/add-feature.md +191 -0
  223. package/src/resources/skills/swiftui/workflows/build-new-app.md +311 -0
  224. package/src/resources/skills/swiftui/workflows/debug-swiftui.md +192 -0
  225. package/src/resources/skills/swiftui/workflows/optimize-performance.md +197 -0
  226. package/src/resources/skills/swiftui/workflows/ship-app.md +203 -0
  227. package/src/resources/skills/swiftui/workflows/write-tests.md +235 -0
@@ -0,0 +1,786 @@
1
+ // Migration parser test suite
2
+ // Tests for parsing old .planning directories into typed PlanningProject structures.
3
+ // Uses synthetic fixture directories — no real .planning dirs needed.
4
+
5
+ import { mkdtempSync, mkdirSync, rmSync, writeFileSync } from 'node:fs';
6
+ import { join } from 'node:path';
7
+ import { tmpdir } from 'node:os';
8
+
9
+ import { parsePlanningDirectory } from '../migrate/parser.ts';
10
+ import { validatePlanningDirectory } from '../migrate/validator.ts';
11
+
12
+ import type { PlanningProject, ValidationResult } from '../migrate/types.ts';
13
+
14
+ let passed = 0;
15
+ let failed = 0;
16
+
17
+ function assert(condition: boolean, message: string): void {
18
+ if (condition) {
19
+ passed++;
20
+ } else {
21
+ failed++;
22
+ console.error(` FAIL: ${message}`);
23
+ }
24
+ }
25
+
26
+ function assertEq<T>(actual: T, expected: T, message: string): void {
27
+ if (JSON.stringify(actual) === JSON.stringify(expected)) {
28
+ passed++;
29
+ } else {
30
+ failed++;
31
+ console.error(` FAIL: ${message} — expected ${JSON.stringify(expected)}, got ${JSON.stringify(actual)}`);
32
+ }
33
+ }
34
+
35
+ // ─── Fixture Helpers ───────────────────────────────────────────────────────
36
+
37
+ function createFixtureBase(): string {
38
+ return mkdtempSync(join(tmpdir(), 'gsd-migrate-test-'));
39
+ }
40
+
41
+ function createPlanningDir(base: string): string {
42
+ const dir = join(base, '.planning');
43
+ mkdirSync(dir, { recursive: true });
44
+ return dir;
45
+ }
46
+
47
+ function writeFile(dir: string, ...pathParts: string[]): (content: string) => void {
48
+ return (content: string) => {
49
+ const filePath = join(dir, ...pathParts);
50
+ mkdirSync(join(filePath, '..'), { recursive: true });
51
+ writeFileSync(filePath, content);
52
+ };
53
+ }
54
+
55
+ function cleanup(base: string): void {
56
+ rmSync(base, { recursive: true, force: true });
57
+ }
58
+
59
+ // ─── Sample Fixtures ───────────────────────────────────────────────────────
60
+
61
+ const SAMPLE_ROADMAP = `# Project Roadmap
62
+
63
+ ## Phases
64
+
65
+ - [x] 29 — Auth System
66
+ - [ ] 30 — Dashboard
67
+ - [ ] 31 — Notifications
68
+ `;
69
+
70
+ const SAMPLE_PROJECT = `# My Project
71
+
72
+ A sample project for testing the migration parser.
73
+
74
+ ## Goals
75
+
76
+ - Build a thing
77
+ - Ship it
78
+ `;
79
+
80
+ const SAMPLE_REQUIREMENTS = `# Requirements
81
+
82
+ ## Active
83
+
84
+ ### R001 — User Authentication
85
+ - Status: active
86
+ - Description: Users must be able to log in.
87
+
88
+ ### R002 — Dashboard View
89
+ - Status: active
90
+ - Description: Main dashboard page.
91
+
92
+ ## Validated
93
+
94
+ ### R003 — Session Management
95
+ - Status: validated
96
+ - Description: Sessions expire after 24h.
97
+
98
+ ## Deferred
99
+
100
+ ### R004 — OAuth Support
101
+ - Status: deferred
102
+ - Description: Third-party login.
103
+ `;
104
+
105
+ const SAMPLE_STATE = `# State
106
+
107
+ **Current Phase:** 30-dashboard
108
+ **Status:** in-progress
109
+ `;
110
+
111
+ const SAMPLE_CONFIG = JSON.stringify({
112
+ projectName: 'test-project',
113
+ version: '1.0',
114
+ });
115
+
116
+ const SAMPLE_PLAN_XML = `---
117
+ phase: "29-auth-system"
118
+ plan: "01"
119
+ type: "implementation"
120
+ wave: 1
121
+ depends_on: []
122
+ files_modified: [src/auth.ts, src/login.ts]
123
+ autonomous: true
124
+ must_haves:
125
+ truths:
126
+ - Users can log in
127
+ artifacts:
128
+ - src/auth.ts
129
+ key_links: []
130
+ ---
131
+
132
+ # 29-01: Implement Auth
133
+
134
+ <objective>
135
+ Build the authentication system with JWT tokens and session management.
136
+ </objective>
137
+
138
+ <tasks>
139
+ <task>Create auth middleware</task>
140
+ <task>Add login endpoint</task>
141
+ <task>Add logout endpoint</task>
142
+ </tasks>
143
+
144
+ <context>
145
+ The project needs authentication before any other features can be built.
146
+ Auth tokens use JWT with RS256 signing.
147
+ </context>
148
+
149
+ <verification>
150
+ - Login returns valid JWT
151
+ - Middleware rejects invalid tokens
152
+ - Logout invalidates session
153
+ </verification>
154
+
155
+ <success_criteria>
156
+ All auth endpoints respond correctly and tokens are validated.
157
+ </success_criteria>
158
+ `;
159
+
160
+ const SAMPLE_SUMMARY = `---
161
+ phase: "29-auth-system"
162
+ plan: "01"
163
+ subsystem: "auth"
164
+ tags:
165
+ - authentication
166
+ - security
167
+ requires: []
168
+ provides:
169
+ - auth-middleware
170
+ - jwt-validation
171
+ affects:
172
+ - api-routes
173
+ tech-stack:
174
+ - jsonwebtoken
175
+ - express
176
+ key-files:
177
+ - src/auth.ts
178
+ - src/middleware/auth.ts
179
+ key-decisions:
180
+ - Use RS256 for JWT signing
181
+ - Store refresh tokens in DB
182
+ patterns-established:
183
+ - Middleware-based auth
184
+ duration: "2h"
185
+ completed: "2026-01-15"
186
+ ---
187
+
188
+ # 29-01: Auth Implementation Summary
189
+
190
+ Authentication system implemented with JWT tokens.
191
+
192
+ ## What Happened
193
+
194
+ Built the auth middleware and login/logout endpoints.
195
+
196
+ ## Files Modified
197
+
198
+ - \`src/auth.ts\` — Core auth logic
199
+ - \`src/middleware/auth.ts\` — Express middleware
200
+ `;
201
+
202
+ const SAMPLE_RESEARCH = `# Auth Research
203
+
204
+ ## JWT vs Session Tokens
205
+
206
+ JWT tokens are stateless and work well for microservices.
207
+ Session tokens require server-side storage but are easier to revoke.
208
+
209
+ ## Decision
210
+
211
+ Use JWT with short expiry + refresh tokens.
212
+ `;
213
+
214
+ const SAMPLE_MILESTONE_ROADMAP = `# Milestone v2.2 Roadmap
215
+
216
+ ## Phases
217
+
218
+ - [x] 29 — Auth System
219
+ - [x] 30 — Dashboard
220
+ `;
221
+
222
+ const SAMPLE_MILESTONE_SECTIONED_ROADMAP = `# Project Roadmap
223
+
224
+ ## v2.0 — Foundation
225
+
226
+ <details>
227
+ <summary>Completed</summary>
228
+
229
+ - [x] 01 — Project Setup
230
+ - [x] 02 — Database Schema
231
+
232
+ </details>
233
+
234
+ ## v2.5 — Features
235
+
236
+ - [x] 29 — Auth System
237
+ - [ ] 30 — Dashboard
238
+ - [ ] 31 — Notifications
239
+ `;
240
+
241
+ const SAMPLE_QUICK_PLAN = `# 001: Fix Login Bug
242
+
243
+ ## Description
244
+
245
+ Fix the login button not responding on mobile.
246
+
247
+ ## Steps
248
+
249
+ 1. Debug click handler
250
+ 2. Fix event propagation
251
+ 3. Test on mobile
252
+ `;
253
+
254
+ const SAMPLE_QUICK_SUMMARY = `# 001: Fix Login Bug — Summary
255
+
256
+ Fixed the login button by correcting the touch event handler.
257
+ `;
258
+
259
+ // ═══════════════════════════════════════════════════════════════════════════
260
+ // Test Groups
261
+ // ═══════════════════════════════════════════════════════════════════════════
262
+
263
+ async function main(): Promise<void> {
264
+
265
+ // ─── Test 1: Complete .planning directory ──────────────────────────────
266
+ console.log('\n=== Complete .planning directory with all file types ===');
267
+ {
268
+ const base = createFixtureBase();
269
+ try {
270
+ const planning = createPlanningDir(base);
271
+
272
+ // Root files
273
+ writeFileSync(join(planning, 'PROJECT.md'), SAMPLE_PROJECT);
274
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
275
+ writeFileSync(join(planning, 'REQUIREMENTS.md'), SAMPLE_REQUIREMENTS);
276
+ writeFileSync(join(planning, 'STATE.md'), SAMPLE_STATE);
277
+ writeFileSync(join(planning, 'config.json'), SAMPLE_CONFIG);
278
+
279
+ // Phase directory with plan, summary, research
280
+ const phaseDir = join(planning, 'phases', '29-auth-system');
281
+ mkdirSync(phaseDir, { recursive: true });
282
+ writeFileSync(join(phaseDir, '29-01-PLAN.md'), SAMPLE_PLAN_XML);
283
+ writeFileSync(join(phaseDir, '29-01-SUMMARY.md'), SAMPLE_SUMMARY);
284
+ writeFileSync(join(phaseDir, '29-RESEARCH.md'), SAMPLE_RESEARCH);
285
+
286
+ // Second phase directory
287
+ const phase2Dir = join(planning, 'phases', '30-dashboard');
288
+ mkdirSync(phase2Dir, { recursive: true });
289
+ writeFileSync(join(phase2Dir, '30-01-PLAN.md'), `---
290
+ phase: "30-dashboard"
291
+ plan: "01"
292
+ type: "implementation"
293
+ wave: 1
294
+ depends_on: [29-01]
295
+ files_modified: []
296
+ autonomous: false
297
+ ---
298
+
299
+ # 30-01: Build Dashboard
300
+
301
+ <objective>
302
+ Create the main dashboard view.
303
+ </objective>
304
+
305
+ <tasks>
306
+ <task>Create dashboard component</task>
307
+ <task>Add data fetching</task>
308
+ </tasks>
309
+
310
+ <context>
311
+ Dashboard needs auth to be complete first.
312
+ </context>
313
+ `);
314
+
315
+ // Quick tasks
316
+ const quickDir = join(planning, 'quick', '001-fix-login');
317
+ mkdirSync(quickDir, { recursive: true });
318
+ writeFileSync(join(quickDir, '001-PLAN.md'), SAMPLE_QUICK_PLAN);
319
+ writeFileSync(join(quickDir, '001-SUMMARY.md'), SAMPLE_QUICK_SUMMARY);
320
+
321
+ // Milestones
322
+ const msDir = join(planning, 'milestones');
323
+ mkdirSync(msDir, { recursive: true });
324
+ writeFileSync(join(msDir, 'v2.2-ROADMAP.md'), SAMPLE_MILESTONE_ROADMAP);
325
+ writeFileSync(join(msDir, 'v2.2-REQUIREMENTS.md'), 'Milestone requirements here.');
326
+
327
+ // Research at root
328
+ const researchDir = join(planning, 'research');
329
+ mkdirSync(researchDir, { recursive: true });
330
+ writeFileSync(join(researchDir, 'architecture.md'), '# Architecture Research\n\nNotes.');
331
+
332
+ const project = await parsePlanningDirectory(planning);
333
+
334
+ // Top-level structure
335
+ assertEq(project.path, planning, 'project.path matches');
336
+ assert(project.project !== null, 'PROJECT.md parsed');
337
+ assert(project.roadmap !== null, 'ROADMAP.md parsed');
338
+ assert(project.requirements.length > 0, 'requirements parsed');
339
+ assert(project.state !== null, 'STATE.md parsed');
340
+ assert(project.config !== null, 'config.json parsed');
341
+
342
+ // Phases
343
+ assert('29-auth-system' in project.phases, 'phase 29 present');
344
+ assert('30-dashboard' in project.phases, 'phase 30 present');
345
+
346
+ const phase29 = project.phases['29-auth-system'];
347
+ assertEq(phase29?.number, 29, 'phase 29 number');
348
+ assertEq(phase29?.slug, 'auth-system', 'phase 29 slug');
349
+ assert('01' in (phase29?.plans ?? {}), 'phase 29 has plan 01');
350
+ assert('01' in (phase29?.summaries ?? {}), 'phase 29 has summary 01');
351
+ assert((phase29?.research?.length ?? 0) > 0, 'phase 29 has research');
352
+
353
+ // Plan content (XML-in-markdown)
354
+ const plan29 = phase29?.plans?.['01'];
355
+ assert(plan29 !== undefined, 'plan 29-01 exists');
356
+ assert(plan29?.objective?.includes('authentication') ?? false, 'plan objective extracted');
357
+ assert((plan29?.tasks?.length ?? 0) >= 3, 'plan tasks extracted');
358
+ assert(plan29?.context?.includes('JWT') ?? false, 'plan context extracted');
359
+ assert(plan29?.verification !== '', 'plan verification extracted');
360
+ assert(plan29?.successCriteria !== '', 'plan success criteria extracted');
361
+
362
+ // Plan frontmatter
363
+ assertEq(plan29?.frontmatter?.phase, '29-auth-system', 'plan frontmatter phase');
364
+ assertEq(plan29?.frontmatter?.plan, '01', 'plan frontmatter plan');
365
+ assertEq(plan29?.frontmatter?.type, 'implementation', 'plan frontmatter type');
366
+ assertEq(plan29?.frontmatter?.wave, 1, 'plan frontmatter wave');
367
+ assertEq(plan29?.frontmatter?.autonomous, true, 'plan frontmatter autonomous');
368
+
369
+ // Summary content
370
+ const summary29 = phase29?.summaries?.['01'];
371
+ assert(summary29 !== undefined, 'summary 29-01 exists');
372
+ assertEq(summary29?.frontmatter?.phase, '29-auth-system', 'summary frontmatter phase');
373
+ assertEq(summary29?.frontmatter?.plan, '01', 'summary frontmatter plan');
374
+ assertEq(summary29?.frontmatter?.subsystem, 'auth', 'summary frontmatter subsystem');
375
+ assert((summary29?.frontmatter?.tags?.length ?? 0) >= 2, 'summary frontmatter tags');
376
+ assert((summary29?.frontmatter?.provides?.length ?? 0) >= 2, 'summary frontmatter provides');
377
+ assert((summary29?.frontmatter?.affects?.length ?? 0) >= 1, 'summary frontmatter affects');
378
+ assert((summary29?.frontmatter?.['tech-stack']?.length ?? 0) >= 2, 'summary frontmatter tech-stack');
379
+ assert((summary29?.frontmatter?.['key-files']?.length ?? 0) >= 2, 'summary frontmatter key-files');
380
+ assert((summary29?.frontmatter?.['key-decisions']?.length ?? 0) >= 2, 'summary frontmatter key-decisions');
381
+ assert((summary29?.frontmatter?.['patterns-established']?.length ?? 0) >= 1, 'summary frontmatter patterns-established');
382
+ assertEq(summary29?.frontmatter?.duration, '2h', 'summary frontmatter duration');
383
+ assertEq(summary29?.frontmatter?.completed, '2026-01-15', 'summary frontmatter completed');
384
+
385
+ // Quick tasks
386
+ assert(project.quickTasks.length >= 1, 'quick tasks parsed');
387
+ assertEq(project.quickTasks[0]?.number, 1, 'quick task number');
388
+ assert(project.quickTasks[0]?.plan !== null, 'quick task has plan');
389
+ assert(project.quickTasks[0]?.summary !== null, 'quick task has summary');
390
+
391
+ // Milestones
392
+ assert(project.milestones.length >= 1, 'milestones parsed');
393
+
394
+ // Root research
395
+ assert(project.research.length >= 1, 'root research parsed');
396
+
397
+ // Config
398
+ assertEq(project.config?.projectName, 'test-project', 'config projectName');
399
+
400
+ // State
401
+ assert(project.state?.currentPhase?.includes('30') ?? false, 'state current phase');
402
+ assertEq(project.state?.status, 'in-progress', 'state status');
403
+
404
+ // Validation
405
+ assertEq(project.validation.valid, true, 'validation passes for complete dir');
406
+ assertEq(project.validation.issues.length, 0, 'no validation issues');
407
+ } finally {
408
+ cleanup(base);
409
+ }
410
+ }
411
+
412
+ // ─── Test 2: Minimal .planning directory (only ROADMAP.md) ─────────────
413
+ console.log('\n=== Minimal .planning directory (only ROADMAP.md) ===');
414
+ {
415
+ const base = createFixtureBase();
416
+ try {
417
+ const planning = createPlanningDir(base);
418
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
419
+
420
+ const project = await parsePlanningDirectory(planning);
421
+
422
+ assertEq(project.project, null, 'minimal: PROJECT.md is null');
423
+ assert(project.roadmap !== null, 'minimal: ROADMAP.md parsed');
424
+ assertEq(project.requirements.length, 0, 'minimal: no requirements');
425
+ assertEq(project.state, null, 'minimal: no state');
426
+ assertEq(project.config, null, 'minimal: no config');
427
+ assertEq(Object.keys(project.phases).length, 0, 'minimal: no phases');
428
+ assertEq(project.quickTasks.length, 0, 'minimal: no quick tasks');
429
+ assertEq(project.milestones.length, 0, 'minimal: no milestones');
430
+ assertEq(project.research.length, 0, 'minimal: no research');
431
+ assertEq(project.validation.valid, true, 'minimal: validation passes');
432
+ } finally {
433
+ cleanup(base);
434
+ }
435
+ }
436
+
437
+ // ─── Test 3: Missing directory → validation fatal error ────────────────
438
+ console.log('\n=== Missing directory → validation returns fatal error ===');
439
+ {
440
+ const base = createFixtureBase();
441
+ try {
442
+ const result = await validatePlanningDirectory(join(base, 'nonexistent'));
443
+
444
+ assertEq(result.valid, false, 'missing dir: validation fails');
445
+ assert(result.issues.length > 0, 'missing dir: has issues');
446
+ assert(
447
+ result.issues.some(i => i.severity === 'fatal'),
448
+ 'missing dir: has fatal issue'
449
+ );
450
+ } finally {
451
+ cleanup(base);
452
+ }
453
+ }
454
+
455
+ // ─── Test 4: Duplicate phase numbers ───────────────────────────────────
456
+ console.log('\n=== Phase directory with duplicate numbers ===');
457
+ {
458
+ const base = createFixtureBase();
459
+ try {
460
+ const planning = createPlanningDir(base);
461
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
462
+
463
+ const phasesDir = join(planning, 'phases');
464
+ mkdirSync(join(phasesDir, '45-core-infrastructure'), { recursive: true });
465
+ mkdirSync(join(phasesDir, '45-logging-config'), { recursive: true });
466
+
467
+ writeFileSync(
468
+ join(phasesDir, '45-core-infrastructure', '45-01-PLAN.md'),
469
+ '# Core Plan\n\n<objective>Core infra</objective>'
470
+ );
471
+ writeFileSync(
472
+ join(phasesDir, '45-logging-config', '45-01-PLAN.md'),
473
+ '# Logging Plan\n\n<objective>Logging config</objective>'
474
+ );
475
+
476
+ const project = await parsePlanningDirectory(planning);
477
+
478
+ assert('45-core-infrastructure' in project.phases, 'dup nums: core-infrastructure phase present');
479
+ assert('45-logging-config' in project.phases, 'dup nums: logging-config phase present');
480
+ assertEq(project.phases['45-core-infrastructure']?.number, 45, 'dup nums: both have number 45 (a)');
481
+ assertEq(project.phases['45-logging-config']?.number, 45, 'dup nums: both have number 45 (b)');
482
+ } finally {
483
+ cleanup(base);
484
+ }
485
+ }
486
+
487
+ // ─── Test 5: XML-in-markdown plan parsing ──────────────────────────────
488
+ console.log('\n=== Plan file with XML-in-markdown ===');
489
+ {
490
+ const base = createFixtureBase();
491
+ try {
492
+ const planning = createPlanningDir(base);
493
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
494
+
495
+ const phaseDir = join(planning, 'phases', '29-auth-system');
496
+ mkdirSync(phaseDir, { recursive: true });
497
+ writeFileSync(join(phaseDir, '29-01-PLAN.md'), SAMPLE_PLAN_XML);
498
+
499
+ const project = await parsePlanningDirectory(planning);
500
+ const plan = project.phases['29-auth-system']?.plans?.['01'];
501
+
502
+ assert(plan !== undefined, 'xml plan: plan exists');
503
+ assert(plan?.objective?.includes('authentication') ?? false, 'xml plan: objective extracted');
504
+ assert((plan?.tasks?.length ?? 0) === 3, 'xml plan: 3 tasks extracted');
505
+ assert(plan?.tasks?.[0]?.includes('auth middleware') ?? false, 'xml plan: first task content');
506
+ assert(plan?.context?.includes('JWT') ?? false, 'xml plan: context extracted');
507
+ assert(plan?.verification?.includes('Login returns') ?? false, 'xml plan: verification extracted');
508
+ assert(plan?.successCriteria?.includes('endpoints respond') ?? false, 'xml plan: success criteria extracted');
509
+ } finally {
510
+ cleanup(base);
511
+ }
512
+ }
513
+
514
+ // ─── Test 6: Summary file with YAML frontmatter ───────────────────────
515
+ console.log('\n=== Summary file with YAML frontmatter ===');
516
+ {
517
+ const base = createFixtureBase();
518
+ try {
519
+ const planning = createPlanningDir(base);
520
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
521
+
522
+ const phaseDir = join(planning, 'phases', '29-auth-system');
523
+ mkdirSync(phaseDir, { recursive: true });
524
+ writeFileSync(join(phaseDir, '29-01-SUMMARY.md'), SAMPLE_SUMMARY);
525
+
526
+ const project = await parsePlanningDirectory(planning);
527
+ const summary = project.phases['29-auth-system']?.summaries?.['01'];
528
+
529
+ assert(summary !== undefined, 'summary fm: summary exists');
530
+ assertEq(summary?.frontmatter?.phase, '29-auth-system', 'summary fm: phase');
531
+ assertEq(summary?.frontmatter?.plan, '01', 'summary fm: plan');
532
+ assertEq(summary?.frontmatter?.subsystem, 'auth', 'summary fm: subsystem');
533
+ assertEq(summary?.frontmatter?.tags, ['authentication', 'security'], 'summary fm: tags');
534
+ assertEq(summary?.frontmatter?.provides, ['auth-middleware', 'jwt-validation'], 'summary fm: provides');
535
+ assertEq(summary?.frontmatter?.affects, ['api-routes'], 'summary fm: affects');
536
+ assertEq(summary?.frontmatter?.['tech-stack'], ['jsonwebtoken', 'express'], 'summary fm: tech-stack');
537
+ assertEq(summary?.frontmatter?.['key-files'], ['src/auth.ts', 'src/middleware/auth.ts'], 'summary fm: key-files');
538
+ assertEq(summary?.frontmatter?.['key-decisions'], ['Use RS256 for JWT signing', 'Store refresh tokens in DB'], 'summary fm: key-decisions');
539
+ assertEq(summary?.frontmatter?.['patterns-established'], ['Middleware-based auth'], 'summary fm: patterns-established');
540
+ assertEq(summary?.frontmatter?.duration, '2h', 'summary fm: duration');
541
+ assertEq(summary?.frontmatter?.completed, '2026-01-15', 'summary fm: completed');
542
+ } finally {
543
+ cleanup(base);
544
+ }
545
+ }
546
+
547
+ // ─── Test 7: Orphan summaries (no matching plan) ──────────────────────
548
+ console.log('\n=== Orphan summaries (no matching plan) ===');
549
+ {
550
+ const base = createFixtureBase();
551
+ try {
552
+ const planning = createPlanningDir(base);
553
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
554
+
555
+ const phaseDir = join(planning, 'phases', '45-logging-config');
556
+ mkdirSync(phaseDir, { recursive: true });
557
+
558
+ // Summaries without corresponding plans
559
+ writeFileSync(join(phaseDir, '45-04-SUMMARY.md'), `---
560
+ phase: "45-logging-config"
561
+ plan: "04"
562
+ subsystem: "logging"
563
+ ---
564
+
565
+ # 45-04 Summary
566
+
567
+ Orphan summary content.
568
+ `);
569
+ writeFileSync(join(phaseDir, '45-05-SUMMARY.md'), `---
570
+ phase: "45-logging-config"
571
+ plan: "05"
572
+ subsystem: "logging"
573
+ ---
574
+
575
+ # 45-05 Summary
576
+
577
+ Another orphan.
578
+ `);
579
+
580
+ const project = await parsePlanningDirectory(planning);
581
+ const phase = project.phases['45-logging-config'];
582
+
583
+ assert(phase !== undefined, 'orphan: phase exists');
584
+ assertEq(Object.keys(phase?.plans ?? {}).length, 0, 'orphan: no plans');
585
+ assert(Object.keys(phase?.summaries ?? {}).length >= 2, 'orphan: summaries preserved');
586
+ assert('04' in (phase?.summaries ?? {}), 'orphan: summary 04 present');
587
+ assert('05' in (phase?.summaries ?? {}), 'orphan: summary 05 present');
588
+ } finally {
589
+ cleanup(base);
590
+ }
591
+ }
592
+
593
+ // ─── Test 8: .archive/ directory skipped ──────────────────────────────
594
+ console.log('\n=== .archive/ directory → skipped by default ===');
595
+ {
596
+ const base = createFixtureBase();
597
+ try {
598
+ const planning = createPlanningDir(base);
599
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
600
+
601
+ // Normal phase
602
+ const phaseDir = join(planning, 'phases', '29-auth-system');
603
+ mkdirSync(phaseDir, { recursive: true });
604
+ writeFileSync(join(phaseDir, '29-01-PLAN.md'), SAMPLE_PLAN_XML);
605
+
606
+ // Archived phase (should be skipped)
607
+ const archiveDir = join(planning, '.archive', 'v2.5-deploy', '29-old-auth');
608
+ mkdirSync(archiveDir, { recursive: true });
609
+ writeFileSync(join(archiveDir, '29-01-PLAN.md'), '# Archived plan');
610
+
611
+ const project = await parsePlanningDirectory(planning);
612
+
613
+ assert('29-auth-system' in project.phases, 'archive: normal phase present');
614
+ // Archive phases should not appear in the phases map
615
+ assert(!Object.keys(project.phases).some(k => k.includes('old-auth')), 'archive: archived phase not present');
616
+ } finally {
617
+ cleanup(base);
618
+ }
619
+ }
620
+
621
+ // ─── Test 9: Quick tasks ──────────────────────────────────────────────
622
+ console.log('\n=== Quick tasks parsed ===');
623
+ {
624
+ const base = createFixtureBase();
625
+ try {
626
+ const planning = createPlanningDir(base);
627
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
628
+
629
+ // Quick task 1
630
+ const qt1 = join(planning, 'quick', '001-fix-login');
631
+ mkdirSync(qt1, { recursive: true });
632
+ writeFileSync(join(qt1, '001-PLAN.md'), SAMPLE_QUICK_PLAN);
633
+ writeFileSync(join(qt1, '001-SUMMARY.md'), SAMPLE_QUICK_SUMMARY);
634
+
635
+ // Quick task 2 (plan only, no summary)
636
+ const qt2 = join(planning, 'quick', '002-update-deps');
637
+ mkdirSync(qt2, { recursive: true });
638
+ writeFileSync(join(qt2, '002-PLAN.md'), '# 002: Update Dependencies\n\nUpdate all deps.');
639
+
640
+ const project = await parsePlanningDirectory(planning);
641
+
642
+ assertEq(project.quickTasks.length, 2, 'quick: 2 quick tasks');
643
+ assertEq(project.quickTasks[0]?.number, 1, 'quick: first task number');
644
+ assertEq(project.quickTasks[0]?.slug, 'fix-login', 'quick: first task slug');
645
+ assert(project.quickTasks[0]?.plan !== null, 'quick: first task has plan');
646
+ assert(project.quickTasks[0]?.summary !== null, 'quick: first task has summary');
647
+ assertEq(project.quickTasks[1]?.number, 2, 'quick: second task number');
648
+ assert(project.quickTasks[1]?.plan !== null, 'quick: second task has plan');
649
+ assertEq(project.quickTasks[1]?.summary, null, 'quick: second task has no summary');
650
+ } finally {
651
+ cleanup(base);
652
+ }
653
+ }
654
+
655
+ // ─── Test 10: Roadmap with milestone sections and <details> ────────────
656
+ console.log('\n=== Roadmap with milestone sections and <details> blocks ===');
657
+ {
658
+ const base = createFixtureBase();
659
+ try {
660
+ const planning = createPlanningDir(base);
661
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_MILESTONE_SECTIONED_ROADMAP);
662
+
663
+ const project = await parsePlanningDirectory(planning);
664
+
665
+ assert(project.roadmap !== null, 'ms roadmap: roadmap parsed');
666
+ assert((project.roadmap?.milestones?.length ?? 0) >= 2, 'ms roadmap: has milestone sections');
667
+
668
+ // Check collapsed milestone
669
+ const v20 = project.roadmap?.milestones?.find(m => m.id.includes('2.0'));
670
+ assert(v20 !== undefined, 'ms roadmap: v2.0 milestone found');
671
+ assertEq(v20?.collapsed, true, 'ms roadmap: v2.0 is collapsed');
672
+ assert((v20?.phases?.length ?? 0) >= 2, 'ms roadmap: v2.0 has phases');
673
+ assert(v20?.phases?.every(p => p.done) ?? false, 'ms roadmap: v2.0 phases all done');
674
+
675
+ // Check active milestone
676
+ const v25 = project.roadmap?.milestones?.find(m => m.id.includes('2.5'));
677
+ assert(v25 !== undefined, 'ms roadmap: v2.5 milestone found');
678
+ assertEq(v25?.collapsed, false, 'ms roadmap: v2.5 is not collapsed');
679
+ assert((v25?.phases?.length ?? 0) >= 3, 'ms roadmap: v2.5 has phases');
680
+
681
+ // Check completion state
682
+ const phase29 = v25?.phases?.find(p => p.number === 29);
683
+ assert(phase29?.done === true, 'ms roadmap: phase 29 is done');
684
+ const phase30 = v25?.phases?.find(p => p.number === 30);
685
+ assert(phase30?.done === false, 'ms roadmap: phase 30 is not done');
686
+ } finally {
687
+ cleanup(base);
688
+ }
689
+ }
690
+
691
+ // ─── Test 11: Non-standard phase files → extra files ──────────────────
692
+ console.log('\n=== Non-standard phase files → collected as extra files ===');
693
+ {
694
+ const base = createFixtureBase();
695
+ try {
696
+ const planning = createPlanningDir(base);
697
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
698
+
699
+ const phaseDir = join(planning, 'phases', '36-attachment-system');
700
+ mkdirSync(phaseDir, { recursive: true });
701
+ writeFileSync(join(phaseDir, '36-01-PLAN.md'), '<objective>Attachments</objective>');
702
+ writeFileSync(join(phaseDir, 'BASELINE.md'), '# Baseline\n\nBaseline measurements.');
703
+ writeFileSync(join(phaseDir, 'BUNDLE-ANALYSIS.md'), '# Bundle Analysis\n\nResults.');
704
+ writeFileSync(join(phaseDir, 'depcheck-results.txt'), 'unused: pkg-a, pkg-b');
705
+
706
+ const project = await parsePlanningDirectory(planning);
707
+ const phase = project.phases['36-attachment-system'];
708
+
709
+ assert(phase !== undefined, 'extra: phase exists');
710
+ assert((phase?.extraFiles?.length ?? 0) >= 3, 'extra: non-standard files collected');
711
+ assert(
712
+ phase?.extraFiles?.some(f => f.fileName === 'BASELINE.md') ?? false,
713
+ 'extra: BASELINE.md collected'
714
+ );
715
+ assert(
716
+ phase?.extraFiles?.some(f => f.fileName === 'BUNDLE-ANALYSIS.md') ?? false,
717
+ 'extra: BUNDLE-ANALYSIS.md collected'
718
+ );
719
+ assert(
720
+ phase?.extraFiles?.some(f => f.fileName === 'depcheck-results.txt') ?? false,
721
+ 'extra: depcheck-results.txt collected'
722
+ );
723
+ } finally {
724
+ cleanup(base);
725
+ }
726
+ }
727
+
728
+ // ─── Test 12: Validation — missing ROADMAP.md → warning (not fatal) ───
729
+ console.log('\n=== Validation: missing ROADMAP.md → warning (not fatal) ===');
730
+ {
731
+ const base = createFixtureBase();
732
+ try {
733
+ const planning = createPlanningDir(base);
734
+ // Only PROJECT.md, no ROADMAP.md
735
+ writeFileSync(join(planning, 'PROJECT.md'), SAMPLE_PROJECT);
736
+
737
+ const result = await validatePlanningDirectory(planning);
738
+
739
+ assertEq(result.valid, true, 'no roadmap: validation still passes');
740
+ assert(
741
+ result.issues.some(i => i.severity === 'warning' && i.file.includes('ROADMAP')),
742
+ 'no roadmap: warning issue mentions ROADMAP'
743
+ );
744
+ } finally {
745
+ cleanup(base);
746
+ }
747
+ }
748
+
749
+ // ─── Test 13: Validation — missing PROJECT.md → warning ───────────────
750
+ console.log('\n=== Validation: missing PROJECT.md → warning ===');
751
+ {
752
+ const base = createFixtureBase();
753
+ try {
754
+ const planning = createPlanningDir(base);
755
+ writeFileSync(join(planning, 'ROADMAP.md'), SAMPLE_ROADMAP);
756
+ // No PROJECT.md
757
+
758
+ const result = await validatePlanningDirectory(planning);
759
+
760
+ assertEq(result.valid, true, 'no project: validation passes (warning only)');
761
+ assert(
762
+ result.issues.some(i => i.severity === 'warning' && i.file.includes('PROJECT')),
763
+ 'no project: warning issue mentions PROJECT'
764
+ );
765
+ } finally {
766
+ cleanup(base);
767
+ }
768
+ }
769
+
770
+ // ═════════════════════════════════════════════════════════════════════════
771
+ // Results
772
+ // ═════════════════════════════════════════════════════════════════════════
773
+
774
+ console.log(`\n${'='.repeat(40)}`);
775
+ console.log(`Results: ${passed} passed, ${failed} failed`);
776
+ if (failed > 0) {
777
+ process.exit(1);
778
+ } else {
779
+ console.log('All tests passed ✓');
780
+ }
781
+ }
782
+
783
+ main().catch((error) => {
784
+ console.error(error);
785
+ process.exit(1);
786
+ });