@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.
- package/LICENSE +21 -0
- package/README.md +453 -0
- package/dist/app-paths.d.ts +4 -0
- package/dist/app-paths.js +6 -0
- package/dist/cli.d.ts +1 -0
- package/dist/cli.js +269 -0
- package/dist/loader.d.ts +2 -0
- package/dist/loader.js +70 -0
- package/dist/logo.d.ts +16 -0
- package/dist/logo.js +25 -0
- package/dist/onboarding.d.ts +43 -0
- package/dist/onboarding.js +418 -0
- package/dist/pi-migration.d.ts +14 -0
- package/dist/pi-migration.js +57 -0
- package/dist/resource-loader.d.ts +22 -0
- package/dist/resource-loader.js +60 -0
- package/dist/tool-bootstrap.d.ts +4 -0
- package/dist/tool-bootstrap.js +74 -0
- package/dist/wizard.d.ts +7 -0
- package/dist/wizard.js +25 -0
- package/package.json +60 -0
- package/patches/@mariozechner+pi-coding-agent+0.57.1.patch +108 -0
- package/patches/@mariozechner+pi-tui+0.57.1.patch +47 -0
- package/pkg/dist/modes/interactive/theme/dark.json +85 -0
- package/pkg/dist/modes/interactive/theme/light.json +84 -0
- package/pkg/dist/modes/interactive/theme/theme-schema.json +335 -0
- package/pkg/dist/modes/interactive/theme/theme.d.ts +78 -0
- package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/pkg/dist/modes/interactive/theme/theme.js +949 -0
- package/pkg/dist/modes/interactive/theme/theme.js.map +1 -0
- package/pkg/package.json +8 -0
- package/scripts/postinstall.js +127 -0
- package/src/resources/GSD-WORKFLOW.md +661 -0
- package/src/resources/agents/researcher.md +29 -0
- package/src/resources/agents/scout.md +56 -0
- package/src/resources/agents/worker.md +31 -0
- package/src/resources/extensions/ask-user-questions.ts +249 -0
- package/src/resources/extensions/bg-shell/index.ts +2808 -0
- package/src/resources/extensions/browser-tools/BROWSER-TOOLS-V2-PROPOSAL.md +1277 -0
- package/src/resources/extensions/browser-tools/core.js +1057 -0
- package/src/resources/extensions/browser-tools/index.ts +4989 -0
- package/src/resources/extensions/browser-tools/package.json +20 -0
- package/src/resources/extensions/context7/index.ts +428 -0
- package/src/resources/extensions/context7/package.json +11 -0
- package/src/resources/extensions/get-secrets-from-user.ts +352 -0
- package/src/resources/extensions/google-search/index.ts +323 -0
- package/src/resources/extensions/google-search/package.json +9 -0
- package/src/resources/extensions/gsd/activity-log.ts +69 -0
- package/src/resources/extensions/gsd/auto.ts +2744 -0
- package/src/resources/extensions/gsd/commands.ts +313 -0
- package/src/resources/extensions/gsd/crash-recovery.ts +85 -0
- package/src/resources/extensions/gsd/dashboard-overlay.ts +521 -0
- package/src/resources/extensions/gsd/docs/preferences-reference.md +176 -0
- package/src/resources/extensions/gsd/doctor.ts +690 -0
- package/src/resources/extensions/gsd/files.ts +732 -0
- package/src/resources/extensions/gsd/git-service.ts +597 -0
- package/src/resources/extensions/gsd/gitignore.ts +168 -0
- package/src/resources/extensions/gsd/guided-flow.ts +817 -0
- package/src/resources/extensions/gsd/index.ts +558 -0
- package/src/resources/extensions/gsd/metrics.ts +374 -0
- package/src/resources/extensions/gsd/migrate/command.ts +218 -0
- package/src/resources/extensions/gsd/migrate/index.ts +42 -0
- package/src/resources/extensions/gsd/migrate/parser.ts +323 -0
- package/src/resources/extensions/gsd/migrate/parsers.ts +624 -0
- package/src/resources/extensions/gsd/migrate/preview.ts +48 -0
- package/src/resources/extensions/gsd/migrate/transformer.ts +346 -0
- package/src/resources/extensions/gsd/migrate/types.ts +370 -0
- package/src/resources/extensions/gsd/migrate/validator.ts +55 -0
- package/src/resources/extensions/gsd/migrate/writer.ts +539 -0
- package/src/resources/extensions/gsd/observability-validator.ts +408 -0
- package/src/resources/extensions/gsd/package.json +11 -0
- package/src/resources/extensions/gsd/paths.ts +308 -0
- package/src/resources/extensions/gsd/preferences.ts +757 -0
- package/src/resources/extensions/gsd/prompt-loader.ts +50 -0
- package/src/resources/extensions/gsd/prompts/complete-milestone.md +25 -0
- package/src/resources/extensions/gsd/prompts/complete-slice.md +29 -0
- package/src/resources/extensions/gsd/prompts/discuss.md +189 -0
- package/src/resources/extensions/gsd/prompts/doctor-heal.md +29 -0
- package/src/resources/extensions/gsd/prompts/execute-task.md +61 -0
- package/src/resources/extensions/gsd/prompts/guided-complete-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/guided-discuss-milestone.md +3 -0
- package/src/resources/extensions/gsd/prompts/guided-discuss-slice.md +59 -0
- package/src/resources/extensions/gsd/prompts/guided-execute-task.md +1 -0
- package/src/resources/extensions/gsd/prompts/guided-plan-milestone.md +23 -0
- package/src/resources/extensions/gsd/prompts/guided-plan-slice.md +1 -0
- package/src/resources/extensions/gsd/prompts/guided-research-slice.md +11 -0
- package/src/resources/extensions/gsd/prompts/guided-resume-task.md +1 -0
- package/src/resources/extensions/gsd/prompts/plan-milestone.md +65 -0
- package/src/resources/extensions/gsd/prompts/plan-slice.md +51 -0
- package/src/resources/extensions/gsd/prompts/queue.md +85 -0
- package/src/resources/extensions/gsd/prompts/reassess-roadmap.md +48 -0
- package/src/resources/extensions/gsd/prompts/replan-slice.md +39 -0
- package/src/resources/extensions/gsd/prompts/research-milestone.md +37 -0
- package/src/resources/extensions/gsd/prompts/research-slice.md +28 -0
- package/src/resources/extensions/gsd/prompts/review-migration.md +66 -0
- package/src/resources/extensions/gsd/prompts/run-uat.md +109 -0
- package/src/resources/extensions/gsd/prompts/system.md +187 -0
- package/src/resources/extensions/gsd/prompts/worktree-merge.md +123 -0
- package/src/resources/extensions/gsd/session-forensics.ts +487 -0
- package/src/resources/extensions/gsd/skill-discovery.ts +137 -0
- package/src/resources/extensions/gsd/state.ts +460 -0
- package/src/resources/extensions/gsd/templates/context.md +76 -0
- package/src/resources/extensions/gsd/templates/decisions.md +8 -0
- package/src/resources/extensions/gsd/templates/milestone-summary.md +73 -0
- package/src/resources/extensions/gsd/templates/plan.md +131 -0
- package/src/resources/extensions/gsd/templates/preferences.md +24 -0
- package/src/resources/extensions/gsd/templates/project.md +31 -0
- package/src/resources/extensions/gsd/templates/reassessment.md +28 -0
- package/src/resources/extensions/gsd/templates/requirements.md +81 -0
- package/src/resources/extensions/gsd/templates/research.md +46 -0
- package/src/resources/extensions/gsd/templates/roadmap.md +118 -0
- package/src/resources/extensions/gsd/templates/slice-context.md +58 -0
- package/src/resources/extensions/gsd/templates/slice-summary.md +99 -0
- package/src/resources/extensions/gsd/templates/state.md +19 -0
- package/src/resources/extensions/gsd/templates/task-plan.md +52 -0
- package/src/resources/extensions/gsd/templates/task-summary.md +57 -0
- package/src/resources/extensions/gsd/templates/uat.md +54 -0
- package/src/resources/extensions/gsd/tests/activity-log-prune.test.ts +327 -0
- package/src/resources/extensions/gsd/tests/auto-preflight.test.ts +56 -0
- package/src/resources/extensions/gsd/tests/auto-supervisor.test.mjs +53 -0
- package/src/resources/extensions/gsd/tests/complete-milestone.test.ts +225 -0
- package/src/resources/extensions/gsd/tests/cost-projection.test.ts +160 -0
- package/src/resources/extensions/gsd/tests/derive-state-deps.test.ts +341 -0
- package/src/resources/extensions/gsd/tests/derive-state.test.ts +689 -0
- package/src/resources/extensions/gsd/tests/discuss-prompt.test.ts +38 -0
- package/src/resources/extensions/gsd/tests/doctor.test.ts +505 -0
- package/src/resources/extensions/gsd/tests/git-service.test.ts +1313 -0
- package/src/resources/extensions/gsd/tests/idle-recovery.test.ts +308 -0
- package/src/resources/extensions/gsd/tests/metrics-io.test.ts +201 -0
- package/src/resources/extensions/gsd/tests/metrics.test.ts +217 -0
- package/src/resources/extensions/gsd/tests/migrate-command.test.ts +390 -0
- package/src/resources/extensions/gsd/tests/migrate-parser.test.ts +786 -0
- package/src/resources/extensions/gsd/tests/migrate-transformer.test.ts +657 -0
- package/src/resources/extensions/gsd/tests/migrate-validator-parsers.test.ts +443 -0
- package/src/resources/extensions/gsd/tests/migrate-writer-integration.test.ts +318 -0
- package/src/resources/extensions/gsd/tests/migrate-writer.test.ts +420 -0
- package/src/resources/extensions/gsd/tests/must-have-parser.test.ts +309 -0
- package/src/resources/extensions/gsd/tests/parsers.test.ts +1351 -0
- package/src/resources/extensions/gsd/tests/plan-milestone.test.ts +163 -0
- package/src/resources/extensions/gsd/tests/plan-quality-validator.test.ts +386 -0
- package/src/resources/extensions/gsd/tests/reassess-prompt.test.ts +171 -0
- package/src/resources/extensions/gsd/tests/remote-questions.test.ts +155 -0
- package/src/resources/extensions/gsd/tests/remote-status.test.ts +99 -0
- package/src/resources/extensions/gsd/tests/replan-slice.test.ts +521 -0
- package/src/resources/extensions/gsd/tests/requirements.test.ts +125 -0
- package/src/resources/extensions/gsd/tests/resolve-ts-hooks.mjs +34 -0
- package/src/resources/extensions/gsd/tests/resolve-ts.mjs +11 -0
- package/src/resources/extensions/gsd/tests/run-uat.test.ts +348 -0
- package/src/resources/extensions/gsd/tests/unit-runtime.test.ts +247 -0
- package/src/resources/extensions/gsd/tests/workflow-config.test.mjs +53 -0
- package/src/resources/extensions/gsd/tests/workspace-index.test.ts +94 -0
- package/src/resources/extensions/gsd/tests/worktree-integration.test.ts +253 -0
- package/src/resources/extensions/gsd/tests/worktree-manager.test.ts +160 -0
- package/src/resources/extensions/gsd/tests/worktree.test.ts +264 -0
- package/src/resources/extensions/gsd/types.ts +159 -0
- package/src/resources/extensions/gsd/unit-runtime.ts +184 -0
- package/src/resources/extensions/gsd/workspace-index.ts +203 -0
- package/src/resources/extensions/gsd/worktree-command.ts +845 -0
- package/src/resources/extensions/gsd/worktree-manager.ts +392 -0
- package/src/resources/extensions/gsd/worktree.ts +183 -0
- package/src/resources/extensions/mac-tools/index.ts +852 -0
- package/src/resources/extensions/mac-tools/swift-cli/Package.swift +22 -0
- package/src/resources/extensions/mac-tools/swift-cli/Sources/main.swift +1318 -0
- package/src/resources/extensions/mcporter/index.ts +429 -0
- package/src/resources/extensions/remote-questions/config.ts +81 -0
- package/src/resources/extensions/remote-questions/discord-adapter.ts +128 -0
- package/src/resources/extensions/remote-questions/format.ts +163 -0
- package/src/resources/extensions/remote-questions/manager.ts +192 -0
- package/src/resources/extensions/remote-questions/remote-command.ts +307 -0
- package/src/resources/extensions/remote-questions/slack-adapter.ts +92 -0
- package/src/resources/extensions/remote-questions/status.ts +31 -0
- package/src/resources/extensions/remote-questions/store.ts +77 -0
- package/src/resources/extensions/remote-questions/types.ts +75 -0
- package/src/resources/extensions/search-the-web/cache.ts +78 -0
- package/src/resources/extensions/search-the-web/command-search-provider.ts +95 -0
- package/src/resources/extensions/search-the-web/format.ts +258 -0
- package/src/resources/extensions/search-the-web/http.ts +238 -0
- package/src/resources/extensions/search-the-web/index.ts +65 -0
- package/src/resources/extensions/search-the-web/native-search.ts +157 -0
- package/src/resources/extensions/search-the-web/provider.ts +118 -0
- package/src/resources/extensions/search-the-web/tavily.ts +116 -0
- package/src/resources/extensions/search-the-web/tool-fetch-page.ts +519 -0
- package/src/resources/extensions/search-the-web/tool-llm-context.ts +561 -0
- package/src/resources/extensions/search-the-web/tool-search.ts +576 -0
- package/src/resources/extensions/search-the-web/url-utils.ts +91 -0
- package/src/resources/extensions/shared/confirm-ui.ts +126 -0
- package/src/resources/extensions/shared/interview-ui.ts +613 -0
- package/src/resources/extensions/shared/next-action-ui.ts +197 -0
- package/src/resources/extensions/shared/progress-widget.ts +282 -0
- package/src/resources/extensions/shared/terminal.ts +23 -0
- package/src/resources/extensions/shared/thinking-widget.ts +107 -0
- package/src/resources/extensions/shared/ui.ts +400 -0
- package/src/resources/extensions/shared/wizard-ui.ts +551 -0
- package/src/resources/extensions/slash-commands/audit.ts +88 -0
- package/src/resources/extensions/slash-commands/clear.ts +10 -0
- package/src/resources/extensions/slash-commands/create-extension.ts +297 -0
- package/src/resources/extensions/slash-commands/create-slash-command.ts +234 -0
- package/src/resources/extensions/slash-commands/index.ts +12 -0
- package/src/resources/extensions/subagent/agents.ts +126 -0
- package/src/resources/extensions/subagent/index.ts +1020 -0
- package/src/resources/extensions/voice/index.ts +195 -0
- package/src/resources/extensions/voice/speech-recognizer.swift +154 -0
- package/src/resources/skills/debug-like-expert/SKILL.md +231 -0
- package/src/resources/skills/debug-like-expert/references/debugging-mindset.md +253 -0
- package/src/resources/skills/debug-like-expert/references/hypothesis-testing.md +373 -0
- package/src/resources/skills/debug-like-expert/references/investigation-techniques.md +337 -0
- package/src/resources/skills/debug-like-expert/references/verification-patterns.md +425 -0
- package/src/resources/skills/debug-like-expert/references/when-to-research.md +361 -0
- package/src/resources/skills/frontend-design/SKILL.md +45 -0
- package/src/resources/skills/swiftui/SKILL.md +208 -0
- package/src/resources/skills/swiftui/references/animations.md +921 -0
- package/src/resources/skills/swiftui/references/architecture.md +1561 -0
- package/src/resources/skills/swiftui/references/layout-system.md +1186 -0
- package/src/resources/skills/swiftui/references/navigation.md +1492 -0
- package/src/resources/skills/swiftui/references/networking-async.md +214 -0
- package/src/resources/skills/swiftui/references/performance.md +1706 -0
- package/src/resources/skills/swiftui/references/platform-integration.md +204 -0
- package/src/resources/skills/swiftui/references/state-management.md +1443 -0
- package/src/resources/skills/swiftui/references/swiftdata.md +297 -0
- package/src/resources/skills/swiftui/references/testing-debugging.md +247 -0
- package/src/resources/skills/swiftui/references/uikit-appkit-interop.md +218 -0
- package/src/resources/skills/swiftui/workflows/add-feature.md +191 -0
- package/src/resources/skills/swiftui/workflows/build-new-app.md +311 -0
- package/src/resources/skills/swiftui/workflows/debug-swiftui.md +192 -0
- package/src/resources/skills/swiftui/workflows/optimize-performance.md +197 -0
- package/src/resources/skills/swiftui/workflows/ship-app.md +203 -0
- package/src/resources/skills/swiftui/workflows/write-tests.md +235 -0
|
@@ -0,0 +1,657 @@
|
|
|
1
|
+
// Migration transformer test suite
|
|
2
|
+
// Tests for transforming parsed PlanningProject into GSDProject structures.
|
|
3
|
+
// Uses synthetic in-memory fixtures — no filesystem needed.
|
|
4
|
+
// Transformer is pure: PlanningProject → GSDProject.
|
|
5
|
+
|
|
6
|
+
import { transformToGSD } from '../migrate/transformer.ts';
|
|
7
|
+
import type {
|
|
8
|
+
PlanningProject,
|
|
9
|
+
PlanningPhase,
|
|
10
|
+
PlanningPlan,
|
|
11
|
+
PlanningSummary,
|
|
12
|
+
PlanningRoadmap,
|
|
13
|
+
PlanningRoadmapEntry,
|
|
14
|
+
PlanningRoadmapMilestone,
|
|
15
|
+
PlanningRequirement,
|
|
16
|
+
PlanningResearch,
|
|
17
|
+
GSDProject,
|
|
18
|
+
GSDMilestone,
|
|
19
|
+
GSDSlice,
|
|
20
|
+
GSDTask,
|
|
21
|
+
} from '../migrate/types.ts';
|
|
22
|
+
|
|
23
|
+
let passed = 0;
|
|
24
|
+
let failed = 0;
|
|
25
|
+
|
|
26
|
+
function assert(condition: boolean, message: string): void {
|
|
27
|
+
if (condition) {
|
|
28
|
+
passed++;
|
|
29
|
+
} else {
|
|
30
|
+
failed++;
|
|
31
|
+
console.error(` FAIL: ${message}`);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function assertEq<T>(actual: T, expected: T, message: string): void {
|
|
36
|
+
if (JSON.stringify(actual) === JSON.stringify(expected)) {
|
|
37
|
+
passed++;
|
|
38
|
+
} else {
|
|
39
|
+
failed++;
|
|
40
|
+
console.error(` FAIL: ${message} — expected ${JSON.stringify(expected)}, got ${JSON.stringify(actual)}`);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
// ─── Fixture Helpers ───────────────────────────────────────────────────────
|
|
45
|
+
|
|
46
|
+
function emptyProject(overrides: Partial<PlanningProject> = {}): PlanningProject {
|
|
47
|
+
return {
|
|
48
|
+
path: '/fake/.planning',
|
|
49
|
+
project: null,
|
|
50
|
+
roadmap: null,
|
|
51
|
+
requirements: [],
|
|
52
|
+
state: null,
|
|
53
|
+
config: null,
|
|
54
|
+
phases: {},
|
|
55
|
+
quickTasks: [],
|
|
56
|
+
milestones: [],
|
|
57
|
+
research: [],
|
|
58
|
+
validation: { valid: true, issues: [] },
|
|
59
|
+
...overrides,
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function flatRoadmap(entries: PlanningRoadmapEntry[]): PlanningRoadmap {
|
|
64
|
+
return {
|
|
65
|
+
raw: entries.map((e) => `- [${e.done ? 'x' : ' '}] Phase ${e.number}: ${e.title}`).join('\n'),
|
|
66
|
+
milestones: [],
|
|
67
|
+
phases: entries,
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
function milestoneRoadmap(milestones: PlanningRoadmapMilestone[]): PlanningRoadmap {
|
|
72
|
+
return {
|
|
73
|
+
raw: milestones.map((m) => `## ${m.id}: ${m.title}`).join('\n'),
|
|
74
|
+
milestones,
|
|
75
|
+
phases: [],
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
function roadmapEntry(number: number, title: string, done = false): PlanningRoadmapEntry {
|
|
80
|
+
return { number, title, done, raw: `- [${done ? 'x' : ' '}] Phase ${number}: ${title}` };
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
function makePhase(dirName: string, number: number, slug: string, overrides: Partial<PlanningPhase> = {}): PlanningPhase {
|
|
84
|
+
return {
|
|
85
|
+
dirName,
|
|
86
|
+
number,
|
|
87
|
+
slug,
|
|
88
|
+
plans: {},
|
|
89
|
+
summaries: {},
|
|
90
|
+
research: [],
|
|
91
|
+
verifications: [],
|
|
92
|
+
extraFiles: [],
|
|
93
|
+
...overrides,
|
|
94
|
+
};
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
function makePlan(planNumber: string, overrides: Partial<PlanningPlan> = {}): PlanningPlan {
|
|
98
|
+
return {
|
|
99
|
+
fileName: `00-${planNumber}-PLAN.md`,
|
|
100
|
+
planNumber,
|
|
101
|
+
frontmatter: {
|
|
102
|
+
phase: '00',
|
|
103
|
+
plan: planNumber,
|
|
104
|
+
type: 'implementation',
|
|
105
|
+
wave: null,
|
|
106
|
+
depends_on: [],
|
|
107
|
+
files_modified: [],
|
|
108
|
+
autonomous: false,
|
|
109
|
+
must_haves: null,
|
|
110
|
+
},
|
|
111
|
+
objective: `Objective for plan ${planNumber}`,
|
|
112
|
+
tasks: [`Task 1 for plan ${planNumber}`],
|
|
113
|
+
context: '',
|
|
114
|
+
verification: '',
|
|
115
|
+
successCriteria: '',
|
|
116
|
+
raw: '',
|
|
117
|
+
...overrides,
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
function makeSummary(planNumber: string, overrides: Partial<PlanningSummary> = {}): PlanningSummary {
|
|
122
|
+
return {
|
|
123
|
+
fileName: `00-${planNumber}-SUMMARY.md`,
|
|
124
|
+
planNumber,
|
|
125
|
+
frontmatter: {
|
|
126
|
+
phase: '00',
|
|
127
|
+
plan: planNumber,
|
|
128
|
+
subsystem: 'core',
|
|
129
|
+
tags: [],
|
|
130
|
+
requires: [],
|
|
131
|
+
provides: [`feature-${planNumber}`],
|
|
132
|
+
affects: [],
|
|
133
|
+
'tech-stack': [],
|
|
134
|
+
'key-files': [`file-${planNumber}.ts`],
|
|
135
|
+
'key-decisions': [`decision-${planNumber}`],
|
|
136
|
+
'patterns-established': [],
|
|
137
|
+
duration: '2h',
|
|
138
|
+
completed: '2026-01-15',
|
|
139
|
+
},
|
|
140
|
+
body: `Summary body for plan ${planNumber}`,
|
|
141
|
+
raw: '',
|
|
142
|
+
...overrides,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
function makeRequirement(id: string, title: string, status = 'active'): PlanningRequirement {
|
|
147
|
+
return { id, title, status, description: `Description for ${id}`, raw: '' };
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
function makeResearch(fileName: string, content: string): PlanningResearch {
|
|
151
|
+
return { fileName, content };
|
|
152
|
+
}
|
|
153
|
+
|
|
154
|
+
// ─── Scenario 1: Flat Single-Milestone (3 phases → M001 with S01/S02/S03) ──
|
|
155
|
+
|
|
156
|
+
{
|
|
157
|
+
console.log('Scenario 1: Flat single-milestone');
|
|
158
|
+
|
|
159
|
+
const project = emptyProject({
|
|
160
|
+
project: '# My Project\nA cool project.',
|
|
161
|
+
roadmap: flatRoadmap([
|
|
162
|
+
roadmapEntry(1, 'setup'),
|
|
163
|
+
roadmapEntry(2, 'core-logic'),
|
|
164
|
+
roadmapEntry(3, 'polish'),
|
|
165
|
+
]),
|
|
166
|
+
phases: {
|
|
167
|
+
'1-setup': makePhase('1-setup', 1, 'setup', {
|
|
168
|
+
plans: { '01': makePlan('01') },
|
|
169
|
+
}),
|
|
170
|
+
'2-core-logic': makePhase('2-core-logic', 2, 'core-logic', {
|
|
171
|
+
plans: { '01': makePlan('01'), '02': makePlan('02') },
|
|
172
|
+
}),
|
|
173
|
+
'3-polish': makePhase('3-polish', 3, 'polish', {
|
|
174
|
+
plans: { '01': makePlan('01') },
|
|
175
|
+
}),
|
|
176
|
+
},
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
const result = transformToGSD(project);
|
|
180
|
+
|
|
181
|
+
assertEq(result.milestones.length, 1, 'flat: produces 1 milestone');
|
|
182
|
+
assert(result.milestones[0]?.id === 'M001', 'flat: milestone ID is M001');
|
|
183
|
+
assertEq(result.milestones[0]?.slices.length, 3, 'flat: 3 slices');
|
|
184
|
+
assertEq(result.milestones[0]?.slices[0]?.id, 'S01', 'flat: first slice is S01');
|
|
185
|
+
assertEq(result.milestones[0]?.slices[1]?.id, 'S02', 'flat: second slice is S02');
|
|
186
|
+
assertEq(result.milestones[0]?.slices[2]?.id, 'S03', 'flat: third slice is S03');
|
|
187
|
+
assert(result.milestones[0]?.slices[0]?.title.length > 0, 'flat: slice title not empty');
|
|
188
|
+
assertEq(result.milestones[0]?.slices[0]?.tasks.length, 1, 'flat: S01 has 1 task');
|
|
189
|
+
assertEq(result.milestones[0]?.slices[1]?.tasks.length, 2, 'flat: S02 has 2 tasks');
|
|
190
|
+
assertEq(result.milestones[0]?.slices[2]?.tasks.length, 1, 'flat: S03 has 1 task');
|
|
191
|
+
assertEq(result.milestones[0]?.slices[0]?.tasks[0]?.id, 'T01', 'flat: first task is T01');
|
|
192
|
+
assertEq(result.milestones[0]?.slices[1]?.tasks[1]?.id, 'T02', 'flat: second task in S02 is T02');
|
|
193
|
+
assert(result.projectContent.includes('My Project'), 'flat: projectContent preserved');
|
|
194
|
+
assertEq(result.milestones[0]?.boundaryMap, [], 'flat: boundaryMap defaults to empty');
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
// ─── Scenario 2: Multi-Milestone (2 milestones with independent numbering) ──
|
|
198
|
+
|
|
199
|
+
{
|
|
200
|
+
console.log('Scenario 2: Multi-milestone');
|
|
201
|
+
|
|
202
|
+
const project = emptyProject({
|
|
203
|
+
roadmap: milestoneRoadmap([
|
|
204
|
+
{
|
|
205
|
+
id: 'v1',
|
|
206
|
+
title: 'Version One',
|
|
207
|
+
collapsed: false,
|
|
208
|
+
phases: [roadmapEntry(1, 'alpha'), roadmapEntry(2, 'beta')],
|
|
209
|
+
},
|
|
210
|
+
{
|
|
211
|
+
id: 'v2',
|
|
212
|
+
title: 'Version Two',
|
|
213
|
+
collapsed: false,
|
|
214
|
+
phases: [roadmapEntry(1, 'gamma'), roadmapEntry(2, 'delta'), roadmapEntry(3, 'epsilon')],
|
|
215
|
+
},
|
|
216
|
+
]),
|
|
217
|
+
phases: {
|
|
218
|
+
'1-alpha': makePhase('1-alpha', 1, 'alpha', { plans: { '01': makePlan('01') } }),
|
|
219
|
+
'2-beta': makePhase('2-beta', 2, 'beta', { plans: { '01': makePlan('01') } }),
|
|
220
|
+
'1-gamma': makePhase('1-gamma', 1, 'gamma', { plans: { '01': makePlan('01') } }),
|
|
221
|
+
'2-delta': makePhase('2-delta', 2, 'delta', { plans: { '01': makePlan('01') } }),
|
|
222
|
+
'3-epsilon': makePhase('3-epsilon', 3, 'epsilon', { plans: { '01': makePlan('01') } }),
|
|
223
|
+
},
|
|
224
|
+
});
|
|
225
|
+
|
|
226
|
+
const result = transformToGSD(project);
|
|
227
|
+
|
|
228
|
+
assertEq(result.milestones.length, 2, 'multi: 2 milestones');
|
|
229
|
+
assertEq(result.milestones[0]?.id, 'M001', 'multi: first milestone M001');
|
|
230
|
+
assertEq(result.milestones[1]?.id, 'M002', 'multi: second milestone M002');
|
|
231
|
+
assertEq(result.milestones[0]?.slices.length, 2, 'multi: M001 has 2 slices');
|
|
232
|
+
assertEq(result.milestones[1]?.slices.length, 3, 'multi: M002 has 3 slices');
|
|
233
|
+
// Independent numbering: both start at S01
|
|
234
|
+
assertEq(result.milestones[0]?.slices[0]?.id, 'S01', 'multi: M001 starts at S01');
|
|
235
|
+
assertEq(result.milestones[1]?.slices[0]?.id, 'S01', 'multi: M002 starts at S01');
|
|
236
|
+
assertEq(result.milestones[1]?.slices[2]?.id, 'S03', 'multi: M002 third slice is S03');
|
|
237
|
+
assert(result.milestones[0]?.title.length > 0, 'multi: M001 has title');
|
|
238
|
+
assert(result.milestones[1]?.title.length > 0, 'multi: M002 has title');
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
// ─── Scenario 3: Decimal Phase Ordering (1, 2, 2.1, 2.2, 3 → S01–S05) ──
|
|
242
|
+
|
|
243
|
+
{
|
|
244
|
+
console.log('Scenario 3: Decimal phase ordering');
|
|
245
|
+
|
|
246
|
+
const project = emptyProject({
|
|
247
|
+
roadmap: flatRoadmap([
|
|
248
|
+
roadmapEntry(1, 'foundation'),
|
|
249
|
+
roadmapEntry(2, 'main-feature'),
|
|
250
|
+
roadmapEntry(2.1, 'sub-feature-a'),
|
|
251
|
+
roadmapEntry(2.2, 'sub-feature-b'),
|
|
252
|
+
roadmapEntry(3, 'finalize'),
|
|
253
|
+
]),
|
|
254
|
+
phases: {
|
|
255
|
+
'1-foundation': makePhase('1-foundation', 1, 'foundation'),
|
|
256
|
+
'2-main-feature': makePhase('2-main-feature', 2, 'main-feature'),
|
|
257
|
+
'2.1-sub-feature-a': makePhase('2.1-sub-feature-a', 2.1, 'sub-feature-a'),
|
|
258
|
+
'2.2-sub-feature-b': makePhase('2.2-sub-feature-b', 2.2, 'sub-feature-b'),
|
|
259
|
+
'3-finalize': makePhase('3-finalize', 3, 'finalize'),
|
|
260
|
+
},
|
|
261
|
+
});
|
|
262
|
+
|
|
263
|
+
const result = transformToGSD(project);
|
|
264
|
+
|
|
265
|
+
assertEq(result.milestones[0]?.slices.length, 5, 'decimal: 5 slices total');
|
|
266
|
+
assertEq(result.milestones[0]?.slices[0]?.id, 'S01', 'decimal: first is S01');
|
|
267
|
+
assertEq(result.milestones[0]?.slices[1]?.id, 'S02', 'decimal: second is S02');
|
|
268
|
+
assertEq(result.milestones[0]?.slices[2]?.id, 'S03', 'decimal: third is S03');
|
|
269
|
+
assertEq(result.milestones[0]?.slices[3]?.id, 'S04', 'decimal: fourth is S04');
|
|
270
|
+
assertEq(result.milestones[0]?.slices[4]?.id, 'S05', 'decimal: fifth is S05');
|
|
271
|
+
// Order must be by float value: 1, 2, 2.1, 2.2, 3
|
|
272
|
+
assert(
|
|
273
|
+
result.milestones[0]?.slices[0]?.title.toLowerCase().includes('foundation'),
|
|
274
|
+
'decimal: S01 is foundation (phase 1)',
|
|
275
|
+
);
|
|
276
|
+
assert(
|
|
277
|
+
result.milestones[0]?.slices[4]?.title.toLowerCase().includes('finalize'),
|
|
278
|
+
'decimal: S05 is finalize (phase 3)',
|
|
279
|
+
);
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
// ─── Scenario 4: Completion State ──────────────────────────────────────────
|
|
283
|
+
|
|
284
|
+
{
|
|
285
|
+
console.log('Scenario 4: Completion state mapping');
|
|
286
|
+
|
|
287
|
+
const project = emptyProject({
|
|
288
|
+
roadmap: flatRoadmap([
|
|
289
|
+
roadmapEntry(1, 'done-phase', true),
|
|
290
|
+
roadmapEntry(2, 'active-phase', false),
|
|
291
|
+
]),
|
|
292
|
+
phases: {
|
|
293
|
+
'1-done-phase': makePhase('1-done-phase', 1, 'done-phase', {
|
|
294
|
+
plans: { '01': makePlan('01'), '02': makePlan('02') },
|
|
295
|
+
summaries: {
|
|
296
|
+
'01': makeSummary('01'),
|
|
297
|
+
// plan 02 has no summary → task not done
|
|
298
|
+
},
|
|
299
|
+
}),
|
|
300
|
+
'2-active-phase': makePhase('2-active-phase', 2, 'active-phase', {
|
|
301
|
+
plans: { '01': makePlan('01') },
|
|
302
|
+
}),
|
|
303
|
+
},
|
|
304
|
+
});
|
|
305
|
+
|
|
306
|
+
const result = transformToGSD(project);
|
|
307
|
+
const doneSlice = result.milestones[0]?.slices[0];
|
|
308
|
+
const activeSlice = result.milestones[0]?.slices[1];
|
|
309
|
+
|
|
310
|
+
assert(doneSlice?.done === true, 'completion: done phase → done slice');
|
|
311
|
+
assert(activeSlice?.done === false, 'completion: active phase → not-done slice');
|
|
312
|
+
assert(doneSlice?.tasks[0]?.done === true, 'completion: plan with summary → done task');
|
|
313
|
+
assert(doneSlice?.tasks[1]?.done === false, 'completion: plan without summary → not-done task');
|
|
314
|
+
assert(doneSlice?.tasks[0]?.summary !== null, 'completion: done task has summary data');
|
|
315
|
+
assert(doneSlice?.tasks[1]?.summary === null, 'completion: not-done task has null summary');
|
|
316
|
+
assertEq(doneSlice?.tasks[0]?.summary?.completedAt, '2026-01-15', 'completion: summary completedAt from frontmatter');
|
|
317
|
+
assertEq(doneSlice?.tasks[0]?.summary?.duration, '2h', 'completion: summary duration from frontmatter');
|
|
318
|
+
assertEq(doneSlice?.tasks[0]?.summary?.provides, ['feature-01'], 'completion: summary provides from frontmatter');
|
|
319
|
+
assertEq(doneSlice?.tasks[0]?.summary?.keyFiles, ['file-01.ts'], 'completion: summary keyFiles from frontmatter');
|
|
320
|
+
assert(doneSlice?.tasks[0]?.summary?.whatHappened?.includes('Summary body'), 'completion: summary whatHappened from body');
|
|
321
|
+
assert(doneSlice?.summary !== null, 'completion: done slice has slice summary');
|
|
322
|
+
assert(activeSlice?.summary === null, 'completion: active slice has null summary');
|
|
323
|
+
assertEq(doneSlice?.tasks[0]?.estimate, '2h', 'completion: task estimate from summary duration');
|
|
324
|
+
}
|
|
325
|
+
|
|
326
|
+
// ─── Scenario 5: Research Consolidation ────────────────────────────────────
|
|
327
|
+
|
|
328
|
+
{
|
|
329
|
+
console.log('Scenario 5: Research consolidation');
|
|
330
|
+
|
|
331
|
+
const project = emptyProject({
|
|
332
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'researched-phase')]),
|
|
333
|
+
research: [
|
|
334
|
+
makeResearch('SUMMARY.md', '# Project Summary\nOverview content.'),
|
|
335
|
+
makeResearch('ARCHITECTURE.md', '# Architecture\nArch details.'),
|
|
336
|
+
makeResearch('PITFALLS.md', '# Pitfalls\nThings to avoid.'),
|
|
337
|
+
],
|
|
338
|
+
phases: {
|
|
339
|
+
'1-researched-phase': makePhase('1-researched-phase', 1, 'researched-phase', {
|
|
340
|
+
research: [
|
|
341
|
+
makeResearch('FEATURES.md', '# Phase Features\nFeature list.'),
|
|
342
|
+
],
|
|
343
|
+
}),
|
|
344
|
+
},
|
|
345
|
+
});
|
|
346
|
+
|
|
347
|
+
const result = transformToGSD(project);
|
|
348
|
+
|
|
349
|
+
// Project-level research → milestone research
|
|
350
|
+
assert(result.milestones[0]?.research !== null, 'research: milestone has consolidated research');
|
|
351
|
+
assert(result.milestones[0]?.research!.includes('Project Summary'), 'research: includes SUMMARY content');
|
|
352
|
+
assert(result.milestones[0]?.research!.includes('Architecture'), 'research: includes ARCHITECTURE content');
|
|
353
|
+
assert(result.milestones[0]?.research!.includes('Pitfalls'), 'research: includes PITFALLS content');
|
|
354
|
+
|
|
355
|
+
// Fixed ordering: SUMMARY before ARCHITECTURE before PITFALLS
|
|
356
|
+
const summaryIdx = result.milestones[0]?.research!.indexOf('Project Summary') ?? -1;
|
|
357
|
+
const archIdx = result.milestones[0]?.research!.indexOf('Architecture') ?? -1;
|
|
358
|
+
const pitfallIdx = result.milestones[0]?.research!.indexOf('Pitfalls') ?? -1;
|
|
359
|
+
assert(summaryIdx < archIdx, 'research: SUMMARY before ARCHITECTURE in consolidated');
|
|
360
|
+
assert(archIdx < pitfallIdx, 'research: ARCHITECTURE before PITFALLS in consolidated');
|
|
361
|
+
|
|
362
|
+
// Phase-level research → slice research
|
|
363
|
+
const slice = result.milestones[0]?.slices[0];
|
|
364
|
+
assert(slice?.research !== null, 'research: slice has phase research');
|
|
365
|
+
assert(slice?.research!.includes('Phase Features'), 'research: slice research includes phase content');
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
// ─── Scenario 6: Requirements Classification ──────────────────────────────
|
|
369
|
+
|
|
370
|
+
{
|
|
371
|
+
console.log('Scenario 6: Requirements classification');
|
|
372
|
+
|
|
373
|
+
const project = emptyProject({
|
|
374
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'req-phase')]),
|
|
375
|
+
requirements: [
|
|
376
|
+
makeRequirement('R001', 'Core Feature', 'active'),
|
|
377
|
+
makeRequirement('R002', 'Secondary Feature', 'validated'),
|
|
378
|
+
makeRequirement('R003', 'Deferred Feature', 'deferred'),
|
|
379
|
+
],
|
|
380
|
+
phases: {
|
|
381
|
+
'1-req-phase': makePhase('1-req-phase', 1, 'req-phase'),
|
|
382
|
+
},
|
|
383
|
+
});
|
|
384
|
+
|
|
385
|
+
const result = transformToGSD(project);
|
|
386
|
+
|
|
387
|
+
assertEq(result.requirements.length, 3, 'requirements: 3 requirements');
|
|
388
|
+
assertEq(result.requirements[0]?.id, 'R001', 'requirements: first is R001');
|
|
389
|
+
assertEq(result.requirements[0]?.status, 'active', 'requirements: R001 status active');
|
|
390
|
+
assertEq(result.requirements[1]?.status, 'validated', 'requirements: R002 status validated');
|
|
391
|
+
assertEq(result.requirements[2]?.status, 'deferred', 'requirements: R003 status deferred');
|
|
392
|
+
assert(result.requirements[0]?.title === 'Core Feature', 'requirements: R001 title preserved');
|
|
393
|
+
assert(result.requirements[0]?.description.includes('Description for R001'), 'requirements: R001 description preserved');
|
|
394
|
+
assertEq(result.requirements[0]?.class, 'core-capability', 'requirements: default class');
|
|
395
|
+
assertEq(result.requirements[0]?.source, 'inferred', 'requirements: default source');
|
|
396
|
+
assertEq(result.requirements[0]?.primarySlice, 'none yet', 'requirements: default primarySlice');
|
|
397
|
+
}
|
|
398
|
+
|
|
399
|
+
// ─── Scenario 7: Empty Phase (no plans → slice with 0 tasks) ───────────────
|
|
400
|
+
|
|
401
|
+
{
|
|
402
|
+
console.log('Scenario 7: Empty phase');
|
|
403
|
+
|
|
404
|
+
const project = emptyProject({
|
|
405
|
+
roadmap: flatRoadmap([
|
|
406
|
+
roadmapEntry(1, 'empty-phase'),
|
|
407
|
+
roadmapEntry(2, 'non-empty-phase'),
|
|
408
|
+
]),
|
|
409
|
+
phases: {
|
|
410
|
+
'1-empty-phase': makePhase('1-empty-phase', 1, 'empty-phase'),
|
|
411
|
+
'2-non-empty-phase': makePhase('2-non-empty-phase', 2, 'non-empty-phase', {
|
|
412
|
+
plans: { '01': makePlan('01') },
|
|
413
|
+
}),
|
|
414
|
+
},
|
|
415
|
+
});
|
|
416
|
+
|
|
417
|
+
const result = transformToGSD(project);
|
|
418
|
+
|
|
419
|
+
assertEq(result.milestones[0]?.slices[0]?.tasks.length, 0, 'empty: empty phase → 0 tasks');
|
|
420
|
+
assertEq(result.milestones[0]?.slices[1]?.tasks.length, 1, 'empty: non-empty phase → 1 task');
|
|
421
|
+
assert(result.milestones[0]?.slices[0]?.id === 'S01', 'empty: empty slice still gets ID');
|
|
422
|
+
}
|
|
423
|
+
|
|
424
|
+
// ─── Scenario 8: Demo Derivation from Plan Objective ───────────────────────
|
|
425
|
+
|
|
426
|
+
{
|
|
427
|
+
console.log('Scenario 8: Demo derivation');
|
|
428
|
+
|
|
429
|
+
const project = emptyProject({
|
|
430
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'demo-phase')]),
|
|
431
|
+
phases: {
|
|
432
|
+
'1-demo-phase': makePhase('1-demo-phase', 1, 'demo-phase', {
|
|
433
|
+
plans: {
|
|
434
|
+
'01': makePlan('01', { objective: 'Build the authentication system with JWT tokens.' }),
|
|
435
|
+
},
|
|
436
|
+
}),
|
|
437
|
+
},
|
|
438
|
+
});
|
|
439
|
+
|
|
440
|
+
const result = transformToGSD(project);
|
|
441
|
+
|
|
442
|
+
assert(result.milestones[0]?.slices[0]?.demo.length > 0, 'demo: slice demo is not empty');
|
|
443
|
+
assert(
|
|
444
|
+
result.milestones[0]?.slices[0]?.demo.includes('authentication') ||
|
|
445
|
+
result.milestones[0]?.slices[0]?.demo.includes('Build'),
|
|
446
|
+
'demo: slice demo derived from first plan objective',
|
|
447
|
+
);
|
|
448
|
+
assert(result.milestones[0]?.slices[0]?.goal.length > 0, 'demo: slice goal is not empty');
|
|
449
|
+
}
|
|
450
|
+
|
|
451
|
+
// ─── Scenario 9: Field Defaults and Type Safety ────────────────────────────
|
|
452
|
+
|
|
453
|
+
{
|
|
454
|
+
console.log('Scenario 9: Field defaults');
|
|
455
|
+
|
|
456
|
+
const project = emptyProject({
|
|
457
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'defaults-phase')]),
|
|
458
|
+
phases: {
|
|
459
|
+
'1-defaults-phase': makePhase('1-defaults-phase', 1, 'defaults-phase', {
|
|
460
|
+
plans: {
|
|
461
|
+
'01': makePlan('01', {
|
|
462
|
+
frontmatter: {
|
|
463
|
+
phase: '01',
|
|
464
|
+
plan: '01',
|
|
465
|
+
type: 'implementation',
|
|
466
|
+
wave: null,
|
|
467
|
+
depends_on: [],
|
|
468
|
+
files_modified: ['src/auth.ts', 'src/db.ts'],
|
|
469
|
+
autonomous: false,
|
|
470
|
+
must_haves: { truths: ['Auth works', 'DB connected'], artifacts: [], key_links: [] },
|
|
471
|
+
},
|
|
472
|
+
}),
|
|
473
|
+
},
|
|
474
|
+
}),
|
|
475
|
+
},
|
|
476
|
+
});
|
|
477
|
+
|
|
478
|
+
const result = transformToGSD(project);
|
|
479
|
+
const slice = result.milestones[0]?.slices[0];
|
|
480
|
+
const task = slice?.tasks[0];
|
|
481
|
+
|
|
482
|
+
assertEq(slice?.risk, 'medium', 'defaults: slice risk defaults to medium');
|
|
483
|
+
assertEq(slice?.depends, [], 'defaults: S01 has no depends');
|
|
484
|
+
assert(task?.description.length > 0, 'defaults: task description not empty');
|
|
485
|
+
assertEq(task?.files, ['src/auth.ts', 'src/db.ts'], 'defaults: task files from frontmatter');
|
|
486
|
+
assertEq(task?.mustHaves, ['Auth works', 'DB connected'], 'defaults: task mustHaves from frontmatter');
|
|
487
|
+
assertEq(task?.done, false, 'defaults: task without summary is not done');
|
|
488
|
+
assertEq(task?.estimate, '', 'defaults: task without summary has empty estimate');
|
|
489
|
+
assert(task?.summary === null, 'defaults: task without summary has null summary');
|
|
490
|
+
}
|
|
491
|
+
|
|
492
|
+
// ─── Scenario 10: Sequential Depends ──────────────────────────────────────
|
|
493
|
+
|
|
494
|
+
{
|
|
495
|
+
console.log('Scenario 10: Sequential depends');
|
|
496
|
+
|
|
497
|
+
const project = emptyProject({
|
|
498
|
+
roadmap: flatRoadmap([
|
|
499
|
+
roadmapEntry(1, 'first'),
|
|
500
|
+
roadmapEntry(2, 'second'),
|
|
501
|
+
roadmapEntry(3, 'third'),
|
|
502
|
+
]),
|
|
503
|
+
phases: {
|
|
504
|
+
'1-first': makePhase('1-first', 1, 'first'),
|
|
505
|
+
'2-second': makePhase('2-second', 2, 'second'),
|
|
506
|
+
'3-third': makePhase('3-third', 3, 'third'),
|
|
507
|
+
},
|
|
508
|
+
});
|
|
509
|
+
|
|
510
|
+
const result = transformToGSD(project);
|
|
511
|
+
const slices = result.milestones[0]?.slices;
|
|
512
|
+
|
|
513
|
+
assertEq(slices?.[0]?.depends, [], 'depends: S01 has empty depends');
|
|
514
|
+
assertEq(slices?.[1]?.depends, ['S01'], 'depends: S02 depends on S01');
|
|
515
|
+
assertEq(slices?.[2]?.depends, ['S02'], 'depends: S03 depends on S02');
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
// ─── Scenario 11: Requirements with unknown status and missing IDs ─────────
|
|
519
|
+
|
|
520
|
+
{
|
|
521
|
+
console.log('Scenario 11: Requirements edge cases');
|
|
522
|
+
|
|
523
|
+
const project = emptyProject({
|
|
524
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'req-edge')]),
|
|
525
|
+
requirements: [
|
|
526
|
+
makeRequirement('', 'No ID Feature', 'active'),
|
|
527
|
+
makeRequirement('', 'Another No ID', 'validated'),
|
|
528
|
+
makeRequirement('R005', 'Has ID', 'something-weird'),
|
|
529
|
+
makeRequirement('R006', 'Deferred One', 'DEFERRED'),
|
|
530
|
+
],
|
|
531
|
+
phases: {
|
|
532
|
+
'1-req-edge': makePhase('1-req-edge', 1, 'req-edge'),
|
|
533
|
+
},
|
|
534
|
+
});
|
|
535
|
+
|
|
536
|
+
const result = transformToGSD(project);
|
|
537
|
+
|
|
538
|
+
assertEq(result.requirements[0]?.id, 'R001', 'req-edge: empty id gets R001');
|
|
539
|
+
assertEq(result.requirements[1]?.id, 'R002', 'req-edge: second empty id gets R002');
|
|
540
|
+
assertEq(result.requirements[2]?.id, 'R005', 'req-edge: existing id preserved');
|
|
541
|
+
assertEq(result.requirements[2]?.status, 'active', 'req-edge: unknown status normalized to active');
|
|
542
|
+
assertEq(result.requirements[3]?.status, 'deferred', 'req-edge: uppercase DEFERRED normalized');
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
// ─── Scenario 12: Vision derivation ────────────────────────────────────────
|
|
546
|
+
|
|
547
|
+
{
|
|
548
|
+
console.log('Scenario 12: Vision derivation');
|
|
549
|
+
|
|
550
|
+
// Vision from project description
|
|
551
|
+
const project1 = emptyProject({
|
|
552
|
+
project: '# Cool Project\nA revolutionary tool for developers.',
|
|
553
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'vision-phase')]),
|
|
554
|
+
phases: { '1-vision-phase': makePhase('1-vision-phase', 1, 'vision-phase') },
|
|
555
|
+
});
|
|
556
|
+
|
|
557
|
+
const result1 = transformToGSD(project1);
|
|
558
|
+
assert(result1.milestones[0]?.vision.includes('revolutionary'), 'vision: derived from project first line');
|
|
559
|
+
|
|
560
|
+
// Vision fallback when no project
|
|
561
|
+
const project2 = emptyProject({
|
|
562
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'fallback')]),
|
|
563
|
+
phases: { '1-fallback': makePhase('1-fallback', 1, 'fallback') },
|
|
564
|
+
});
|
|
565
|
+
|
|
566
|
+
const result2 = transformToGSD(project2);
|
|
567
|
+
assert(result2.milestones[0]?.vision.length > 0, 'vision: fallback is non-empty');
|
|
568
|
+
}
|
|
569
|
+
|
|
570
|
+
// ─── Scenario 13: Decisions content from summaries ─────────────────────────
|
|
571
|
+
|
|
572
|
+
{
|
|
573
|
+
console.log('Scenario 13: Decisions content');
|
|
574
|
+
|
|
575
|
+
const project = emptyProject({
|
|
576
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'decision-phase', true)]),
|
|
577
|
+
phases: {
|
|
578
|
+
'1-decision-phase': makePhase('1-decision-phase', 1, 'decision-phase', {
|
|
579
|
+
plans: { '01': makePlan('01') },
|
|
580
|
+
summaries: { '01': makeSummary('01') },
|
|
581
|
+
}),
|
|
582
|
+
},
|
|
583
|
+
});
|
|
584
|
+
|
|
585
|
+
const result = transformToGSD(project);
|
|
586
|
+
|
|
587
|
+
assert(result.decisionsContent.includes('decision-01'), 'decisions: extracts key-decisions from summaries');
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
// ─── Scenario 14: No undefined values in output ───────────────────────────
|
|
591
|
+
|
|
592
|
+
{
|
|
593
|
+
console.log('Scenario 14: No undefined values');
|
|
594
|
+
|
|
595
|
+
const project = emptyProject({
|
|
596
|
+
project: '# Test\nDescription.',
|
|
597
|
+
roadmap: flatRoadmap([
|
|
598
|
+
roadmapEntry(1, 'full-phase', true),
|
|
599
|
+
roadmapEntry(2, 'empty-phase', false),
|
|
600
|
+
]),
|
|
601
|
+
requirements: [makeRequirement('R001', 'Req', 'active')],
|
|
602
|
+
research: [makeResearch('SUMMARY.md', 'Research content')],
|
|
603
|
+
phases: {
|
|
604
|
+
'1-full-phase': makePhase('1-full-phase', 1, 'full-phase', {
|
|
605
|
+
plans: { '01': makePlan('01') },
|
|
606
|
+
summaries: { '01': makeSummary('01') },
|
|
607
|
+
research: [makeResearch('FEATURES.md', 'Features')],
|
|
608
|
+
}),
|
|
609
|
+
'2-empty-phase': makePhase('2-empty-phase', 2, 'empty-phase'),
|
|
610
|
+
},
|
|
611
|
+
});
|
|
612
|
+
|
|
613
|
+
const result = transformToGSD(project);
|
|
614
|
+
|
|
615
|
+
// Deep check for undefined values
|
|
616
|
+
function checkNoUndefined(obj: unknown, path: string): void {
|
|
617
|
+
if (obj === undefined) {
|
|
618
|
+
assert(false, `no-undefined: ${path} is undefined`);
|
|
619
|
+
return;
|
|
620
|
+
}
|
|
621
|
+
if (obj === null) return; // null is allowed (e.g. research, summary)
|
|
622
|
+
if (Array.isArray(obj)) {
|
|
623
|
+
for (let i = 0; i < obj.length; i++) {
|
|
624
|
+
checkNoUndefined(obj[i], `${path}[${i}]`);
|
|
625
|
+
}
|
|
626
|
+
} else if (typeof obj === 'object') {
|
|
627
|
+
for (const [key, val] of Object.entries(obj as Record<string, unknown>)) {
|
|
628
|
+
checkNoUndefined(val, `${path}.${key}`);
|
|
629
|
+
}
|
|
630
|
+
}
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
checkNoUndefined(result, 'result');
|
|
634
|
+
assert(true, 'no-undefined: deep check completed without finding undefined values');
|
|
635
|
+
}
|
|
636
|
+
|
|
637
|
+
// ─── Scenario 15: Research with no files ───────────────────────────────────
|
|
638
|
+
|
|
639
|
+
{
|
|
640
|
+
console.log('Scenario 15: Empty research');
|
|
641
|
+
|
|
642
|
+
const project = emptyProject({
|
|
643
|
+
roadmap: flatRoadmap([roadmapEntry(1, 'no-research')]),
|
|
644
|
+
phases: { '1-no-research': makePhase('1-no-research', 1, 'no-research') },
|
|
645
|
+
});
|
|
646
|
+
|
|
647
|
+
const result = transformToGSD(project);
|
|
648
|
+
assert(result.milestones[0]?.research === null, 'empty-research: milestone research is null');
|
|
649
|
+
assert(result.milestones[0]?.slices[0]?.research === null, 'empty-research: slice research is null');
|
|
650
|
+
}
|
|
651
|
+
|
|
652
|
+
// ─── Results ───────────────────────────────────────────────────────────────
|
|
653
|
+
|
|
654
|
+
console.log(`\n${passed + failed} assertions: ${passed} passed, ${failed} failed`);
|
|
655
|
+
if (failed > 0) {
|
|
656
|
+
process.exit(1);
|
|
657
|
+
}
|