@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,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
|
+
});
|