aiwcli 0.12.6 → 0.12.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (163) hide show
  1. package/bin/dev.cmd +3 -3
  2. package/bin/dev.js +16 -16
  3. package/bin/run.cmd +3 -3
  4. package/bin/run.js +21 -21
  5. package/dist/commands/branch.js +7 -2
  6. package/dist/lib/bmad-installer.js +37 -37
  7. package/dist/lib/terminal.d.ts +2 -0
  8. package/dist/lib/terminal.js +57 -7
  9. package/dist/templates/CLAUDE.md +232 -205
  10. package/dist/templates/_shared/.claude/settings.json +65 -65
  11. package/dist/templates/_shared/.claude/{commands/handoff.md → skills/handoff/SKILL.md} +13 -12
  12. package/dist/templates/_shared/.claude/{commands/handoff-resume.md → skills/handoff-resume/SKILL.md} +13 -12
  13. package/dist/templates/_shared/.codex/workflows/handoff.md +226 -226
  14. package/dist/templates/_shared/.windsurf/workflows/handoff.md +226 -226
  15. package/dist/templates/_shared/handoff-system/CLAUDE.md +15 -3
  16. package/dist/templates/_shared/handoff-system/lib/document-generator.ts +215 -215
  17. package/dist/templates/_shared/handoff-system/lib/handoff-reader.ts +158 -158
  18. package/dist/templates/_shared/handoff-system/scripts/resume_handoff.ts +373 -373
  19. package/dist/templates/_shared/handoff-system/scripts/save_handoff.ts +469 -469
  20. package/dist/templates/_shared/handoff-system/workflows/handoff-resume.md +66 -66
  21. package/dist/templates/_shared/handoff-system/workflows/handoff.md +254 -254
  22. package/dist/templates/_shared/hooks-ts/_utils/git-state.ts +2 -2
  23. package/dist/templates/_shared/hooks-ts/archive_plan.ts +159 -159
  24. package/dist/templates/_shared/hooks-ts/context_monitor.ts +147 -147
  25. package/dist/templates/_shared/hooks-ts/file-suggestion.ts +128 -128
  26. package/dist/templates/_shared/hooks-ts/pre_compact.ts +49 -49
  27. package/dist/templates/_shared/hooks-ts/session_end.ts +196 -196
  28. package/dist/templates/_shared/hooks-ts/session_start.ts +163 -163
  29. package/dist/templates/_shared/hooks-ts/task_create_capture.ts +48 -48
  30. package/dist/templates/_shared/hooks-ts/task_update_capture.ts +74 -74
  31. package/dist/templates/_shared/hooks-ts/user_prompt_submit.ts +93 -93
  32. package/dist/templates/_shared/lib-ts/CLAUDE.md +367 -367
  33. package/dist/templates/_shared/lib-ts/base/atomic-write.ts +138 -138
  34. package/dist/templates/_shared/lib-ts/base/constants.ts +24 -6
  35. package/dist/templates/_shared/lib-ts/base/git-state.ts +58 -58
  36. package/dist/templates/_shared/lib-ts/base/hook-utils.ts +582 -582
  37. package/dist/templates/_shared/lib-ts/base/inference.ts +301 -301
  38. package/dist/templates/_shared/lib-ts/base/logger.ts +247 -247
  39. package/dist/templates/_shared/lib-ts/base/state-io.ts +202 -202
  40. package/dist/templates/_shared/lib-ts/base/stop-words.ts +184 -184
  41. package/dist/templates/_shared/lib-ts/base/utils.ts +184 -184
  42. package/dist/templates/_shared/lib-ts/context/CLAUDE.md +134 -0
  43. package/dist/templates/_shared/lib-ts/context/context-formatter.ts +566 -566
  44. package/dist/templates/_shared/lib-ts/context/context-selector.ts +524 -524
  45. package/dist/templates/_shared/lib-ts/context/context-store.ts +712 -712
  46. package/dist/templates/_shared/lib-ts/context/plan-manager.ts +312 -312
  47. package/dist/templates/_shared/lib-ts/context/task-tracker.ts +185 -185
  48. package/dist/templates/_shared/lib-ts/package.json +20 -20
  49. package/dist/templates/_shared/lib-ts/templates/formatters.ts +102 -102
  50. package/dist/templates/_shared/lib-ts/templates/plan-context.ts +58 -58
  51. package/dist/templates/_shared/lib-ts/tsconfig.json +13 -13
  52. package/dist/templates/_shared/lib-ts/types.ts +186 -186
  53. package/dist/templates/_shared/scripts/resolve_context.ts +33 -33
  54. package/dist/templates/_shared/scripts/status_line.ts +687 -690
  55. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/ask.md +136 -136
  56. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/index.md +21 -21
  57. package/dist/templates/cc-native/.claude/commands/cc-native/rlm/overview.md +56 -56
  58. package/dist/templates/cc-native/.claude/commands/cc-native/specdev.md +10 -10
  59. package/dist/templates/cc-native/.claude/settings.json +3 -2
  60. package/dist/templates/cc-native/.windsurf/workflows/cc-native/fix.md +8 -8
  61. package/dist/templates/cc-native/.windsurf/workflows/cc-native/implement.md +8 -8
  62. package/dist/templates/cc-native/.windsurf/workflows/cc-native/research.md +8 -8
  63. package/dist/templates/cc-native/CC-NATIVE-README.md +189 -189
  64. package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +304 -304
  65. package/dist/templates/cc-native/_cc-native/agents/CLAUDE.md +143 -143
  66. package/dist/templates/cc-native/_cc-native/agents/PLAN-ORCHESTRATOR.md +213 -213
  67. package/dist/templates/cc-native/_cc-native/agents/plan-questions/PLAN-QUESTIONER.md +70 -70
  68. package/dist/templates/cc-native/_cc-native/artifacts/CLAUDE.md +64 -0
  69. package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/format.ts +1 -1
  70. package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/write.ts +2 -2
  71. package/dist/templates/cc-native/_cc-native/cc-native.config.json +96 -96
  72. package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +14 -24
  73. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +1 -1
  74. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_subagent.ts +54 -54
  75. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_write.ts +51 -51
  76. package/dist/templates/cc-native/_cc-native/hooks/mark_questions_asked.ts +53 -53
  77. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +61 -61
  78. package/dist/templates/cc-native/_cc-native/hooks/validate_task_prompt.ts +76 -0
  79. package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +9 -2
  80. package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +319 -319
  81. package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +144 -144
  82. package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +57 -57
  83. package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +83 -83
  84. package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +79 -79
  85. package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +4 -4
  86. package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +168 -168
  87. package/dist/templates/cc-native/_cc-native/lib-ts/plan-discovery.ts +80 -80
  88. package/dist/templates/cc-native/_cc-native/lib-ts/plan-enhancement.ts +41 -41
  89. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/CLAUDE.md +480 -480
  90. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/embedding-indexer.ts +287 -287
  91. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/hyde.ts +148 -148
  92. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/index.ts +54 -54
  93. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/logger.ts +58 -58
  94. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/ollama-client.ts +208 -208
  95. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/retrieval-pipeline.ts +460 -460
  96. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-indexer.ts +446 -446
  97. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-loader.ts +280 -280
  98. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-searcher.ts +274 -274
  99. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/types.ts +201 -201
  100. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/vector-store.ts +278 -278
  101. package/dist/templates/cc-native/_cc-native/lib-ts/settings.ts +184 -184
  102. package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +275 -275
  103. package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +18 -18
  104. package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +1 -1
  105. package/dist/templates/cc-native/_cc-native/plan-review/CLAUDE.md +149 -0
  106. package/dist/templates/cc-native/_cc-native/plan-review/agents/CLAUDE.md +143 -0
  107. package/dist/templates/cc-native/_cc-native/plan-review/agents/PLAN-ORCHESTRATOR.md +213 -0
  108. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-questions/PLAN-QUESTIONER.md +70 -0
  109. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-EVOLUTION.md +62 -0
  110. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-PATTERNS.md +61 -0
  111. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ARCH-STRUCTURE.md +62 -0
  112. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/ASSUMPTION-TRACER.md +56 -0
  113. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CLARITY-AUDITOR.md +53 -0
  114. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-FEASIBILITY.md +66 -0
  115. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-GAPS.md +70 -0
  116. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/COMPLETENESS-ORDERING.md +62 -0
  117. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/CONSTRAINT-VALIDATOR.md +72 -0
  118. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-ADR-VALIDATOR.md +61 -0
  119. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DESIGN-SCALE-MATCHER.md +64 -0
  120. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DEVILS-ADVOCATE.md +56 -0
  121. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/DOCUMENTATION-PHILOSOPHY.md +86 -0
  122. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HANDOFF-READINESS.md +59 -0
  123. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/HIDDEN-COMPLEXITY.md +58 -0
  124. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/INCREMENTAL-DELIVERY.md +66 -0
  125. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-DEPENDENCY.md +62 -0
  126. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-FMEA.md +66 -0
  127. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-PREMORTEM.md +71 -0
  128. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/RISK-REVERSIBILITY.md +74 -0
  129. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SCOPE-BOUNDARY.md +77 -0
  130. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SIMPLICITY-GUARDIAN.md +62 -0
  131. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/SKEPTIC.md +68 -0
  132. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-BEHAVIOR-AUDITOR.md +61 -0
  133. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-CHARACTERIZATION.md +71 -0
  134. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-FIRST-VALIDATOR.md +61 -0
  135. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TESTDRIVEN-PYRAMID-ANALYZER.md +61 -0
  136. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-COSTS.md +67 -0
  137. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/TRADEOFF-STAKEHOLDERS.md +65 -0
  138. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-COVERAGE.md +74 -0
  139. package/dist/templates/cc-native/_cc-native/plan-review/agents/plan-review/VERIFY-STRENGTH.md +69 -0
  140. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/agent-selection.ts +3 -3
  141. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/corroboration.ts +1 -1
  142. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/graduation.ts +1 -1
  143. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/orchestrator.ts +2 -2
  144. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/output-builder.ts +3 -3
  145. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/plan-questions.ts +6 -6
  146. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/review-pipeline.ts +15 -15
  147. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/agent.ts +5 -5
  148. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/base/base-agent.ts +4 -4
  149. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/claude-agent.ts +4 -4
  150. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/codex-agent.ts +6 -6
  151. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/gemini-agent.ts +1 -1
  152. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/providers/orchestrator-claude-agent.ts +4 -4
  153. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/types.ts +3 -3
  154. package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/verdict.ts +1 -1
  155. package/oclif.manifest.json +1 -1
  156. package/package.json +108 -108
  157. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +0 -21
  158. package/dist/templates/cc-native/_cc-native/lib-ts/nul +0 -3
  159. /package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/index.ts +0 -0
  160. /package/dist/templates/cc-native/_cc-native/{lib-ts/artifacts → artifacts/lib}/tracker.ts +0 -0
  161. /package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/index.ts +0 -0
  162. /package/dist/templates/cc-native/_cc-native/{lib-ts → plan-review/lib}/reviewers/schemas.ts +0 -0
  163. /package/dist/templates/cc-native/_cc-native/{workflows → plan-review/workflows}/specdev.md +0 -0
