aiwcli 0.12.3 → 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.
Files changed (125) 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 +205 -205
  10. package/dist/templates/_shared/.claude/commands/handoff-resume.md +12 -64
  11. package/dist/templates/_shared/.claude/commands/handoff.md +12 -198
  12. package/dist/templates/_shared/.claude/settings.json +65 -65
  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 +421 -0
  16. package/dist/templates/_shared/{lib-ts/handoff → handoff-system/lib}/document-generator.ts +215 -216
  17. package/dist/templates/_shared/{lib-ts/handoff → handoff-system/lib}/handoff-reader.ts +157 -158
  18. package/dist/templates/_shared/{scripts → handoff-system/scripts}/resume_handoff.ts +373 -373
  19. package/dist/templates/_shared/{scripts → handoff-system/scripts}/save_handoff.ts +469 -358
  20. package/dist/templates/_shared/handoff-system/workflows/handoff-resume.md +66 -0
  21. package/dist/templates/_shared/{workflows → 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 -183
  28. package/dist/templates/_shared/hooks-ts/session_start.ts +163 -151
  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 +303 -303
  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 -130
  40. package/dist/templates/_shared/lib-ts/base/stop-words.ts +184 -184
  41. package/dist/templates/_shared/lib-ts/base/subprocess-utils.ts +56 -0
  42. package/dist/templates/_shared/lib-ts/base/utils.ts +184 -184
  43. package/dist/templates/_shared/lib-ts/context/context-formatter.ts +566 -560
  44. package/dist/templates/_shared/lib-ts/context/context-selector.ts +524 -515
  45. package/dist/templates/_shared/lib-ts/context/context-store.ts +712 -668
  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 -180
  53. package/dist/templates/_shared/scripts/resolve_context.ts +33 -33
  54. package/dist/templates/_shared/scripts/status_line.ts +690 -690
  55. package/dist/templates/cc-native/.claude/commands/{rlm → cc-native/rlm}/ask.md +136 -136
  56. package/dist/templates/cc-native/.claude/commands/{rlm → cc-native/rlm}/index.md +21 -21
  57. package/dist/templates/cc-native/.claude/commands/{rlm → 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/.windsurf/workflows/cc-native/fix.md +8 -8
  60. package/dist/templates/cc-native/.windsurf/workflows/cc-native/implement.md +8 -8
  61. package/dist/templates/cc-native/.windsurf/workflows/cc-native/research.md +8 -8
  62. package/dist/templates/cc-native/CC-NATIVE-README.md +189 -189
  63. package/dist/templates/cc-native/TEMPLATE-SCHEMA.md +304 -304
  64. package/dist/templates/cc-native/_cc-native/agents/CLAUDE.md +143 -143
  65. package/dist/templates/cc-native/_cc-native/agents/PLAN-ORCHESTRATOR.md +213 -213
  66. package/dist/templates/cc-native/_cc-native/agents/plan-questions/PLAN-QUESTIONER.md +70 -70
  67. package/dist/templates/cc-native/_cc-native/cc-native.config.json +96 -96
  68. package/dist/templates/cc-native/_cc-native/hooks/CLAUDE.md +247 -247
  69. package/dist/templates/cc-native/_cc-native/hooks/cc-native-plan-review.ts +76 -76
  70. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_subagent.ts +54 -54
  71. package/dist/templates/cc-native/_cc-native/hooks/enhance_plan_post_write.ts +51 -51
  72. package/dist/templates/cc-native/_cc-native/hooks/mark_questions_asked.ts +53 -53
  73. package/dist/templates/cc-native/_cc-native/hooks/plan_questions_early.ts +61 -61
  74. package/dist/templates/cc-native/_cc-native/lib-ts/agent-selection.ts +163 -163
  75. package/dist/templates/cc-native/_cc-native/lib-ts/aggregate-agents.ts +156 -156
  76. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/format.ts +597 -597
  77. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/index.ts +26 -26
  78. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/tracker.ts +107 -107
  79. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts/write.ts +119 -119
  80. package/dist/templates/cc-native/_cc-native/lib-ts/artifacts.ts +21 -21
  81. package/dist/templates/cc-native/_cc-native/lib-ts/cc-native-state.ts +319 -319
  82. package/dist/templates/cc-native/_cc-native/lib-ts/cli-output-parser.ts +144 -144
  83. package/dist/templates/cc-native/_cc-native/lib-ts/config.ts +57 -57
  84. package/dist/templates/cc-native/_cc-native/lib-ts/constants.ts +83 -83
  85. package/dist/templates/cc-native/_cc-native/lib-ts/corroboration.ts +119 -119
  86. package/dist/templates/cc-native/_cc-native/lib-ts/debug.ts +79 -79
  87. package/dist/templates/cc-native/_cc-native/lib-ts/graduation.ts +132 -132
  88. package/dist/templates/cc-native/_cc-native/lib-ts/index.ts +116 -116
  89. package/dist/templates/cc-native/_cc-native/lib-ts/json-parser.ts +168 -168
  90. package/dist/templates/cc-native/_cc-native/lib-ts/orchestrator.ts +70 -70
  91. package/dist/templates/cc-native/_cc-native/lib-ts/output-builder.ts +130 -130
  92. package/dist/templates/cc-native/_cc-native/lib-ts/plan-discovery.ts +80 -80
  93. package/dist/templates/cc-native/_cc-native/lib-ts/plan-enhancement.ts +41 -41
  94. package/dist/templates/cc-native/_cc-native/lib-ts/plan-questions.ts +101 -101
  95. package/dist/templates/cc-native/_cc-native/lib-ts/review-pipeline.ts +511 -511
  96. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/agent.ts +71 -71
  97. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/base/base-agent.ts +217 -217
  98. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/index.ts +12 -12
  99. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/claude-agent.ts +66 -65
  100. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/codex-agent.ts +184 -184
  101. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/gemini-agent.ts +39 -39
  102. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/providers/orchestrator-claude-agent.ts +196 -195
  103. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/schemas.ts +201 -201
  104. package/dist/templates/cc-native/_cc-native/lib-ts/reviewers/types.ts +21 -21
  105. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/CLAUDE.md +480 -480
  106. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/embedding-indexer.ts +287 -287
  107. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/hyde.ts +148 -148
  108. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/index.ts +54 -54
  109. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/logger.ts +58 -58
  110. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/ollama-client.ts +208 -208
  111. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/retrieval-pipeline.ts +460 -460
  112. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-indexer.ts +446 -447
  113. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-loader.ts +280 -280
  114. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/transcript-searcher.ts +274 -274
  115. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/types.ts +201 -201
  116. package/dist/templates/cc-native/_cc-native/lib-ts/rlm/vector-store.ts +278 -278
  117. package/dist/templates/cc-native/_cc-native/lib-ts/settings.ts +184 -184
  118. package/dist/templates/cc-native/_cc-native/lib-ts/state.ts +275 -275
  119. package/dist/templates/cc-native/_cc-native/lib-ts/tsconfig.json +18 -18
  120. package/dist/templates/cc-native/_cc-native/lib-ts/types.ts +329 -329
  121. package/dist/templates/cc-native/_cc-native/lib-ts/verdict.ts +72 -72
  122. package/dist/templates/cc-native/_cc-native/workflows/specdev.md +9 -9
  123. package/oclif.manifest.json +1 -1
  124. package/package.json +108 -108
  125. package/dist/templates/cc-native/_cc-native/lib-ts/nul +0 -3
@@ -1,306 +1,306 @@
1
- /**
2
- * Constants and path utilities for shared context management.
3
- * See SPEC.md §2
4
- */
5
-
6
- import * as fs from "node:fs";
1
+ /**
2
+ * Constants and path utilities for shared context management.
3
+ * See SPEC.md §2
4
+ */
5
+
6
+ import * as fs from "node:fs";
7
7
  import * as path from "node:path";
8
-
9
- import { logWarn } from "./logger.js";
10
-
11
- // Directory names (relative to project root)
12
- const OUTPUT_DIR = "_output";
13
- const CONTEXTS_DIR = "contexts";
14
- const ARCHIVE_DIR = "_archive";
15
- const INDEX_FILENAME = "index.json";
16
-
17
- // Context ID validation
18
- export const MAX_CONTEXT_ID_LENGTH = 64;
19
- export const VALID_CONTEXT_ID_PATTERN =
20
- /^[a-z0-9][a-z0-9_-]*[a-z0-9]$|^[a-z0-9]$/;
21
-
22
- // File size limits
23
- export const MAX_EVENT_SIZE = 64 * 1024;
24
- export const MAX_INDEX_SIZE = 1024 * 1024;
25
-
26
- // Performance constants
27
- export const MAX_RETRY_ATTEMPTS = 2;
28
- export const RETRY_BACKOFF_MS = [500, 1000];
29
-
30
- // Windows reserved filenames
31
- const WINDOWS_RESERVED = new Set([
32
- "AUX", "COM1", "COM2", "COM3",
33
- "COM4", "COM5", "COM6", "COM7", "COM8", "COM9", "CON", "LPT1", "LPT2",
34
- "LPT3", "LPT4", "LPT5", "LPT6", "LPT7", "LPT8", "LPT9", "NUL", "PRN",
35
- ]);
36
-
37
- /**
38
- * Sanitize a string into a valid context ID.
39
- * See SPEC.md §2.3
40
- */
41
- export function sanitizeContextId(contextId: string): string {
42
- if (!contextId) return "context";
43
-
44
- let result = contextId.toLowerCase();
45
- result = result.replaceAll(/[^a-z0-9_-]/g, "-");
46
- result = result.replaceAll(/[-_]+/g, "-");
47
- result = result.replaceAll(/^[-_]+|[-_]+$/g, "");
48
-
49
- if (result.length > MAX_CONTEXT_ID_LENGTH) {
50
- result = result.slice(0, MAX_CONTEXT_ID_LENGTH).replace(/[-_]+$/, "");
51
- }
52
-
53
- return result || "context";
54
- }
55
-
56
- /**
57
- * Validate and normalize context ID.
58
- * Throws only for security violations (path traversal).
59
- * See SPEC.md §2.3
60
- */
61
- export function validateContextId(contextId: string): string {
62
- if (!contextId) return "context";
63
-
64
- // SECURITY: Check for path traversal BEFORE any normalization
65
- if (
66
- contextId.includes("..") ||
67
- contextId.includes("/") ||
68
- contextId.includes("\\")
69
- ) {
70
- throw new Error(
71
- `Invalid context ID '${contextId}': path traversal not allowed`,
72
- );
73
- }
74
-
75
- // Check for URL-encoded variants
76
- const lower = contextId.toLowerCase();
77
- if (
78
- lower.includes("%2e") ||
79
- lower.includes("%2f") ||
80
- lower.includes("%5c")
81
- ) {
82
- throw new Error(
83
- `Invalid context ID '${contextId}': encoded path traversal not allowed`,
84
- );
85
- }
86
-
87
- return sanitizeContextId(contextId);
88
- }
89
-
90
- /**
91
- * Get project root from environment or cwd.
92
- * Priority: CLAUDE_PROJECT_DIR > payload cwd > process.cwd()
93
- * See SPEC.md §2.2
94
- */
95
- export function getProjectRoot(payloadCwd?: string): string {
96
- const envDir = process.env.CLAUDE_PROJECT_DIR;
97
- if (envDir) {
98
- if (!path.isAbsolute(envDir)) {
99
- logWarn("utils", `CLAUDE_PROJECT_DIR is not absolute: '${envDir}', ignoring`);
100
- } else if (envDir.includes("..")) {
101
- logWarn("utils", `CLAUDE_PROJECT_DIR contains '..': '${envDir}', ignoring`);
102
- } else {
103
- return envDir;
104
- }
8
+
9
+ import { logWarn } from "./logger.js";
10
+
11
+ // Directory names (relative to project root)
12
+ const OUTPUT_DIR = "_output";
13
+ const CONTEXTS_DIR = "contexts";
14
+ const ARCHIVE_DIR = "_archive";
15
+ const INDEX_FILENAME = "index.json";
16
+
17
+ // Context ID validation
18
+ export const MAX_CONTEXT_ID_LENGTH = 64;
19
+ export const VALID_CONTEXT_ID_PATTERN =
20
+ /^[a-z0-9][a-z0-9_-]*[a-z0-9]$|^[a-z0-9]$/;
21
+
22
+ // File size limits
23
+ export const MAX_EVENT_SIZE = 64 * 1024;
24
+ export const MAX_INDEX_SIZE = 1024 * 1024;
25
+
26
+ // Performance constants
27
+ export const MAX_RETRY_ATTEMPTS = 2;
28
+ export const RETRY_BACKOFF_MS = [500, 1000];
29
+
30
+ // Windows reserved filenames
31
+ const WINDOWS_RESERVED = new Set([
32
+ "AUX", "COM1", "COM2", "COM3",
33
+ "COM4", "COM5", "COM6", "COM7", "COM8", "COM9", "CON", "LPT1", "LPT2",
34
+ "LPT3", "LPT4", "LPT5", "LPT6", "LPT7", "LPT8", "LPT9", "NUL", "PRN",
35
+ ]);
36
+
37
+ /**
38
+ * Sanitize a string into a valid context ID.
39
+ * See SPEC.md §2.3
40
+ */
41
+ export function sanitizeContextId(contextId: string): string {
42
+ if (!contextId) return "context";
43
+
44
+ let result = contextId.toLowerCase();
45
+ result = result.replaceAll(/[^a-z0-9_-]/g, "-");
46
+ result = result.replaceAll(/[-_]+/g, "-");
47
+ result = result.replaceAll(/^[-_]+|[-_]+$/g, "");
48
+
49
+ if (result.length > MAX_CONTEXT_ID_LENGTH) {
50
+ result = result.slice(0, MAX_CONTEXT_ID_LENGTH).replace(/[-_]+$/, "");
105
51
  }
106
-
107
- if (payloadCwd) return payloadCwd;
108
- return process.cwd();
109
- }
110
-
111
- // §2.4 Path functions
112
-
113
- export function getAiwcliDir(projectRoot?: string): string {
114
- return path.join(projectRoot ?? getProjectRoot(), ".aiwcli");
115
- }
116
-
117
- export function getOutputDir(projectRoot?: string): string {
118
- return path.join(projectRoot ?? getProjectRoot(), OUTPUT_DIR);
119
- }
120
-
121
- export function getContextsDir(projectRoot?: string): string {
122
- return path.join(getOutputDir(projectRoot), CONTEXTS_DIR);
123
- }
124
-
125
- export function getContextDir(
126
- contextId: string,
127
- projectRoot?: string,
128
- ): string {
129
- const validatedId = validateContextId(contextId);
130
- const contextsDir = getContextsDir(projectRoot);
131
- const resultPath = path.join(contextsDir, validatedId);
132
-
133
- // SECURITY: Verify resolved path stays within contexts directory
134
- const resolved = path.resolve(resultPath);
135
- const contextsResolved = path.resolve(contextsDir);
136
- if (
137
- !resolved.toLowerCase().startsWith(contextsResolved.toLowerCase())
138
- ) {
139
- throw new Error(
140
- `Invalid context ID '${contextId}': path escapes contexts directory`,
141
- );
142
- }
143
-
144
- return resultPath;
145
- }
146
-
147
- export function getContextPlansDir(
148
- contextId: string,
149
- projectRoot?: string,
150
- ): string {
151
- return path.join(getContextDir(contextId, projectRoot), "plans");
152
- }
153
-
154
- export function getContextHandoffsDir(
155
- contextId: string,
156
- projectRoot?: string,
157
- ): string {
158
- return path.join(getContextDir(contextId, projectRoot), "handoffs");
159
- }
160
-
161
- export function getContextReviewsDir(
162
- contextId: string,
163
- projectRoot?: string,
164
- ): string {
165
- return path.join(getContextDir(contextId, projectRoot), "reviews");
166
- }
167
-
168
- export function getIndexPath(projectRoot?: string): string {
169
- return path.join(getOutputDir(projectRoot), INDEX_FILENAME);
170
- }
171
-
172
- export function getContextFilePath(
173
- contextId: string,
174
- projectRoot?: string,
175
- ): string {
176
- return path.join(getContextDir(contextId, projectRoot), "context.json");
177
- }
178
-
179
- export function getEventsFilePath(
180
- contextId: string,
181
- projectRoot?: string,
182
- ): string {
183
- return path.join(getContextDir(contextId, projectRoot), "events.jsonl");
184
- }
185
-
186
- export function getAutoStatePath(
187
- contextId: string,
188
- projectRoot?: string,
189
- ): string {
190
- return path.join(
191
- getContextDir(contextId, projectRoot),
192
- "auto-state.json",
193
- );
194
- }
195
-
196
- export function getArchiveDir(projectRoot?: string): string {
197
- return path.join(getContextsDir(projectRoot), ARCHIVE_DIR);
198
- }
199
-
200
- export function getArchiveContextDir(
201
- contextId: string,
202
- projectRoot?: string,
203
- ): string {
204
- const validatedId = validateContextId(contextId);
205
- return path.join(getArchiveDir(projectRoot), validatedId);
206
- }
207
-
208
- export function getArchiveIndexPath(projectRoot?: string): string {
209
- return path.join(getArchiveDir(projectRoot), INDEX_FILENAME);
210
- }
211
-
212
- /**
213
- * Get path for a new handoff folder with datetime naming.
214
- * Handles collisions by appending -N suffix.
215
- * See SPEC.md §2.4
216
- */
217
- export function getHandoffFolderPath(
218
- contextId: string,
219
- projectRoot?: string,
220
- ): string {
221
- const handoffsDir = getContextHandoffsDir(contextId, projectRoot);
222
- const now = new Date();
223
- const timestamp = [
224
- now.getFullYear().toString(),
225
- String(now.getMonth() + 1).padStart(2, "0"),
226
- String(now.getDate()).padStart(2, "0"),
227
- "-",
228
- String(now.getHours()).padStart(2, "0"),
229
- String(now.getMinutes()).padStart(2, "0"),
230
- ].join("");
231
- // Format: YYYY-MM-DD-HHMM
232
- const ts = `${timestamp.slice(0, 4)}-${timestamp.slice(4, 6)}-${timestamp.slice(6, 8)}${timestamp.slice(8)}`;
233
-
234
- let folder = path.join(handoffsDir, ts);
235
- let counter = 1;
236
- while (fs.existsSync(folder)) {
237
- folder = path.join(handoffsDir, `${ts}-${counter}`);
238
- counter++;
52
+
53
+ return result || "context";
54
+ }
55
+
56
+ /**
57
+ * Validate and normalize context ID.
58
+ * Throws only for security violations (path traversal).
59
+ * See SPEC.md §2.3
60
+ */
61
+ export function validateContextId(contextId: string): string {
62
+ if (!contextId) return "context";
63
+
64
+ // SECURITY: Check for path traversal BEFORE any normalization
65
+ if (
66
+ contextId.includes("..") ||
67
+ contextId.includes("/") ||
68
+ contextId.includes("\\")
69
+ ) {
70
+ throw new Error(
71
+ `Invalid context ID '${contextId}': path traversal not allowed`,
72
+ );
239
73
  }
240
-
241
- return folder;
242
- }
243
-
244
- /**
245
- * Get path for a new review folder.
246
- * See SPEC.md §2.4
247
- */
248
- export function getReviewFolderPath(
249
- contextId: string,
250
- iteration: number,
251
- projectRoot?: string,
252
- ): string {
253
- const reviewsDir = path.join(
254
- getContextReviewsDir(contextId, projectRoot),
255
- "cc-native",
256
- );
257
- const now = new Date();
258
- const ts = [
259
- now.getFullYear().toString(),
260
- "-",
261
- String(now.getMonth() + 1).padStart(2, "0"),
262
- "-",
263
- String(now.getDate()).padStart(2, "0"),
264
- "-",
265
- String(now.getHours()).padStart(2, "0"),
266
- String(now.getMinutes()).padStart(2, "0"),
267
- ].join("");
268
- return path.join(reviewsDir, `${ts}-iteration-${iteration}`);
269
- }
270
-
271
- // §2.5 — Filename sanitization
272
-
273
- export function sanitizeFilename(
274
- s: string,
275
- maxLen = 32,
276
- allowLeadingDot = false,
277
- ): string {
278
- let result = s.replaceAll(/[^A-Za-z0-9._-]+/g, "_");
279
- result = result.replaceAll(/^[._-]+|[._-]+$/g, "").slice(0, maxLen) || "unknown";
280
-
281
- if (!allowLeadingDot) {
282
- result = result.replace(/^\.+/, "");
283
- }
284
-
285
- const baseName = (result.split(".")[0] ?? result).toUpperCase();
286
- if (WINDOWS_RESERVED.has(baseName)) {
287
- result = `_${result}`;
288
- }
289
-
290
- return result || "unknown";
291
- }
292
-
293
- export function sanitizeTitle(s: string, maxLen = 50): string {
294
- let result = s.toLowerCase().trim();
295
- result = result.replaceAll(' ', "-");
296
- result = result.replaceAll(/[^a-z0-9._-]+/g, "_");
297
- result = result.replaceAll(/[-_]+/g, "-");
298
- result = result.replaceAll(/^[._-]+|[._-]+$/g, "").slice(0, maxLen) || "unknown";
299
-
300
- const baseName = (result.split(".")[0] ?? result).toUpperCase();
301
- if (WINDOWS_RESERVED.has(baseName)) {
302
- result = `_${result}`;
303
- }
304
-
305
- return result || "unknown";
306
- }
74
+
75
+ // Check for URL-encoded variants
76
+ const lower = contextId.toLowerCase();
77
+ if (
78
+ lower.includes("%2e") ||
79
+ lower.includes("%2f") ||
80
+ lower.includes("%5c")
81
+ ) {
82
+ throw new Error(
83
+ `Invalid context ID '${contextId}': encoded path traversal not allowed`,
84
+ );
85
+ }
86
+
87
+ return sanitizeContextId(contextId);
88
+ }
89
+
90
+ /**
91
+ * Get project root from environment or cwd.
92
+ * Priority: CLAUDE_PROJECT_DIR > payload cwd > process.cwd()
93
+ * See SPEC.md §2.2
94
+ */
95
+ export function getProjectRoot(payloadCwd?: string): string {
96
+ const envDir = process.env.CLAUDE_PROJECT_DIR;
97
+ if (envDir) {
98
+ if (!path.isAbsolute(envDir)) {
99
+ logWarn("utils", `CLAUDE_PROJECT_DIR is not absolute: '${envDir}', ignoring`);
100
+ } else if (envDir.includes("..")) {
101
+ logWarn("utils", `CLAUDE_PROJECT_DIR contains '..': '${envDir}', ignoring`);
102
+ } else {
103
+ return envDir;
104
+ }
105
+ }
106
+
107
+ if (payloadCwd) return payloadCwd;
108
+ return process.cwd();
109
+ }
110
+
111
+ // §2.4 — Path functions
112
+
113
+ export function getAiwcliDir(projectRoot?: string): string {
114
+ return path.join(projectRoot ?? getProjectRoot(), ".aiwcli");
115
+ }
116
+
117
+ export function getOutputDir(projectRoot?: string): string {
118
+ return path.join(projectRoot ?? getProjectRoot(), OUTPUT_DIR);
119
+ }
120
+
121
+ export function getContextsDir(projectRoot?: string): string {
122
+ return path.join(getOutputDir(projectRoot), CONTEXTS_DIR);
123
+ }
124
+
125
+ export function getContextDir(
126
+ contextId: string,
127
+ projectRoot?: string,
128
+ ): string {
129
+ const validatedId = validateContextId(contextId);
130
+ const contextsDir = getContextsDir(projectRoot);
131
+ const resultPath = path.join(contextsDir, validatedId);
132
+
133
+ // SECURITY: Verify resolved path stays within contexts directory
134
+ const resolved = path.resolve(resultPath);
135
+ const contextsResolved = path.resolve(contextsDir);
136
+ if (
137
+ !resolved.toLowerCase().startsWith(contextsResolved.toLowerCase())
138
+ ) {
139
+ throw new Error(
140
+ `Invalid context ID '${contextId}': path escapes contexts directory`,
141
+ );
142
+ }
143
+
144
+ return resultPath;
145
+ }
146
+
147
+ export function getContextPlansDir(
148
+ contextId: string,
149
+ projectRoot?: string,
150
+ ): string {
151
+ return path.join(getContextDir(contextId, projectRoot), "plans");
152
+ }
153
+
154
+ export function getContextHandoffsDir(
155
+ contextId: string,
156
+ projectRoot?: string,
157
+ ): string {
158
+ return path.join(getContextDir(contextId, projectRoot), "handoffs");
159
+ }
160
+
161
+ export function getContextReviewsDir(
162
+ contextId: string,
163
+ projectRoot?: string,
164
+ ): string {
165
+ return path.join(getContextDir(contextId, projectRoot), "reviews");
166
+ }
167
+
168
+ export function getIndexPath(projectRoot?: string): string {
169
+ return path.join(getOutputDir(projectRoot), INDEX_FILENAME);
170
+ }
171
+
172
+ export function getContextFilePath(
173
+ contextId: string,
174
+ projectRoot?: string,
175
+ ): string {
176
+ return path.join(getContextDir(contextId, projectRoot), "context.json");
177
+ }
178
+
179
+ export function getEventsFilePath(
180
+ contextId: string,
181
+ projectRoot?: string,
182
+ ): string {
183
+ return path.join(getContextDir(contextId, projectRoot), "events.jsonl");
184
+ }
185
+
186
+ export function getAutoStatePath(
187
+ contextId: string,
188
+ projectRoot?: string,
189
+ ): string {
190
+ return path.join(
191
+ getContextDir(contextId, projectRoot),
192
+ "auto-state.json",
193
+ );
194
+ }
195
+
196
+ export function getArchiveDir(projectRoot?: string): string {
197
+ return path.join(getContextsDir(projectRoot), ARCHIVE_DIR);
198
+ }
199
+
200
+ export function getArchiveContextDir(
201
+ contextId: string,
202
+ projectRoot?: string,
203
+ ): string {
204
+ const validatedId = validateContextId(contextId);
205
+ return path.join(getArchiveDir(projectRoot), validatedId);
206
+ }
207
+
208
+ export function getArchiveIndexPath(projectRoot?: string): string {
209
+ return path.join(getArchiveDir(projectRoot), INDEX_FILENAME);
210
+ }
211
+
212
+ /**
213
+ * Get path for a new handoff folder with datetime naming.
214
+ * Handles collisions by appending -N suffix.
215
+ * See SPEC.md §2.4
216
+ */
217
+ export function getHandoffFolderPath(
218
+ contextId: string,
219
+ projectRoot?: string,
220
+ ): string {
221
+ const handoffsDir = getContextHandoffsDir(contextId, projectRoot);
222
+ const now = new Date();
223
+ const timestamp = [
224
+ now.getFullYear().toString(),
225
+ String(now.getMonth() + 1).padStart(2, "0"),
226
+ String(now.getDate()).padStart(2, "0"),
227
+ "-",
228
+ String(now.getHours()).padStart(2, "0"),
229
+ String(now.getMinutes()).padStart(2, "0"),
230
+ ].join("");
231
+ // Format: YYYY-MM-DD-HHMM
232
+ const ts = `${timestamp.slice(0, 4)}-${timestamp.slice(4, 6)}-${timestamp.slice(6, 8)}${timestamp.slice(8)}`;
233
+
234
+ let folder = path.join(handoffsDir, ts);
235
+ let counter = 1;
236
+ while (fs.existsSync(folder)) {
237
+ folder = path.join(handoffsDir, `${ts}-${counter}`);
238
+ counter++;
239
+ }
240
+
241
+ return folder;
242
+ }
243
+
244
+ /**
245
+ * Get path for a new review folder.
246
+ * See SPEC.md §2.4
247
+ */
248
+ export function getReviewFolderPath(
249
+ contextId: string,
250
+ iteration: number,
251
+ projectRoot?: string,
252
+ ): string {
253
+ const reviewsDir = path.join(
254
+ getContextReviewsDir(contextId, projectRoot),
255
+ "cc-native",
256
+ );
257
+ const now = new Date();
258
+ const ts = [
259
+ now.getFullYear().toString(),
260
+ "-",
261
+ String(now.getMonth() + 1).padStart(2, "0"),
262
+ "-",
263
+ String(now.getDate()).padStart(2, "0"),
264
+ "-",
265
+ String(now.getHours()).padStart(2, "0"),
266
+ String(now.getMinutes()).padStart(2, "0"),
267
+ ].join("");
268
+ return path.join(reviewsDir, `${ts}-iteration-${iteration}`);
269
+ }
270
+
271
+ // §2.5 — Filename sanitization
272
+
273
+ export function sanitizeFilename(
274
+ s: string,
275
+ maxLen = 32,
276
+ allowLeadingDot = false,
277
+ ): string {
278
+ let result = s.replaceAll(/[^A-Za-z0-9._-]+/g, "_");
279
+ result = result.replaceAll(/^[._-]+|[._-]+$/g, "").slice(0, maxLen) || "unknown";
280
+
281
+ if (!allowLeadingDot) {
282
+ result = result.replace(/^\.+/, "");
283
+ }
284
+
285
+ const baseName = (result.split(".")[0] ?? result).toUpperCase();
286
+ if (WINDOWS_RESERVED.has(baseName)) {
287
+ result = `_${result}`;
288
+ }
289
+
290
+ return result || "unknown";
291
+ }
292
+
293
+ export function sanitizeTitle(s: string, maxLen = 50): string {
294
+ let result = s.toLowerCase().trim();
295
+ result = result.replaceAll(' ', "-");
296
+ result = result.replaceAll(/[^a-z0-9._-]+/g, "_");
297
+ result = result.replaceAll(/[-_]+/g, "-");
298
+ result = result.replaceAll(/^[._-]+|[._-]+$/g, "").slice(0, maxLen) || "unknown";
299
+
300
+ const baseName = (result.split(".")[0] ?? result).toUpperCase();
301
+ if (WINDOWS_RESERVED.has(baseName)) {
302
+ result = `_${result}`;
303
+ }
304
+
305
+ return result || "unknown";
306
+ }