aiwcli 0.12.6 → 0.12.7
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/bin/dev.cmd +3 -3
- package/bin/dev.js +16 -16
- package/bin/run.cmd +3 -3
- package/bin/run.js +21 -21
- package/dist/commands/branch.js +7 -2
- package/dist/lib/bmad-installer.js +37 -37
- package/dist/lib/terminal.d.ts +2 -0
- package/dist/lib/terminal.js +57 -7
- package/dist/templates/CLAUDE.md +205 -205
- package/dist/templates/_shared/.claude/commands/handoff-resume.md +12 -12
- package/dist/templates/_shared/.claude/commands/handoff.md +12 -12
- package/dist/templates/_shared/.claude/settings.json +65 -65
- package/dist/templates/_shared/.codex/workflows/handoff.md +226 -226
- package/dist/templates/_shared/.windsurf/workflows/handoff.md +226 -226
- package/dist/templates/_shared/handoff-system/CLAUDE.md +421 -421
- package/dist/templates/_shared/handoff-system/lib/document-generator.ts +215 -215
- package/dist/templates/_shared/handoff-system/lib/handoff-reader.ts +158 -158
- package/dist/templates/_shared/handoff-system/scripts/resume_handoff.ts +373 -373
- package/dist/templates/_shared/handoff-system/scripts/save_handoff.ts +469 -469
- package/dist/templates/_shared/handoff-system/workflows/handoff-resume.md +66 -66
- package/dist/templates/_shared/handoff-system/workflows/handoff.md +254 -254
- package/dist/templates/_shared/hooks-ts/_utils/git-state.ts +2 -2
- package/dist/templates/_shared/hooks-ts/archive_plan.ts +159 -159
- package/dist/templates/_shared/hooks-ts/context_monitor.ts +147 -147
- package/dist/templates/_shared/hooks-ts/file-suggestion.ts +128 -128
- package/dist/templates/_shared/hooks-ts/pre_compact.ts +49 -49
- package/dist/templates/_shared/hooks-ts/session_end.ts +196 -196
- package/dist/templates/_shared/hooks-ts/session_start.ts +163 -163
- package/dist/templates/_shared/hooks-ts/task_create_capture.ts +48 -48
- package/dist/templates/_shared/hooks-ts/task_update_capture.ts +74 -74
- package/dist/templates/_shared/hooks-ts/user_prompt_submit.ts +93 -93
- package/dist/templates/_shared/lib-ts/CLAUDE.md +367 -367
- package/dist/templates/_shared/lib-ts/base/atomic-write.ts +138 -138
- package/dist/templates/_shared/lib-ts/base/constants.ts +303 -303
- package/dist/templates/_shared/lib-ts/base/git-state.ts +58 -58
- package/dist/templates/_shared/lib-ts/base/hook-utils.ts +582 -582
- package/dist/templates/_shared/lib-ts/base/inference.ts +301 -301
- package/dist/templates/_shared/lib-ts/base/logger.ts +247 -247
- package/dist/templates/_shared/lib-ts/base/state-io.ts +202 -202
- package/dist/templates/_shared/lib-ts/base/stop-words.ts +184 -184
- package/dist/templates/_shared/lib-ts/base/utils.ts +184 -184
- package/dist/templates/_shared/lib-ts/context/context-formatter.ts +566 -566
- package/dist/templates/_shared/lib-ts/context/context-selector.ts +524 -524
- package/dist/templates/_shared/lib-ts/context/context-store.ts +712 -712
- package/dist/templates/_shared/lib-ts/context/plan-manager.ts +312 -312
- package/dist/templates/_shared/lib-ts/context/task-tracker.ts +185 -185
- package/dist/templates/_shared/lib-ts/package.json +20 -20
- package/dist/templates/_shared/lib-ts/templates/formatters.ts +102 -102
- package/dist/templates/_shared/lib-ts/templates/plan-context.ts +58 -58
- package/dist/templates/_shared/lib-ts/tsconfig.json +13 -13
- package/dist/templates/_shared/lib-ts/types.ts +186 -186
- package/dist/templates/_shared/scripts/resolve_context.ts +33 -33
- package/dist/templates/_shared/scripts/status_line.ts +690 -690
- package/dist/templates/cc-native/.claude/commands/cc-native/rlm/ask.md +136 -136
- package/dist/templates/cc-native/.claude/commands/cc-native/rlm/index.md +21 -21
- package/dist/templates/cc-native/.claude/commands/cc-native/rlm/overview.md +56 -56
- package/dist/templates/cc-native/.claude/commands/cc-native/specdev.md +10 -10
- package/dist/templates/cc-native/.windsurf/workflows/cc-native/fix.md +8 -8
- package/dist/templates/cc-native/.windsurf/workflows/cc-native/implement.md +8 -8
- package/dist/templates/cc-native/.windsurf/workflows/cc-native/research.md +8 -8
- package/dist/templates/cc-native/CC-NATIVE-README.md +189 -189
- package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +304 -304
- package/dist/templates/cc-native/_cc-native/agents/CLAUDE.md +143 -143
- package/dist/templates/cc-native/_cc-native/agents/PLAN-ORCHESTRATOR.md +213 -213
- package/dist/templates/cc-native/_cc-native/agents/plan-questions/PLAN-QUESTIONER.md +70 -70
- package/dist/templates/cc-native/_cc-native/cc-native.config.json +96 -96
- package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +247 -247
- package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +76 -76
- package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_subagent.ts +54 -54
- package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_write.ts +51 -51
- package/dist/templates/cc-native/_cc-native/hooks/mark_questions_asked.ts +53 -53
- package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +61 -61
- package/dist/templates/cc-native/_cc-native/lib-ts/agent-selection.ts +163 -163
- package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +156 -156
- package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/format.ts +597 -597
- package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/index.ts +26 -26
- package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/tracker.ts +107 -107
- package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/write.ts +119 -119
- package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +21 -21
- package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +319 -319
- package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +144 -144
- package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +57 -57
- package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +83 -83
- package/dist/templates/cc-native/_cc-native/lib-ts/corroboration.ts +119 -119
- package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +79 -79
- package/dist/templates/cc-native/_cc-native/lib-ts/graduation.ts +132 -132
- package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +116 -116
- package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +168 -168
- package/dist/templates/cc-native/_cc-native/lib-ts/orchestrator.ts +70 -70
- package/dist/templates/cc-native/_cc-native/lib-ts/output-builder.ts +130 -130
- package/dist/templates/cc-native/_cc-native/lib-ts/plan-discovery.ts +80 -80
- package/dist/templates/cc-native/_cc-native/lib-ts/plan-enhancement.ts +41 -41
- package/dist/templates/cc-native/_cc-native/lib-ts/plan-questions.ts +101 -101
- package/dist/templates/cc-native/_cc-native/lib-ts/review-pipeline.ts +511 -511
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/agent.ts +71 -71
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/base/base-agent.ts +217 -217
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/index.ts +12 -12
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/claude-agent.ts +66 -66
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/codex-agent.ts +184 -184
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/gemini-agent.ts +39 -39
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/orchestrator-claude-agent.ts +196 -196
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/schemas.ts +201 -201
- package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/types.ts +21 -21
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/CLAUDE.md +480 -480
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/embedding-indexer.ts +287 -287
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/hyde.ts +148 -148
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/index.ts +54 -54
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/logger.ts +58 -58
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/ollama-client.ts +208 -208
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/retrieval-pipeline.ts +460 -460
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-indexer.ts +446 -446
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-loader.ts +280 -280
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-searcher.ts +274 -274
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/types.ts +201 -201
- package/dist/templates/cc-native/_cc-native/lib-ts/rlm/vector-store.ts +278 -278
- package/dist/templates/cc-native/_cc-native/lib-ts/settings.ts +184 -184
- package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +275 -275
- package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +18 -18
- package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +329 -329
- package/dist/templates/cc-native/_cc-native/lib-ts/verdict.ts +72 -72
- package/dist/templates/cc-native/_cc-native/workflows/specdev.md +9 -9
- package/oclif.manifest.json +1 -1
- package/package.json +108 -108
- package/dist/templates/cc-native/_cc-native/lib-ts/nul +0 -3
|
@@ -1,158 +1,158 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Handoff reader utilities for programmatic resume.
|
|
3
|
-
*
|
|
4
|
-
* Provides functions to find, read, and parse handoff folders
|
|
5
|
-
* created by save_handoff.ts. Used by resume_handoff.ts script
|
|
6
|
-
* and potentially by session_start hooks.
|
|
7
|
-
*/
|
|
8
|
-
|
|
9
|
-
import * as fs from "node:fs";
|
|
10
|
-
import * as path from "node:path";
|
|
11
|
-
import { getContextHandoffsDir } from "../../lib-ts/base/constants.js";
|
|
12
|
-
import { getContext } from "../../lib-ts/context/context-store.js";
|
|
13
|
-
import type { HandoffSections } from "../../lib-ts/types.js";
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Find the most recent handoff folder for a context.
|
|
17
|
-
* Lists subdirectories in the handoffs dir, sorts by name
|
|
18
|
-
* (YYYY-MM-DD-HHMM format ensures lexicographic = chronological).
|
|
19
|
-
* Returns full path to most recent folder, or null.
|
|
20
|
-
*/
|
|
21
|
-
export function findLatestHandoff(contextId: string, projectRoot?: string): string | null {
|
|
22
|
-
const handoffsDir = getContextHandoffsDir(contextId, projectRoot);
|
|
23
|
-
|
|
24
|
-
try {
|
|
25
|
-
if (!fs.existsSync(handoffsDir)) return null;
|
|
26
|
-
const entries = fs.readdirSync(handoffsDir, { withFileTypes: true })
|
|
27
|
-
.filter(e => e.isDirectory())
|
|
28
|
-
.map(e => e.name)
|
|
29
|
-
.sort();
|
|
30
|
-
|
|
31
|
-
if (entries.length === 0) return null;
|
|
32
|
-
return path.join(handoffsDir, entries[entries.length - 1]!);
|
|
33
|
-
} catch {
|
|
34
|
-
return null;
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Read all section files from a handoff folder, structured by priority.
|
|
40
|
-
* Returns content for each file (null if missing/unreadable).
|
|
41
|
-
*/
|
|
42
|
-
export function readHandoffSections(handoffFolder: string): HandoffSections {
|
|
43
|
-
const fileMap: Record<keyof HandoffSections, string> = {
|
|
44
|
-
index: "index.md",
|
|
45
|
-
deadEnds: "dead-ends.md",
|
|
46
|
-
pending: "pending.md",
|
|
47
|
-
plan: "plan.md",
|
|
48
|
-
decisions: "decisions.md",
|
|
49
|
-
completedWork: "completed-work.md",
|
|
50
|
-
context: "context.md",
|
|
51
|
-
};
|
|
52
|
-
|
|
53
|
-
const sections: HandoffSections = {
|
|
54
|
-
index: null,
|
|
55
|
-
deadEnds: null,
|
|
56
|
-
pending: null,
|
|
57
|
-
plan: null,
|
|
58
|
-
decisions: null,
|
|
59
|
-
completedWork: null,
|
|
60
|
-
context: null,
|
|
61
|
-
};
|
|
62
|
-
|
|
63
|
-
for (const [key, filename] of Object.entries(fileMap)) {
|
|
64
|
-
const filePath = path.join(handoffFolder, filename);
|
|
65
|
-
try {
|
|
66
|
-
if (fs.existsSync(filePath)) {
|
|
67
|
-
sections[key as keyof HandoffSections] = fs.readFileSync(filePath, "utf-8");
|
|
68
|
-
}
|
|
69
|
-
} catch {
|
|
70
|
-
// graceful — leave as null
|
|
71
|
-
}
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
return sections;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
/**
|
|
78
|
-
* Parse the handoff folder name to extract creation timestamp.
|
|
79
|
-
* Expects YYYY-MM-DD-HHMM format (with optional -N suffix for collisions).
|
|
80
|
-
*/
|
|
81
|
-
export function getHandoffTimestamp(handoffFolder: string): Date | null {
|
|
82
|
-
const basename = path.basename(handoffFolder);
|
|
83
|
-
// Match YYYY-MM-DD-HHMM with optional collision suffix
|
|
84
|
-
const match = basename.match(/^(\d{4})-(\d{2})-(\d{2})-(\d{2})(\d{2})/);
|
|
85
|
-
if (!match) return null;
|
|
86
|
-
|
|
87
|
-
const [, year, month, day, hour, minute] = match;
|
|
88
|
-
const date = new Date(
|
|
89
|
-
parseInt(year!, 10),
|
|
90
|
-
parseInt(month!, 10) - 1,
|
|
91
|
-
parseInt(day!, 10),
|
|
92
|
-
parseInt(hour!, 10),
|
|
93
|
-
parseInt(minute!, 10),
|
|
94
|
-
);
|
|
95
|
-
|
|
96
|
-
return isNaN(date.getTime()) ? null : date;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
/**
|
|
100
|
-
* Get the plan path referenced by a handoff.
|
|
101
|
-
* First checks plan.md frontmatter for plan_path, then falls back
|
|
102
|
-
* to the context state's plan_path.
|
|
103
|
-
*/
|
|
104
|
-
export function getHandoffPlanReference(
|
|
105
|
-
handoffFolder: string,
|
|
106
|
-
contextId: string,
|
|
107
|
-
projectRoot?: string,
|
|
108
|
-
): string | null {
|
|
109
|
-
// Try plan.md frontmatter
|
|
110
|
-
const planMdPath = path.join(handoffFolder, "plan.md");
|
|
111
|
-
try {
|
|
112
|
-
if (fs.existsSync(planMdPath)) {
|
|
113
|
-
const content = fs.readFileSync(planMdPath, "utf-8");
|
|
114
|
-
const frontmatter = parseFrontmatter(content);
|
|
115
|
-
if (frontmatter["plan_path"]) {
|
|
116
|
-
const pp = frontmatter["plan_path"];
|
|
117
|
-
if (fs.existsSync(pp)) return pp;
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
} catch {
|
|
121
|
-
// fall through to context state
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
// Fall back to context state
|
|
125
|
-
try {
|
|
126
|
-
const context = getContext(contextId, projectRoot);
|
|
127
|
-
if (context?.plan_path && fs.existsSync(context.plan_path)) {
|
|
128
|
-
return context.plan_path;
|
|
129
|
-
}
|
|
130
|
-
} catch {
|
|
131
|
-
// ignore
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
return null;
|
|
135
|
-
}
|
|
136
|
-
|
|
137
|
-
/**
|
|
138
|
-
* Parse YAML-style frontmatter from markdown content.
|
|
139
|
-
* Returns key-value pairs from the --- block.
|
|
140
|
-
*/
|
|
141
|
-
function parseFrontmatter(content: string): Record<string, string> {
|
|
142
|
-
const frontmatter: Record<string, string> = {};
|
|
143
|
-
if (!content.startsWith("---")) return frontmatter;
|
|
144
|
-
|
|
145
|
-
const parts = content.split("---", 3);
|
|
146
|
-
if (parts.length < 3) return frontmatter;
|
|
147
|
-
|
|
148
|
-
for (const line of parts[1]!.trim().split(/\r?\n/)) {
|
|
149
|
-
const colonIdx = line.indexOf(":");
|
|
150
|
-
if (colonIdx !== -1) {
|
|
151
|
-
const key = line.slice(0, colonIdx).trim();
|
|
152
|
-
const value = line.slice(colonIdx + 1).trim();
|
|
153
|
-
frontmatter[key] = value;
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
return frontmatter;
|
|
158
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Handoff reader utilities for programmatic resume.
|
|
3
|
+
*
|
|
4
|
+
* Provides functions to find, read, and parse handoff folders
|
|
5
|
+
* created by save_handoff.ts. Used by resume_handoff.ts script
|
|
6
|
+
* and potentially by session_start hooks.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import * as fs from "node:fs";
|
|
10
|
+
import * as path from "node:path";
|
|
11
|
+
import { getContextHandoffsDir } from "../../lib-ts/base/constants.js";
|
|
12
|
+
import { getContext } from "../../lib-ts/context/context-store.js";
|
|
13
|
+
import type { HandoffSections } from "../../lib-ts/types.js";
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Find the most recent handoff folder for a context.
|
|
17
|
+
* Lists subdirectories in the handoffs dir, sorts by name
|
|
18
|
+
* (YYYY-MM-DD-HHMM format ensures lexicographic = chronological).
|
|
19
|
+
* Returns full path to most recent folder, or null.
|
|
20
|
+
*/
|
|
21
|
+
export function findLatestHandoff(contextId: string, projectRoot?: string): string | null {
|
|
22
|
+
const handoffsDir = getContextHandoffsDir(contextId, projectRoot);
|
|
23
|
+
|
|
24
|
+
try {
|
|
25
|
+
if (!fs.existsSync(handoffsDir)) return null;
|
|
26
|
+
const entries = fs.readdirSync(handoffsDir, { withFileTypes: true })
|
|
27
|
+
.filter(e => e.isDirectory())
|
|
28
|
+
.map(e => e.name)
|
|
29
|
+
.sort();
|
|
30
|
+
|
|
31
|
+
if (entries.length === 0) return null;
|
|
32
|
+
return path.join(handoffsDir, entries[entries.length - 1]!);
|
|
33
|
+
} catch {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Read all section files from a handoff folder, structured by priority.
|
|
40
|
+
* Returns content for each file (null if missing/unreadable).
|
|
41
|
+
*/
|
|
42
|
+
export function readHandoffSections(handoffFolder: string): HandoffSections {
|
|
43
|
+
const fileMap: Record<keyof HandoffSections, string> = {
|
|
44
|
+
index: "index.md",
|
|
45
|
+
deadEnds: "dead-ends.md",
|
|
46
|
+
pending: "pending.md",
|
|
47
|
+
plan: "plan.md",
|
|
48
|
+
decisions: "decisions.md",
|
|
49
|
+
completedWork: "completed-work.md",
|
|
50
|
+
context: "context.md",
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
const sections: HandoffSections = {
|
|
54
|
+
index: null,
|
|
55
|
+
deadEnds: null,
|
|
56
|
+
pending: null,
|
|
57
|
+
plan: null,
|
|
58
|
+
decisions: null,
|
|
59
|
+
completedWork: null,
|
|
60
|
+
context: null,
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
for (const [key, filename] of Object.entries(fileMap)) {
|
|
64
|
+
const filePath = path.join(handoffFolder, filename);
|
|
65
|
+
try {
|
|
66
|
+
if (fs.existsSync(filePath)) {
|
|
67
|
+
sections[key as keyof HandoffSections] = fs.readFileSync(filePath, "utf-8");
|
|
68
|
+
}
|
|
69
|
+
} catch {
|
|
70
|
+
// graceful — leave as null
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return sections;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Parse the handoff folder name to extract creation timestamp.
|
|
79
|
+
* Expects YYYY-MM-DD-HHMM format (with optional -N suffix for collisions).
|
|
80
|
+
*/
|
|
81
|
+
export function getHandoffTimestamp(handoffFolder: string): Date | null {
|
|
82
|
+
const basename = path.basename(handoffFolder);
|
|
83
|
+
// Match YYYY-MM-DD-HHMM with optional collision suffix
|
|
84
|
+
const match = basename.match(/^(\d{4})-(\d{2})-(\d{2})-(\d{2})(\d{2})/);
|
|
85
|
+
if (!match) return null;
|
|
86
|
+
|
|
87
|
+
const [, year, month, day, hour, minute] = match;
|
|
88
|
+
const date = new Date(
|
|
89
|
+
parseInt(year!, 10),
|
|
90
|
+
parseInt(month!, 10) - 1,
|
|
91
|
+
parseInt(day!, 10),
|
|
92
|
+
parseInt(hour!, 10),
|
|
93
|
+
parseInt(minute!, 10),
|
|
94
|
+
);
|
|
95
|
+
|
|
96
|
+
return isNaN(date.getTime()) ? null : date;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
/**
|
|
100
|
+
* Get the plan path referenced by a handoff.
|
|
101
|
+
* First checks plan.md frontmatter for plan_path, then falls back
|
|
102
|
+
* to the context state's plan_path.
|
|
103
|
+
*/
|
|
104
|
+
export function getHandoffPlanReference(
|
|
105
|
+
handoffFolder: string,
|
|
106
|
+
contextId: string,
|
|
107
|
+
projectRoot?: string,
|
|
108
|
+
): string | null {
|
|
109
|
+
// Try plan.md frontmatter
|
|
110
|
+
const planMdPath = path.join(handoffFolder, "plan.md");
|
|
111
|
+
try {
|
|
112
|
+
if (fs.existsSync(planMdPath)) {
|
|
113
|
+
const content = fs.readFileSync(planMdPath, "utf-8");
|
|
114
|
+
const frontmatter = parseFrontmatter(content);
|
|
115
|
+
if (frontmatter["plan_path"]) {
|
|
116
|
+
const pp = frontmatter["plan_path"];
|
|
117
|
+
if (fs.existsSync(pp)) return pp;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
} catch {
|
|
121
|
+
// fall through to context state
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
// Fall back to context state
|
|
125
|
+
try {
|
|
126
|
+
const context = getContext(contextId, projectRoot);
|
|
127
|
+
if (context?.plan_path && fs.existsSync(context.plan_path)) {
|
|
128
|
+
return context.plan_path;
|
|
129
|
+
}
|
|
130
|
+
} catch {
|
|
131
|
+
// ignore
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
return null;
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Parse YAML-style frontmatter from markdown content.
|
|
139
|
+
* Returns key-value pairs from the --- block.
|
|
140
|
+
*/
|
|
141
|
+
function parseFrontmatter(content: string): Record<string, string> {
|
|
142
|
+
const frontmatter: Record<string, string> = {};
|
|
143
|
+
if (!content.startsWith("---")) return frontmatter;
|
|
144
|
+
|
|
145
|
+
const parts = content.split("---", 3);
|
|
146
|
+
if (parts.length < 3) return frontmatter;
|
|
147
|
+
|
|
148
|
+
for (const line of parts[1]!.trim().split(/\r?\n/)) {
|
|
149
|
+
const colonIdx = line.indexOf(":");
|
|
150
|
+
if (colonIdx !== -1) {
|
|
151
|
+
const key = line.slice(0, colonIdx).trim();
|
|
152
|
+
const value = line.slice(colonIdx + 1).trim();
|
|
153
|
+
frontmatter[key] = value;
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
return frontmatter;
|
|
158
|
+
}
|