dual-brain 0.1.22 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,325 @@
1
+ // prompt-intel.mjs — Layer 3 prompt analysis, enrichment, risk detection, and intervention routing.
2
+ // Pure functions only. No I/O, no exec. Caller provides projectBrief and calibration.
3
+
4
+ const INTENT_PATTERNS = {
5
+ fix: /\b(?:fix|bug|broken|error|crash|failing|fails|wrong|issue|problem|broken|not\s+working|doesn't\s+work|doesn't\s+work)\b/i,
6
+ feature: /\b(?:add|create|build|implement|new|introduce|support|enable)\b/i,
7
+ refactor: /\b(?:refactor|clean\s+up|reorganize|simplify|extract|restructure|dedup|consolidate|move)\b/i,
8
+ review: /\b(?:review|check|audit|look\s+at|examine|inspect|assess|evaluate)\b/i,
9
+ ship: /\b(?:ship|deploy|publish|release|push|merge|go\s+live|launch)\b/i,
10
+ explore: /\b(?:what|how|why|where|find|search|explain|show\s+me|tell\s+me|list|which)\b/i,
11
+ test: /\b(?:test|spec|coverage|assert|jest|mocha|vitest|unit\s+test|integration)\b/i,
12
+ docs: /\b(?:doc|readme|comment|jsdoc|document|explain|annotate)\b/i,
13
+ deploy: /\b(?:deploy|provision|infrastructure|ci|cd|pipeline|k8s|kubernetes|docker)\b/i,
14
+ };
15
+
16
+ const RISK_PATTERNS = [
17
+ { type: 'destructive', severity: 'block', re: /\b(?:delete\s+all|remove\s+all|drop\s+all|wipe|destroy|rm\s+-rf|truncate\s+all|nuke)\b/i, detail: 'Prompt contains mass-destructive operation' },
18
+ { type: 'force_push', severity: 'block', re: /(?:force\s+push|--force|-f\s+origin|reset\s+--hard)/i, detail: 'Prompt implies forced git operation' },
19
+ { type: 'secret', severity: 'warn', re: /\b(?:api\s+key|access\s+token|secret\s+key|\.env|private\s+key|bearer\s+token)\b/i, detail: 'Touches secret or credential material' },
20
+ { type: 'auth', severity: 'warn', re: /\b(?:auth(?:entication)?|login|logout|password|credential|jwt|oauth|session\s+token)\b/i, detail: 'Touches authentication code' },
21
+ { type: 'deploy', severity: 'warn', re: /\b(?:ship\s+to|deploy\s+to|release\s+to|push\s+to\s+prod)\b/i, detail: 'Targets a deployment action' },
22
+ { type: 'data_loss', severity: 'warn', re: /\b(?:drop\s+table|alter\s+table|schema\s+migration|migrate\s+data|database\s+reset)\b/i, detail: 'Touches schema or migration — data loss risk' },
23
+ { type: 'production', severity: 'warn', re: /\b(?:production|prod\b|live\s+environment|live\s+site)\b/i, detail: 'References production environment' },
24
+ ];
25
+
26
+ const FILE_REF_RE = /(?:src\/|\.mjs|\.tsx?|\.jsx?|\.json|\.ya?ml|\.sh|line\s+\d+|\bL\d+\b)/i;
27
+ const FUNC_REF_RE = /\b\w+\((?:\)|[^)]{0,40}\))/;
28
+ const STEP_RE = /\b(?:step\s+\d|first[,\s]|then[,\s]|finally[,\s]|must|should\s+(?:use|call|return|handle))\b/i;
29
+ const CRITERIA_RE = /\b(?:accept(?:ance)?\s+criteria|definition\s+of\s+done|constraints?|requirements?|must\s+(?:not|be|have)|should\s+not)\b/i;
30
+
31
+ function clamp(v, min = 1, max = 5) {
32
+ return Math.min(max, Math.max(min, v));
33
+ }
34
+
35
+ function scoreSpecificity(prompt) {
36
+ const hasFile = FILE_REF_RE.test(prompt);
37
+ const hasFunc = FUNC_REF_RE.test(prompt);
38
+ const hasLine = /\bL\d+\b|\bline\s+\d+/i.test(prompt);
39
+ const words = prompt.trim().split(/\s+/).length;
40
+
41
+ if (hasFile && (hasFunc || hasLine)) return 5;
42
+ if (hasFile || hasFunc) return 4;
43
+ if (words >= 10) return 3;
44
+ if (words >= 5) return 2;
45
+ return 1;
46
+ }
47
+
48
+ function scoreActionability(prompt) {
49
+ const hasStep = STEP_RE.test(prompt);
50
+ const hasVerb = /\b(?:fix|add|remove|refactor|create|update|write|delete|move|rename|replace|ensure|make)\b/i.test(prompt);
51
+ const hasOutcome = /\b(?:so\s+that|in\s+order\s+to|result(?:ing)?\s+in|should\s+(?:return|output|produce|show))\b/i.test(prompt);
52
+ const words = prompt.trim().split(/\s+/).length;
53
+
54
+ if (hasStep && hasVerb) return 5;
55
+ if (hasOutcome && hasVerb) return 4;
56
+ if (hasVerb && words >= 6) return 3;
57
+ if (hasVerb) return 2;
58
+ return 1;
59
+ }
60
+
61
+ function scoreSafety(risks) {
62
+ if (risks.some(r => r.severity === 'block')) return 1;
63
+ if (risks.some(r => r.type === 'auth' || r.type === 'secret' || r.type === 'production')) return 2;
64
+ if (risks.some(r => r.type === 'deploy' || r.type === 'data_loss')) return 3;
65
+ if (risks.length > 0) return 4;
66
+ return 5;
67
+ }
68
+
69
+ function scoreCompleteness(prompt) {
70
+ const hasCriteria = CRITERIA_RE.test(prompt);
71
+ const hasContext = /\b(?:because|since|currently|right\s+now|the\s+issue\s+is|error\s+is|it\s+(?:crashes|fails|returns))\b/i.test(prompt);
72
+ const hasScope = FILE_REF_RE.test(prompt);
73
+ const words = prompt.trim().split(/\s+/).length;
74
+
75
+ if (hasCriteria && hasContext && hasScope) return 5;
76
+ if ((hasCriteria || hasContext) && hasScope) return 4;
77
+ if (hasContext || (hasScope && words >= 8)) return 3;
78
+ if (words >= 6) return 2;
79
+ return 1;
80
+ }
81
+
82
+ function detectIntent(prompt) {
83
+ const counts = {};
84
+ for (const [type, re] of Object.entries(INTENT_PATTERNS)) {
85
+ const matches = prompt.match(new RegExp(re.source, 'gi')) ?? [];
86
+ if (matches.length > 0) counts[type] = matches.length;
87
+ }
88
+
89
+ const entries = Object.entries(counts).sort((a, b) => b[1] - a[1]);
90
+ if (entries.length === 0) {
91
+ return { type: 'unknown', confidence: 0, keywords: [] };
92
+ }
93
+
94
+ const [topType, topCount] = entries[0];
95
+ const totalMatches = Object.values(counts).reduce((s, n) => s + n, 0);
96
+ const confidence = Math.min(1, topCount / Math.max(totalMatches, 1) * (entries.length === 1 ? 1.5 : 1));
97
+
98
+ const allWords = prompt.match(new RegExp(INTENT_PATTERNS[topType].source, 'gi')) ?? [];
99
+ const keywords = [...new Set(allWords.map(w => w.toLowerCase().trim()))].slice(0, 5);
100
+
101
+ return { type: topType, confidence: Math.round(confidence * 100) / 100, keywords };
102
+ }
103
+
104
+ function detectRisks(prompt) {
105
+ const found = [];
106
+ for (const { type, severity, re, detail } of RISK_PATTERNS) {
107
+ if (re.test(prompt)) {
108
+ found.push({ type, severity, detail });
109
+ }
110
+ }
111
+ return found;
112
+ }
113
+
114
+ function findMissingInfo(prompt, intent, specificity, completeness) {
115
+ const missing = [];
116
+
117
+ if (intent.type === 'fix') {
118
+ if (!/error|exception|crash|fail|wrong|issue/i.test(prompt)) missing.push('error description or failure symptom');
119
+ if (!FILE_REF_RE.test(prompt)) missing.push('affected file or component');
120
+ if (!/repro|reproduce|steps|trigger|when\s+I/i.test(prompt)) missing.push('reproduction steps');
121
+ }
122
+
123
+ if (intent.type === 'feature') {
124
+ if (!FILE_REF_RE.test(prompt)) missing.push('target file or module to add feature to');
125
+ if (!/user\s+(?:can|should|will)|should\s+(?:allow|enable|support)/i.test(prompt)) missing.push('user-facing outcome');
126
+ }
127
+
128
+ if (intent.type === 'refactor' && !FILE_REF_RE.test(prompt)) {
129
+ missing.push('which file or function to refactor');
130
+ }
131
+
132
+ if (specificity < 2) missing.push('file name or component reference');
133
+ if (completeness < 2) missing.push('context about current behavior');
134
+
135
+ return [...new Set(missing)].slice(0, 3);
136
+ }
137
+
138
+ function chooseIntervention(quality, risks, calibration) {
139
+ if (risks.some(r => r.severity === 'block')) return 'block';
140
+
141
+ const autonomy = calibration?.autonomy ?? 3;
142
+ const specificity = calibration?.specificity ?? 3;
143
+
144
+ if (quality.score >= 4 && risks.length === 0 && autonomy > 4) return 'pass';
145
+ if (quality.score < 2 && specificity < 3) return 'confirm_rewrite';
146
+ if (quality.score < 3 && quality.completeness <= 2) return 'clarify_once';
147
+ return 'silent_enrich';
148
+ }
149
+
150
+ export function analyzePrompt(prompt, projectBrief, calibration) {
151
+ const risks = detectRisks(prompt);
152
+ const intent = detectIntent(prompt);
153
+ const specificity = scoreSpecificity(prompt);
154
+ const actionability= scoreActionability(prompt);
155
+ const safety = scoreSafety(risks);
156
+ const completeness = scoreCompleteness(prompt);
157
+ const score = Math.round(((specificity + actionability + safety + completeness) / 4) * 10) / 10;
158
+
159
+ const quality = { score, specificity, actionability, safety, completeness };
160
+ const missingInfo = findMissingInfo(prompt, intent, specificity, completeness);
161
+ const intervention = chooseIntervention(quality, risks, calibration);
162
+
163
+ return {
164
+ original: prompt,
165
+ quality,
166
+ intent,
167
+ risks,
168
+ missingInfo,
169
+ intervention,
170
+ };
171
+ }
172
+
173
+ function relevantDirtyFiles(dirtyFiles, intent) {
174
+ if (!Array.isArray(dirtyFiles) || dirtyFiles.length === 0) return [];
175
+
176
+ const INTENT_HINTS = {
177
+ fix: /\bfix|bug|error\b/i,
178
+ ship: /.*/,
179
+ review: /.*/,
180
+ feature: /src\//i,
181
+ refactor: /src\//i,
182
+ test: /test|spec/i,
183
+ docs: /\.md$|readme/i,
184
+ };
185
+
186
+ const re = INTENT_HINTS[intent.type] ?? /src\//i;
187
+ return dirtyFiles.filter(f => re.test(f)).slice(0, 4);
188
+ }
189
+
190
+ export function enrichPrompt(prompt, projectBrief, analysis) {
191
+ if (!projectBrief) return prompt;
192
+
193
+ const lines = [prompt, ''];
194
+ const { branch, dirtyFiles = [], recentCommits = [], aheadOfRemote = 0, recentFailures = [] } = projectBrief;
195
+ const { intent } = analysis;
196
+
197
+ const uncommitted = dirtyFiles.length;
198
+ if (branch || uncommitted > 0) {
199
+ const parts = [];
200
+ if (branch) parts.push(`${branch} branch`);
201
+ if (uncommitted > 0) parts.push(`${uncommitted} uncommitted file${uncommitted !== 1 ? 's' : ''}`);
202
+ if (aheadOfRemote > 0) parts.push(`${aheadOfRemote} ahead of remote`);
203
+ lines.push(`[Context: ${parts.join(', ')}]`);
204
+ }
205
+
206
+ const relFiles = relevantDirtyFiles(dirtyFiles, intent);
207
+ if (relFiles.length > 0) {
208
+ lines.push(`[Files: ${relFiles.join(', ')}]`);
209
+ }
210
+
211
+ if (recentCommits.length > 0 && ['fix', 'review', 'ship'].includes(intent.type)) {
212
+ lines.push(`[Recent: ${recentCommits[0].slice(0, 80)}]`);
213
+ }
214
+
215
+ if (recentFailures.length > 0) {
216
+ const related = recentFailures.find(f => {
217
+ const fp = (f.prompt ?? '').toLowerCase();
218
+ const pp = prompt.toLowerCase();
219
+ const words = pp.split(/\s+/).filter(w => w.length > 3);
220
+ return words.some(w => fp.includes(w));
221
+ });
222
+ if (related) {
223
+ lines.push(`[Failures: previous attempt failed — ${(related.error ?? 'unknown error').slice(0, 80)}]`);
224
+ }
225
+ }
226
+
227
+ return lines.slice(0, lines.length).join('\n');
228
+ }
229
+
230
+ export function formatRiskWarning(risks) {
231
+ if (!risks || risks.length === 0) return '';
232
+ const lines = ['⚠️ RISK DETECTED'];
233
+ for (const risk of risks) {
234
+ const icon = risk.severity === 'block' ? '🔴' : '🟡';
235
+ const suffix = risk.severity === 'block' ? ' (BLOCKED)' : ' (proceed with caution)';
236
+ lines.push(` ${icon} ${risk.type}: ${risk.detail}${suffix}`);
237
+ }
238
+ return lines.join('\n');
239
+ }
240
+
241
+ export function formatQuality(analysis) {
242
+ const { quality, intent, intervention } = analysis;
243
+ const q = quality;
244
+ return `Quality: ${q.score}/5 (specificity:${q.specificity} action:${q.actionability} safety:${q.safety} complete:${q.completeness})\nIntent: ${intent.type} (${intent.confidence}) | Intervention: ${intervention}`;
245
+ }
246
+
247
+ export function suggestImprovement(analysis) {
248
+ const { intent, missingInfo, quality } = analysis;
249
+
250
+ const templates = {
251
+ fix: `Fix the [specific function or behavior] in [file path] — it [symptom/error], causing [impact]`,
252
+ feature: `Add [feature name] to [file/component] — it should [user outcome] when [condition]`,
253
+ refactor: `Refactor [function/module] in [file] to [desired outcome] — keep [what to preserve]`,
254
+ review: `Review [file or PR] for [concern] — focus on [specific area or risk]`,
255
+ ship: `Ship [branch/feature] — verify [test status], then merge and publish`,
256
+ explore: `Explain how [specific mechanism] works in [file/component]`,
257
+ test: `Write tests for [function/module] in [file] covering [edge cases]`,
258
+ docs: `Document [function or module] in [file] — cover [params, return, examples]`,
259
+ deploy: `Deploy [service] to [environment] — confirm [readiness checks]`,
260
+ unknown: `Describe what you want changed, which file it's in, and what the expected result is`,
261
+ };
262
+
263
+ const base = templates[intent.type] ?? templates.unknown;
264
+ const tips = missingInfo.length > 0
265
+ ? ` (missing: ${missingInfo.join(', ')})`
266
+ : '';
267
+
268
+ return `Try: '${base}'${tips}`;
269
+ }
270
+
271
+ export function getTaskTemplate(intentType) {
272
+ const templates = {
273
+ fix: {
274
+ needs: ['error_description', 'affected_files', 'reproduction_steps'],
275
+ auto_add: ['test_expectations', 'rollback_plan'],
276
+ },
277
+ feature: {
278
+ needs: ['feature_description', 'affected_area'],
279
+ auto_add: ['test_plan', 'acceptance_criteria'],
280
+ },
281
+ refactor: {
282
+ needs: ['target_code', 'desired_outcome'],
283
+ auto_add: ['test_preservation', 'scope_boundary'],
284
+ },
285
+ review: {
286
+ needs: ['scope'],
287
+ auto_add: ['recent_changes', 'risk_areas'],
288
+ },
289
+ ship: {
290
+ needs: ['readiness_check'],
291
+ auto_add: ['test_status', 'git_status', 'uncommitted_changes'],
292
+ },
293
+ explore: {
294
+ needs: ['topic'],
295
+ auto_add: ['related_files', 'recent_changes'],
296
+ },
297
+ test: {
298
+ needs: ['target_function', 'test_framework'],
299
+ auto_add: ['edge_cases', 'existing_coverage'],
300
+ },
301
+ docs: {
302
+ needs: ['target_module'],
303
+ auto_add: ['param_descriptions', 'usage_example'],
304
+ },
305
+ deploy: {
306
+ needs: ['target_environment', 'service_name'],
307
+ auto_add: ['pre_deploy_checks', 'rollback_plan'],
308
+ },
309
+ unknown: {
310
+ needs: ['prompt_clarification'],
311
+ auto_add: [],
312
+ },
313
+ };
314
+
315
+ return templates[intentType] ?? templates.unknown;
316
+ }
317
+
318
+ export function shouldBlock(analysis) {
319
+ return analysis.risks.some(r => r.severity === 'block');
320
+ }
321
+
322
+ export function getBlockReason(analysis) {
323
+ const blocking = analysis.risks.find(r => r.severity === 'block');
324
+ return blocking ? blocking.detail : null;
325
+ }