@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,346 @@
1
+ // Migration transformer — converts parsed PlanningProject into GSDProject.
2
+ // Pure function: no I/O, no side effects, no imports outside migrate/.
3
+
4
+ import type {
5
+ PlanningProject,
6
+ PlanningPhase,
7
+ PlanningPlan,
8
+ PlanningSummary,
9
+ PlanningRoadmapEntry,
10
+ PlanningRoadmapMilestone,
11
+ PlanningResearch,
12
+ PlanningRequirement,
13
+ GSDProject,
14
+ GSDMilestone,
15
+ GSDSlice,
16
+ GSDTask,
17
+ GSDRequirement,
18
+ GSDSliceSummaryData,
19
+ GSDTaskSummaryData,
20
+ GSDBoundaryEntry,
21
+ } from './types.ts';
22
+
23
+ // ─── Helpers ───────────────────────────────────────────────────────────────
24
+
25
+ function padId(prefix: string, n: number, width = 2): string {
26
+ return `${prefix}${String(n).padStart(width, '0')}`;
27
+ }
28
+
29
+ function milestoneId(n: number): string {
30
+ return padId('M', n, 3);
31
+ }
32
+
33
+ function kebabToTitle(slug: string): string {
34
+ return slug
35
+ .split('-')
36
+ .map((w) => w.charAt(0).toUpperCase() + w.slice(1))
37
+ .join(' ');
38
+ }
39
+
40
+ function firstSentence(text: string): string {
41
+ const trimmed = text.trim();
42
+ const match = trimmed.match(/^[^.!?]*[.!?]/);
43
+ return match ? match[0].trim() : trimmed;
44
+ }
45
+
46
+ /** Preferred research ordering for consolidation. */
47
+ const RESEARCH_ORDER = ['SUMMARY.md', 'ARCHITECTURE.md', 'STACK.md', 'FEATURES.md', 'PITFALLS.md'];
48
+
49
+ function sortResearch(files: PlanningResearch[]): PlanningResearch[] {
50
+ return [...files].sort((a, b) => {
51
+ const ai = RESEARCH_ORDER.indexOf(a.fileName);
52
+ const bi = RESEARCH_ORDER.indexOf(b.fileName);
53
+ const aw = ai === -1 ? RESEARCH_ORDER.length : ai;
54
+ const bw = bi === -1 ? RESEARCH_ORDER.length : bi;
55
+ if (aw !== bw) return aw - bw;
56
+ return a.fileName.localeCompare(b.fileName);
57
+ });
58
+ }
59
+
60
+ function consolidateResearch(files: PlanningResearch[]): string | null {
61
+ if (files.length === 0) return null;
62
+ return sortResearch(files)
63
+ .map((f) => f.content.trim())
64
+ .join('\n\n');
65
+ }
66
+
67
+ // ─── Task Mapping ──────────────────────────────────────────────────────────
68
+
69
+ function buildTaskSummary(summary: PlanningSummary): GSDTaskSummaryData {
70
+ return {
71
+ completedAt: summary.frontmatter.completed ?? '',
72
+ provides: summary.frontmatter.provides ?? [],
73
+ keyFiles: summary.frontmatter['key-files'] ?? [],
74
+ duration: summary.frontmatter.duration ?? '',
75
+ whatHappened: summary.body?.trim() ?? '',
76
+ };
77
+ }
78
+
79
+ function mapTask(plan: PlanningPlan, index: number, summaries: Record<string, PlanningSummary>): GSDTask {
80
+ const summary = summaries[plan.planNumber];
81
+ const done = summary !== undefined;
82
+ return {
83
+ id: padId('T', index + 1),
84
+ title: buildTaskTitle(plan),
85
+ description: plan.objective ?? '',
86
+ done,
87
+ estimate: done ? (summary.frontmatter.duration ?? '') : '',
88
+ files: plan.frontmatter.files_modified ?? [],
89
+ mustHaves: plan.frontmatter.must_haves?.truths ?? [],
90
+ summary: done ? buildTaskSummary(summary) : null,
91
+ };
92
+ }
93
+
94
+ function buildTaskTitle(plan: PlanningPlan): string {
95
+ const fm = plan.frontmatter;
96
+ if (fm.phase && fm.plan) {
97
+ return `${fm.phase} ${fm.plan}`;
98
+ }
99
+ return `Plan ${plan.planNumber}`;
100
+ }
101
+
102
+ // ─── Slice Mapping ─────────────────────────────────────────────────────────
103
+
104
+ function buildSliceSummary(phase: PlanningPhase): GSDSliceSummaryData | null {
105
+ // Aggregate from all summaries in the phase
106
+ const summaryEntries = Object.values(phase.summaries);
107
+ if (summaryEntries.length === 0) return null;
108
+
109
+ const provides: string[] = [];
110
+ const keyFiles: string[] = [];
111
+ const keyDecisions: string[] = [];
112
+ const patternsEstablished: string[] = [];
113
+ let lastCompleted = '';
114
+ let totalDuration = '';
115
+ const bodies: string[] = [];
116
+
117
+ for (const s of summaryEntries) {
118
+ provides.push(...(s.frontmatter.provides ?? []));
119
+ keyFiles.push(...(s.frontmatter['key-files'] ?? []));
120
+ keyDecisions.push(...(s.frontmatter['key-decisions'] ?? []));
121
+ patternsEstablished.push(...(s.frontmatter['patterns-established'] ?? []));
122
+ if (s.frontmatter.completed) lastCompleted = s.frontmatter.completed;
123
+ if (s.frontmatter.duration) totalDuration = s.frontmatter.duration;
124
+ if (s.body?.trim()) bodies.push(s.body.trim());
125
+ }
126
+
127
+ return {
128
+ completedAt: lastCompleted,
129
+ provides,
130
+ keyFiles,
131
+ keyDecisions,
132
+ patternsEstablished,
133
+ duration: totalDuration,
134
+ whatHappened: bodies.join('\n\n'),
135
+ };
136
+ }
137
+
138
+ function deriveDemo(phase: PlanningPhase, slug: string): string {
139
+ // First plan's objective, first sentence
140
+ const planNumbers = Object.keys(phase.plans).sort((a, b) => Number(a) - Number(b));
141
+ if (planNumbers.length > 0) {
142
+ const firstPlan = phase.plans[planNumbers[0]];
143
+ if (firstPlan?.objective) {
144
+ return firstSentence(firstPlan.objective);
145
+ }
146
+ }
147
+ return `unit tests prove ${slug} works`;
148
+ }
149
+
150
+ function mapSlice(
151
+ phase: PlanningPhase | undefined,
152
+ entry: PlanningRoadmapEntry,
153
+ index: number,
154
+ prevSliceId: string | null,
155
+ ): GSDSlice {
156
+ const sliceId = padId('S', index + 1);
157
+ const slug = phase?.slug ?? entry.title;
158
+ const demo = phase ? deriveDemo(phase, slug) : `unit tests prove ${entry.title} works`;
159
+
160
+ let tasks: GSDTask[] = [];
161
+ if (phase) {
162
+ const planNumbers = Object.keys(phase.plans).sort((a, b) => Number(a) - Number(b));
163
+ tasks = planNumbers.map((pn, i) => mapTask(phase.plans[pn], i, phase.summaries));
164
+ }
165
+
166
+ const done = entry.done;
167
+ const sliceSummary = done && phase ? buildSliceSummary(phase) : null;
168
+
169
+ return {
170
+ id: sliceId,
171
+ title: kebabToTitle(slug),
172
+ risk: 'medium',
173
+ depends: prevSliceId ? [prevSliceId] : [],
174
+ done,
175
+ demo,
176
+ goal: demo,
177
+ tasks,
178
+ research: phase ? consolidateResearch(phase.research) : null,
179
+ summary: sliceSummary,
180
+ };
181
+ }
182
+
183
+ // ─── Milestone Building ───────────────────────────────────────────────────
184
+
185
+ function findPhase(phases: Record<string, PlanningPhase>, phaseNumber: number, entryTitle?: string): PlanningPhase | undefined {
186
+ const matches = Object.values(phases).filter((p) => p.number === phaseNumber);
187
+ if (matches.length <= 1) return matches[0];
188
+ // Multiple phases with the same number — try to match by title/slug similarity
189
+ if (entryTitle) {
190
+ const normalizedTitle = entryTitle.toLowerCase().replace(/[^a-z0-9]+/g, ' ').trim();
191
+ const best = matches.find((p) => {
192
+ const normalizedSlug = p.slug.replace(/-/g, ' ').toLowerCase();
193
+ return normalizedSlug === normalizedTitle || normalizedTitle.includes(normalizedSlug) || normalizedSlug.includes(normalizedTitle);
194
+ });
195
+ if (best) return best;
196
+ }
197
+ return matches[0];
198
+ }
199
+
200
+ function buildMilestoneFromEntries(
201
+ id: string,
202
+ title: string,
203
+ entries: PlanningRoadmapEntry[],
204
+ phases: Record<string, PlanningPhase>,
205
+ research: PlanningResearch[],
206
+ ): GSDMilestone {
207
+ // Sort entries by phase number (float sort)
208
+ const sorted = [...entries].sort((a, b) => a.number - b.number);
209
+
210
+ const slices: GSDSlice[] = [];
211
+ for (let i = 0; i < sorted.length; i++) {
212
+ const entry = sorted[i];
213
+ const phase = findPhase(phases, entry.number, entry.title);
214
+ const prevId = i > 0 ? slices[i - 1].id : null;
215
+ slices.push(mapSlice(phase, entry, i, prevId));
216
+ }
217
+
218
+ return {
219
+ id,
220
+ title,
221
+ vision: '',
222
+ successCriteria: [],
223
+ slices,
224
+ research: consolidateResearch(research),
225
+ boundaryMap: [],
226
+ };
227
+ }
228
+
229
+ // ─── Requirements Mapping ──────────────────────────────────────────────────
230
+
231
+ const VALID_STATUSES = new Set(['active', 'validated', 'deferred']);
232
+ const COMPLETE_ALIASES = new Set(['complete', 'completed', 'done', 'shipped']);
233
+
234
+ function normalizeStatus(status: string): 'active' | 'validated' | 'deferred' {
235
+ const lower = status.toLowerCase().trim();
236
+ if (VALID_STATUSES.has(lower)) return lower as 'active' | 'validated' | 'deferred';
237
+ if (COMPLETE_ALIASES.has(lower)) return 'validated';
238
+ return 'active';
239
+ }
240
+
241
+ function mapRequirements(reqs: PlanningRequirement[]): GSDRequirement[] {
242
+ let autoId = 0;
243
+ return reqs.map((req) => {
244
+ autoId++;
245
+ return {
246
+ id: req.id && req.id.trim() !== '' ? req.id : padId('R', autoId, 3),
247
+ title: req.title,
248
+ class: 'core-capability',
249
+ status: normalizeStatus(req.status),
250
+ description: req.description,
251
+ source: 'inferred',
252
+ primarySlice: 'none yet',
253
+ };
254
+ });
255
+ }
256
+
257
+ // ─── Project-Level Derivation ──────────────────────────────────────────────
258
+
259
+ function deriveVision(parsed: PlanningProject): string {
260
+ // Try first non-heading line from PROJECT.md
261
+ if (parsed.project) {
262
+ const lines = parsed.project.split('\n');
263
+ for (const line of lines) {
264
+ const trimmed = line.trim();
265
+ if (trimmed && !trimmed.startsWith('#')) {
266
+ return firstSentence(trimmed);
267
+ }
268
+ }
269
+ }
270
+ // Fallback: roadmap title
271
+ if (parsed.roadmap) {
272
+ if (parsed.roadmap.milestones.length > 0) {
273
+ return parsed.roadmap.milestones[0].title;
274
+ }
275
+ }
276
+ return 'Project migration from .planning format';
277
+ }
278
+
279
+ function deriveDecisions(parsed: PlanningProject): string {
280
+ // Extract key decisions from phase summaries if available
281
+ const decisions: string[] = [];
282
+ for (const phase of Object.values(parsed.phases)) {
283
+ for (const summary of Object.values(phase.summaries)) {
284
+ const kd = summary.frontmatter['key-decisions'] ?? [];
285
+ decisions.push(...kd);
286
+ }
287
+ }
288
+ if (decisions.length === 0) return '';
289
+ return decisions.map((d) => `- ${d}`).join('\n');
290
+ }
291
+
292
+ // ─── Main Entry Point ──────────────────────────────────────────────────────
293
+
294
+ export function transformToGSD(parsed: PlanningProject): GSDProject {
295
+ const milestones: GSDMilestone[] = [];
296
+
297
+ const roadmap = parsed.roadmap;
298
+ const isMultiMilestone = roadmap !== null && roadmap.milestones.length > 0;
299
+ const hasFlatPhases = roadmap !== null && roadmap.phases.length > 0;
300
+
301
+ if (isMultiMilestone) {
302
+ // Multi-milestone mode: each roadmap milestone section → one GSDMilestone
303
+ for (let mi = 0; mi < roadmap!.milestones.length; mi++) {
304
+ const rm = roadmap!.milestones[mi];
305
+ milestones.push(
306
+ buildMilestoneFromEntries(
307
+ milestoneId(mi + 1),
308
+ rm.title,
309
+ rm.phases,
310
+ parsed.phases,
311
+ mi === 0 ? parsed.research : [],
312
+ ),
313
+ );
314
+ }
315
+ } else if (hasFlatPhases) {
316
+ // Single-milestone mode from roadmap phases
317
+ milestones.push(
318
+ buildMilestoneFromEntries('M001', 'Migration', roadmap!.phases, parsed.phases, parsed.research),
319
+ );
320
+ } else {
321
+ // Null/empty roadmap fallback: use filesystem phases, all not-done
322
+ const fsPhases = Object.values(parsed.phases).sort((a, b) => a.number - b.number);
323
+ const entries: PlanningRoadmapEntry[] = fsPhases.map((p) => ({
324
+ number: p.number,
325
+ title: p.slug,
326
+ done: false,
327
+ raw: '',
328
+ }));
329
+ milestones.push(
330
+ buildMilestoneFromEntries('M001', 'Migration', entries, parsed.phases, parsed.research),
331
+ );
332
+ }
333
+
334
+ // Set vision on first milestone (or all if multi)
335
+ const vision = deriveVision(parsed);
336
+ for (const m of milestones) {
337
+ if (!m.vision) m.vision = vision;
338
+ }
339
+
340
+ return {
341
+ milestones,
342
+ projectContent: parsed.project ?? '',
343
+ requirements: mapRequirements(parsed.requirements),
344
+ decisionsContent: deriveDecisions(parsed),
345
+ };
346
+ }