@@ -1,184 +1,184 @@
1
- /**
2
- * Core utilities for shared context management.
3
- * See SPEC.md §14.2, §14.3
4
- */
5
-
6
- import { sanitizeTitle } from "./constants.js";
7
- import { logDebug, logError, logWarn } from "./logger.js";
8
- import { STOP_WORDS } from "./stop-words.js";
9
-
10
- /**
11
- * Print to stderr. For terminal-only UX messages, not diagnostics.
12
- */
13
- export function eprint(...args: any[]): void {
14
- process.stderr.write(args.map(String).join(" ") + "\n");
15
- }
16
-
17
- /**
18
- * Get current local datetime as Date.
19
- */
20
- export function nowLocal(): Date {
21
- return new Date();
22
- }
23
-
24
- /**
25
- * Get current time as ISO 8601 string (local time, no timezone suffix).
26
- * Matches Python datetime.now().isoformat() behavior.
27
- */
28
- export function nowIso(): string {
29
- const d = new Date();
30
- const year = d.getFullYear();
31
- const month = String(d.getMonth() + 1).padStart(2, "0");
32
- const day = String(d.getDate()).padStart(2, "0");
33
- const hours = String(d.getHours()).padStart(2, "0");
34
- const minutes = String(d.getMinutes()).padStart(2, "0");
35
- const seconds = String(d.getSeconds()).padStart(2, "0");
36
- const ms = String(d.getMilliseconds()).padStart(3, "0");
37
- return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}.${ms}`;
38
- }
39
-
40
- /**
41
- * Format datetime for display.
42
- * Returns "YYYY-MM-DD HH:MM:SS"
43
- * See SPEC.md §14.3
44
- */
45
- export function formatTimestamp(dt?: Date): string {
46
- const d = dt ?? nowLocal();
47
- const year = d.getFullYear();
48
- const month = String(d.getMonth() + 1).padStart(2, "0");
49
- const day = String(d.getDate()).padStart(2, "0");
50
- const hours = String(d.getHours()).padStart(2, "0");
51
- const minutes = String(d.getMinutes()).padStart(2, "0");
52
- const seconds = String(d.getSeconds()).padStart(2, "0");
53
- return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
54
- }
55
-
56
- /**
57
- * Parse ISO 8601 timestamp string.
58
- * Returns null if parsing fails.
59
- * See SPEC.md §14.3
60
- */
61
- export function parseIsoTimestamp(isoStr: string): Date | null {
62
- try {
63
- const normalized = isoStr.replace("Z", "+00:00");
64
- const d = new Date(normalized);
65
- if (isNaN(d.getTime())) return null;
66
- return d;
67
- } catch {
68
- return null;
69
- }
70
- }
71
-
72
- /**
73
- * Clean text for stop-word matching in slug generation.
74
- * Strips apostrophes (i'm -> im), removes punctuation, normalizes whitespace.
75
- * See SPEC.md §14.2
76
- */
77
- export function cleanTextForSlug(text: string): string {
78
- if (!text) return "";
79
- let result = text.toLowerCase();
80
- result = result.replaceAll('\'', ""); // i'm -> im, you're -> youre
81
- result = result.replaceAll(/[^a-z0-9\s]/g, " "); // punctuation -> spaces
82
- result = result.replaceAll(/\s+/g, " ").trim();
83
- return result;
84
- }
85
-
86
- /**
87
- * Generate a slug from text using AI inference with stop-word fallbacks.
88
- * Pipeline: AI inference → stop-word post-filter → stop-word fallback → word-length fallback.
89
- * Reusable by both context ID generation and plan archival.
90
- * See SPEC.md §14.2
91
- */
92
- export function generateSlug(
93
- text: string,
94
- maxLen = 150,
95
- fallbackSlug = "context",
96
- ): string {
97
- if (!text || !text.trim()) return fallbackSlug;
98
-
99
- let slug: null | string = null;
100
- const cleanedText = cleanTextForSlug(text);
101
-
102
- // Tier 1: AI inference via generateContextIdSlug (sync — uses execFileSync)
103
- try {
104
- // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef
105
- const { generateContextIdSlug } = require("./inference.js");
106
- const aiSlug = generateContextIdSlug(text);
107
- if (aiSlug) {
108
- const filteredWords = aiSlug
109
- .split(/\s+/)
110
- .filter(
111
- (w: string) => !STOP_WORDS.has(w.toLowerCase()) && w.length > 1,
112
- );
113
- if (filteredWords.length >= 5) {
114
- slug = sanitizeTitle(filteredWords.join(" "), maxLen);
115
- } else {
116
- logDebug(
117
- "utils",
118
- `AI slug too generic after stop-word filter (${filteredWords.length} words remain), using fallback`,
119
- );
120
- }
121
- }
122
- } catch (error: any) {
123
- logWarn("utils", `AI slug generation failed, using fallback: ${error}`);
124
- }
125
-
126
- // Tier 2: Stop-word filtering on cleaned text
127
- if (!slug) {
128
- const words = cleanedText
129
- .split(/\s+/)
130
- .filter((w) => !STOP_WORDS.has(w) && w.length > 1)
131
- .slice(0, 12);
132
- slug = words.length >= 3
133
- ? sanitizeTitle(words.join(" "), maxLen)
134
- : sanitizeTitle(
135
- cleanedText.split(/\s+/).filter((w) => w.length > 2).slice(0, 6).join(" "),
136
- maxLen,
137
- ) || fallbackSlug;
138
- }
139
-
140
- return slug;
141
- }
142
-
143
- /**
144
- * Generate a context ID from a summary string.
145
- * Format: YYMMDD-HHMM-slug
146
- * Delegates slug generation to generateSlug().
147
- * See SPEC.md §14.2
148
- */
149
- export function generateContextId(
150
- summary: string,
151
- existingIds?: Set<string>,
152
- ): string {
153
- const now = new Date();
154
- const yy = String(now.getFullYear()).slice(2);
155
- const mm = String(now.getMonth() + 1).padStart(2, "0");
156
- const dd = String(now.getDate()).padStart(2, "0");
157
- const hh = String(now.getHours()).padStart(2, "0");
158
- const min = String(now.getMinutes()).padStart(2, "0");
159
- const timestamp = `${yy}${mm}${dd}-${hh}${min}`;
160
-
161
- let baseId: string;
162
-
163
- try {
164
- const slug = generateSlug(summary);
165
- baseId = `${timestamp}-${slug}`;
166
- } catch (error: any) {
167
- logError(
168
- "utils",
169
- `Context ID generation failed entirely, using timestamp: ${error}`,
170
- );
171
- baseId = `${timestamp}-context`;
172
- }
173
-
174
- if (!existingIds || !existingIds.has(baseId)) {
175
- return baseId;
176
- }
177
-
178
- let counter = 2;
179
- while (existingIds.has(`${baseId}-${counter}`)) {
180
- counter++;
181
- }
182
-
183
- return `${baseId}-${counter}`;
184
- }
1
+ /**
2
+ * Core utilities for shared context management.
3
+ * See SPEC.md §14.2, §14.3
4
+ */
5
+
6
+ import { sanitizeTitle } from "./constants.js";
7
+ import { logDebug, logError, logWarn } from "./logger.js";
8
+ import { STOP_WORDS } from "./stop-words.js";
9
+
10
+ /**
11
+ * Print to stderr. For terminal-only UX messages, not diagnostics.
12
+ */
13
+ export function eprint(...args: any[]): void {
14
+ process.stderr.write(args.map(String).join(" ") + "\n");
15
+ }
16
+
17
+ /**
18
+ * Get current local datetime as Date.
19
+ */
20
+ export function nowLocal(): Date {
21
+ return new Date();
22
+ }
23
+
24
+ /**
25
+ * Get current time as ISO 8601 string (local time, no timezone suffix).
26
+ * Matches Python datetime.now().isoformat() behavior.
27
+ */
28
+ export function nowIso(): string {
29
+ const d = new Date();
30
+ const year = d.getFullYear();
31
+ const month = String(d.getMonth() + 1).padStart(2, "0");
32
+ const day = String(d.getDate()).padStart(2, "0");
33
+ const hours = String(d.getHours()).padStart(2, "0");
34
+ const minutes = String(d.getMinutes()).padStart(2, "0");
35
+ const seconds = String(d.getSeconds()).padStart(2, "0");
36
+ const ms = String(d.getMilliseconds()).padStart(3, "0");
37
+ return `${year}-${month}-${day}T${hours}:${minutes}:${seconds}.${ms}`;
38
+ }
39
+
40
+ /**
41
+ * Format datetime for display.
42
+ * Returns "YYYY-MM-DD HH:MM:SS"
43
+ * See SPEC.md §14.3
44
+ */
45
+ export function formatTimestamp(dt?: Date): string {
46
+ const d = dt ?? nowLocal();
47
+ const year = d.getFullYear();
48
+ const month = String(d.getMonth() + 1).padStart(2, "0");
49
+ const day = String(d.getDate()).padStart(2, "0");
50
+ const hours = String(d.getHours()).padStart(2, "0");
51
+ const minutes = String(d.getMinutes()).padStart(2, "0");
52
+ const seconds = String(d.getSeconds()).padStart(2, "0");
53
+ return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
54
+ }
55
+
56
+ /**
57
+ * Parse ISO 8601 timestamp string.
58
+ * Returns null if parsing fails.
59
+ * See SPEC.md §14.3
60
+ */
61
+ export function parseIsoTimestamp(isoStr: string): Date | null {
62
+ try {
63
+ const normalized = isoStr.replace("Z", "+00:00");
64
+ const d = new Date(normalized);
65
+ if (isNaN(d.getTime())) return null;
66
+ return d;
67
+ } catch {
68
+ return null;
69
+ }
70
+ }
71
+
72
+ /**
73
+ * Clean text for stop-word matching in slug generation.
74
+ * Strips apostrophes (i'm -> im), removes punctuation, normalizes whitespace.
75
+ * See SPEC.md §14.2
76
+ */
77
+ export function cleanTextForSlug(text: string): string {
78
+ if (!text) return "";
79
+ let result = text.toLowerCase();
80
+ result = result.replaceAll('\'', ""); // i'm -> im, you're -> youre
81
+ result = result.replaceAll(/[^a-z0-9\s]/g, " "); // punctuation -> spaces
82
+ result = result.replaceAll(/\s+/g, " ").trim();
83
+ return result;
84
+ }
85
+
86
+ /**
87
+ * Generate a slug from text using AI inference with stop-word fallbacks.
88
+ * Pipeline: AI inference → stop-word post-filter → stop-word fallback → word-length fallback.
89
+ * Reusable by both context ID generation and plan archival.
90
+ * See SPEC.md §14.2
91
+ */
92
+ export function generateSlug(
93
+ text: string,
94
+ maxLen = 150,
95
+ fallbackSlug = "context",
96
+ ): string {
97
+ if (!text || !text.trim()) return fallbackSlug;
98
+
99
+ let slug: null | string = null;
100
+ const cleanedText = cleanTextForSlug(text);
101
+
102
+ // Tier 1: AI inference via generateContextIdSlug (sync — uses execFileSync)
103
+ try {
104
+ // eslint-disable-next-line @typescript-eslint/no-require-imports, no-undef
105
+ const { generateContextIdSlug } = require("./inference.js");
106
+ const aiSlug = generateContextIdSlug(text);
107
+ if (aiSlug) {
108
+ const filteredWords = aiSlug
109
+ .split(/\s+/)
110
+ .filter(
111
+ (w: string) => !STOP_WORDS.has(w.toLowerCase()) && w.length > 1,
112
+ );
113
+ if (filteredWords.length >= 5) {
114
+ slug = sanitizeTitle(filteredWords.join(" "), maxLen);
115
+ } else {
116
+ logDebug(
117
+ "utils",
118
+ `AI slug too generic after stop-word filter (${filteredWords.length} words remain), using fallback`,
119
+ );
120
+ }
121
+ }
122
+ } catch (error: any) {
123
+ logWarn("utils", `AI slug generation failed, using fallback: ${error}`);
124
+ }
125
+
126
+ // Tier 2: Stop-word filtering on cleaned text
127
+ if (!slug) {
128
+ const words = cleanedText
129
+ .split(/\s+/)
130
+ .filter((w) => !STOP_WORDS.has(w) && w.length > 1)
131
+ .slice(0, 12);
132
+ slug = words.length >= 3
133
+ ? sanitizeTitle(words.join(" "), maxLen)
134
+ : sanitizeTitle(
135
+ cleanedText.split(/\s+/).filter((w) => w.length > 2).slice(0, 6).join(" "),
136
+ maxLen,
137
+ ) || fallbackSlug;
138
+ }
139
+
140
+ return slug;
141
+ }
142
+
143
+ /**
144
+ * Generate a context ID from a summary string.
145
+ * Format: YYMMDD-HHMM-slug
146
+ * Delegates slug generation to generateSlug().
147
+ * See SPEC.md §14.2
148
+ */
149
+ export function generateContextId(
150
+ summary: string,
151
+ existingIds?: Set<string>,
152
+ ): string {
153
+ const now = new Date();
154
+ const yy = String(now.getFullYear()).slice(2);
155
+ const mm = String(now.getMonth() + 1).padStart(2, "0");
156
+ const dd = String(now.getDate()).padStart(2, "0");
157
+ const hh = String(now.getHours()).padStart(2, "0");
158
+ const min = String(now.getMinutes()).padStart(2, "0");
159
+ const timestamp = `${yy}${mm}${dd}-${hh}${min}`;
160
+
161
+ let baseId: string;
162
+
163
+ try {
164
+ const slug = generateSlug(summary);
165
+ baseId = `${timestamp}-${slug}`;
166
+ } catch (error: any) {
167
+ logError(
168
+ "utils",
169
+ `Context ID generation failed entirely, using timestamp: ${error}`,
170
+ );
171
+ baseId = `${timestamp}-context`;
172
+ }
173
+
174
+ if (!existingIds || !existingIds.has(baseId)) {
175
+ return baseId;
176
+ }
177
+
178
+ let counter = 2;
179
+ while (existingIds.has(`${baseId}-${counter}`)) {
180
+ counter++;
181
+ }
182
+
183
+ return `${baseId}-${counter}`;
184
+ }
@@ -0,0 +1,134 @@
1
+ # Context Management System
2
+
3
+ Shared library for context lifecycle management: state machine, session binding, plan tracking, and artifact routing.
4
+
5
+ ## Overview
6
+
7
+ A "context" is a named work session with a state machine that tracks mode transitions, staged artifacts, and session history. Contexts persist to disk as `state.json` files under `_output/contexts/{id}/`. They are the backbone of the handoff and session-restore system.
8
+
9
+ ## File Structure
10
+
11
+ ```
12
+ context/
13
+ ├── CLAUDE.md ← This file
14
+ ├── context-store.ts ← Core CRUD + state machine transitions
15
+ ├── context-formatter.ts ← Context → human-readable text for injection
16
+ ├── context-selector.ts ← Find active/relevant contexts by criteria
17
+ ├── plan-manager.ts ← Plan file archive, discovery, hash, path extraction
18
+ └── task-tracker.ts ← Task progress tracking within a context
19
+ ```
20
+
21
+ ## State Machine
22
+
23
+ ```
24
+ ┌─────────────┐
25
+ │ created │
26
+ └──────┬──────┘
27
+ │ first user prompt / maybeActivate()
28
+
29
+ ┌─────────────┐
30
+ ┌──▶│ active │◀──────────────────┐
31
+ │ └──────┬──────┘ │
32
+ │ │ session_end + artifact │ session_start (clear)
33
+ │ ▼ │
34
+ │ ┌─────────────────┐ │
35
+ │ │ has_staged_work │ ──────────────┘
36
+ │ └──────┬──────────┘
37
+ │ │ work_consumed = true
38
+ │ ▼
39
+ │ ┌─────────────┐
40
+ │ │ archived │──▶ (terminal)
41
+ │ └─────────────┘
42
+
43
+ └─── reopenContext()
44
+ ```
45
+
46
+ **Modes:**
47
+ - `created` — context initialized, not yet bound to a session
48
+ - `active` — session in progress, context is current
49
+ - `has_staged_work` — session ended with staged artifact (plan or handoff)
50
+ - `archived` — context complete and closed
51
+
52
+ ## API Reference
53
+
54
+ ### `context-store.ts`
55
+
56
+ | Function | Purpose |
57
+ |----------|---------|
58
+ | `createContext(id, opts)` | Create new context with initial state |
59
+ | `getContext(contextId)` | Read context by ID |
60
+ | `getAllContexts(mode?, root?)` | List all contexts, optionally filtered by mode |
61
+ | `getContextBySessionId(sessionId, root?)` | Find context that owns a session ID |
62
+ | `updateContext(contextId, patch)` | Partial state update |
63
+ | `bindSession(contextId, sessionId)` | Attach session ID to context |
64
+ | `updateMode(contextId, mode)` | Transition state machine mode |
65
+ | `maybeActivate(contextId)` | Activate if in `created` mode (idempotent) |
66
+ | `completeContext(contextId)` | Mark complete, archive |
67
+ | `archiveContext(contextId)` | Archive without completing |
68
+ | `reopenContext(contextId)` | Transition `archived` → `active` |
69
+ | `createContextFromPrompt(prompt)` | Create context from user prompt text |
70
+ | `loadState(contextId)` | Raw state.json read |
71
+ | `saveState(contextId, state)` | Raw state.json write |
72
+ | `determineArtifactType(context)` | Detect `"plan"` or `"handoff"` from staged state |
73
+
74
+ ### `context-formatter.ts`
75
+
76
+ Formats context state for injection into Claude's context window.
77
+
78
+ | Function | Purpose |
79
+ |----------|---------|
80
+ | `formatContext(context)` | Full context as human-readable markdown |
81
+ | `formatContextSummary(context)` | Short one-line summary |
82
+
83
+ ### `context-selector.ts`
84
+
85
+ Finds contexts by various criteria.
86
+
87
+ | Function | Purpose |
88
+ |----------|---------|
89
+ | `selectActiveContext(root?)` | Find the single active context (errors if multiple) |
90
+ | `findContextByMode(mode, root?)` | Find contexts in a given mode |
91
+ | `findStagedWorkContext(root?)` | Find context with `has_staged_work` mode |
92
+
93
+ ### `plan-manager.ts`
94
+
95
+ Manages plan file lifecycle within a context.
96
+
97
+ | Function | Purpose |
98
+ |----------|---------|
99
+ | `archivePlan(contextId, planPath)` | Copy plan to context's `plans/` folder |
100
+ | `findLatestPlan(contextId)` | Find most recent archived plan |
101
+ | `generatePlanId()` | Generate unique plan ID |
102
+ | `normalizePlanContent(text)` | Strip metadata for hashing |
103
+ | `extractPlanAnchors(content)` | Extract key phrases from plan for matching |
104
+ | `findPlanPathInTranscript(transcriptPath)` | Parse plan path from JSONL transcript |
105
+ | `extractPlanPathFromResult(toolResult)` | Extract plan path from tool result JSON |
106
+
107
+ ### `task-tracker.ts`
108
+
109
+ Tracks task progress (ISC criteria) within a context.
110
+
111
+ | Function | Purpose |
112
+ |----------|---------|
113
+ | `initTaskTracker(contextId)` | Create task tracker for context |
114
+ | `addTask(contextId, task)` | Add tracked task |
115
+ | `updateTask(contextId, taskId, patch)` | Update task status |
116
+ | `getTaskSummary(contextId)` | Progress summary |
117
+
118
+ ## Which Hooks Use This System
119
+
120
+ | Hook | Usage |
121
+ |------|-------|
122
+ | `session_start.ts` | `getContextBySessionId()`, `bindSession()`, `updateMode()`, `getAllContexts()` |
123
+ | `session_end.ts` | `getContextBySessionId()`, `updateMode()`, `saveState()` |
124
+ | `user_prompt_submit.ts` | `getAllContexts()`, `maybeActivate()`, `determineArtifactType()` |
125
+ | `archive_plan.ts` | `getContextBySessionId()`, `archivePlan()` |
126
+ | `save_handoff.ts` | `getContextBySessionId()`, `updateContext()` |
127
+ | `cc-native-plan-review.ts` | `getContextBySessionId()`, `getAllContexts()` |
128
+
129
+ ## Design Decisions
130
+
131
+ - **Single state.json per context:** All state is in one file. No distributed state. Atomic writes prevent corruption.
132
+ - **No moves out of lib-ts:** Context is pure library code imported by ~8 shared hooks. Moving it would require updating all those import paths for no structural gain. The subfolder is already co-located; it just needed documentation.
133
+ - **`maybeActivate()` is idempotent:** Can be called from any hook without checking current mode — safe to call repeatedly.
134
+ - **`determineArtifactType()` drives session restore:** Returns `"plan"` or `"handoff"` to dispatch the correct restoration path in `session_start.ts`.