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,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
|
+
}
|