@polymorphism-tech/morph-spec 3.1.0 → 3.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.
- package/CLAUDE.md +534 -0
- package/README.md +78 -4
- package/bin/morph-spec.js +50 -1
- package/bin/render-template.js +56 -10
- package/bin/task-manager.cjs +101 -7
- package/docs/cli-auto-detection.md +219 -0
- package/docs/llm-interaction-config.md +735 -0
- package/docs/troubleshooting.md +269 -0
- package/package.json +5 -1
- package/src/commands/advance-phase.js +93 -2
- package/src/commands/approve.js +221 -0
- package/src/commands/capture-pattern.js +121 -0
- package/src/commands/generate.js +128 -1
- package/src/commands/init.js +37 -0
- package/src/commands/migrate-state.js +158 -0
- package/src/commands/search-patterns.js +126 -0
- package/src/commands/spawn-team.js +172 -0
- package/src/commands/task.js +2 -2
- package/src/commands/update.js +36 -0
- package/src/commands/upgrade.js +346 -0
- package/src/generator/.gitkeep +0 -0
- package/src/generator/config-generator.js +206 -0
- package/src/generator/templates/config.json.template +40 -0
- package/src/generator/templates/project.md.template +67 -0
- package/src/lib/checkpoint-hooks.js +258 -0
- package/src/lib/metadata-extractor.js +380 -0
- package/src/lib/phase-state-machine.js +214 -0
- package/src/lib/state-manager.js +120 -0
- package/src/lib/template-data-sources.js +325 -0
- package/src/lib/validators/content-validator.js +351 -0
- package/src/llm/.gitkeep +0 -0
- package/src/llm/analyzer.js +215 -0
- package/src/llm/environment-detector.js +43 -0
- package/src/llm/few-shot-examples.js +216 -0
- package/src/llm/project-config-schema.json +188 -0
- package/src/llm/prompt-builder.js +96 -0
- package/src/llm/schema-validator.js +121 -0
- package/src/orchestrator.js +206 -0
- package/src/sanitizer/.gitkeep +0 -0
- package/src/sanitizer/context-sanitizer.js +221 -0
- package/src/sanitizer/patterns.js +163 -0
- package/src/scanner/.gitkeep +0 -0
- package/src/scanner/project-scanner.js +242 -0
- package/src/types/index.js +477 -0
- package/src/ui/.gitkeep +0 -0
- package/src/ui/diff-display.js +91 -0
- package/src/ui/interactive-wizard.js +96 -0
- package/src/ui/user-review.js +211 -0
- package/src/ui/wizard-questions.js +190 -0
- package/src/writer/.gitkeep +0 -0
- package/src/writer/file-writer.js +86 -0
|
@@ -0,0 +1,351 @@
|
|
|
1
|
+
import { readFileSync, existsSync } from 'fs';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Content Validator - Validates structure and content of output files
|
|
5
|
+
*
|
|
6
|
+
* Ensures spec.md, tasks.json, and other outputs have required sections
|
|
7
|
+
* and proper structure before allowing phase transitions.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Validate spec.md structure
|
|
12
|
+
* @param {string} specPath - Path to spec.md file
|
|
13
|
+
* @returns {Object} Validation result
|
|
14
|
+
*/
|
|
15
|
+
export function validateSpecContent(specPath) {
|
|
16
|
+
if (!existsSync(specPath)) {
|
|
17
|
+
return {
|
|
18
|
+
valid: false,
|
|
19
|
+
missing: ['File does not exist'],
|
|
20
|
+
errors: ['Spec file not found at: ' + specPath]
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const content = readFileSync(specPath, 'utf8');
|
|
25
|
+
|
|
26
|
+
// Required sections for a complete spec
|
|
27
|
+
const requiredSections = [
|
|
28
|
+
'## Overview',
|
|
29
|
+
'## Requirements',
|
|
30
|
+
'## Technical Design',
|
|
31
|
+
'## Data Model',
|
|
32
|
+
'## API Contracts'
|
|
33
|
+
];
|
|
34
|
+
|
|
35
|
+
const missing = requiredSections.filter(section => !content.includes(section));
|
|
36
|
+
|
|
37
|
+
// Additional quality checks
|
|
38
|
+
const errors = [];
|
|
39
|
+
const warnings = [];
|
|
40
|
+
|
|
41
|
+
// Check minimum content length
|
|
42
|
+
if (content.length < 500) {
|
|
43
|
+
warnings.push('Spec seems very short (< 500 characters). Consider adding more detail.');
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Check for placeholder text that should be replaced
|
|
47
|
+
const placeholders = ['TODO', 'TBD', 'PLACEHOLDER', '{{'];
|
|
48
|
+
placeholders.forEach(placeholder => {
|
|
49
|
+
if (content.includes(placeholder)) {
|
|
50
|
+
warnings.push(`Spec contains "${placeholder}" - ensure all placeholders are replaced`);
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
// Check for anti-patterns mentioned in the plan
|
|
55
|
+
if (content.toLowerCase().includes('manual azure portal')) {
|
|
56
|
+
errors.push(
|
|
57
|
+
'Spec mentions manual Azure portal creation. ' +
|
|
58
|
+
'All infrastructure must be defined in Bicep (Infrastructure as Code).'
|
|
59
|
+
);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
if (content.toLowerCase().includes('create resource manually')) {
|
|
63
|
+
errors.push(
|
|
64
|
+
'Spec mentions manual resource creation. ' +
|
|
65
|
+
'Use IaC (Bicep) for all infrastructure resources.'
|
|
66
|
+
);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return {
|
|
70
|
+
valid: missing.length === 0 && errors.length === 0,
|
|
71
|
+
missing,
|
|
72
|
+
errors,
|
|
73
|
+
warnings,
|
|
74
|
+
sections: {
|
|
75
|
+
required: requiredSections.length,
|
|
76
|
+
found: requiredSections.length - missing.length
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Validate tasks.json structure
|
|
83
|
+
* @param {string} tasksPath - Path to tasks.json file
|
|
84
|
+
* @returns {Object} Validation result
|
|
85
|
+
*/
|
|
86
|
+
export function validateTasksContent(tasksPath) {
|
|
87
|
+
if (!existsSync(tasksPath)) {
|
|
88
|
+
return {
|
|
89
|
+
valid: false,
|
|
90
|
+
errors: ['Tasks file not found at: ' + tasksPath]
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
let tasks;
|
|
95
|
+
try {
|
|
96
|
+
const content = readFileSync(tasksPath, 'utf8');
|
|
97
|
+
tasks = JSON.parse(content);
|
|
98
|
+
} catch (error) {
|
|
99
|
+
return {
|
|
100
|
+
valid: false,
|
|
101
|
+
errors: ['Invalid JSON in tasks file: ' + error.message]
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
const errors = [];
|
|
106
|
+
const warnings = [];
|
|
107
|
+
|
|
108
|
+
// Check required top-level fields
|
|
109
|
+
if (!tasks.feature) {
|
|
110
|
+
errors.push('Missing "feature" field in tasks.json');
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
if (!tasks.tasks || !Array.isArray(tasks.tasks)) {
|
|
114
|
+
errors.push('Missing or invalid "tasks" array in tasks.json');
|
|
115
|
+
return { valid: false, errors, warnings };
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
if (tasks.tasks.length === 0) {
|
|
119
|
+
errors.push('Tasks array is empty - no tasks defined');
|
|
120
|
+
return { valid: false, errors, warnings };
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Validate individual tasks
|
|
124
|
+
tasks.tasks.forEach((task, index) => {
|
|
125
|
+
const taskId = task.id || `Task ${index}`;
|
|
126
|
+
|
|
127
|
+
// Check required fields
|
|
128
|
+
if (!task.id) {
|
|
129
|
+
errors.push(`${taskId}: Missing "id" field`);
|
|
130
|
+
} else if (!/^(T\d{3}|CHECKPOINT_\d{3})$/.test(task.id)) {
|
|
131
|
+
warnings.push(`${taskId}: ID should follow format T### or CHECKPOINT_###`);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (!task.title) {
|
|
135
|
+
errors.push(`${taskId}: Missing "title" field`);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (!task.description) {
|
|
139
|
+
errors.push(`${taskId}: Missing "description" field`);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
if (!task.dependencies) {
|
|
143
|
+
errors.push(`${taskId}: Missing "dependencies" field (use empty array if no deps)`);
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
// For regular tasks (not checkpoints)
|
|
147
|
+
if (task.id && task.id.startsWith('T')) {
|
|
148
|
+
if (!task.category) {
|
|
149
|
+
warnings.push(`${taskId}: Missing "category" field`);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
if (!task.estimatedMinutes) {
|
|
153
|
+
warnings.push(`${taskId}: Missing "estimatedMinutes" field`);
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
if (!task.files || task.files.length === 0) {
|
|
157
|
+
warnings.push(`${taskId}: No files specified - consider adding affected files`);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// For checkpoints
|
|
162
|
+
if (task.id && task.id.startsWith('CHECKPOINT')) {
|
|
163
|
+
if (!task.afterTasks || task.afterTasks.length === 0) {
|
|
164
|
+
warnings.push(`${taskId}: Checkpoint should specify "afterTasks"`);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (!task.validations || task.validations.length === 0) {
|
|
168
|
+
warnings.push(`${taskId}: Checkpoint should specify "validations"`);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
// Check for orphaned tasks (missing dependencies)
|
|
174
|
+
const taskIds = new Set(tasks.tasks.map(t => t.id));
|
|
175
|
+
tasks.tasks.forEach(task => {
|
|
176
|
+
if (task.dependencies && Array.isArray(task.dependencies)) {
|
|
177
|
+
task.dependencies.forEach(depId => {
|
|
178
|
+
if (depId && !taskIds.has(depId)) {
|
|
179
|
+
errors.push(`${task.id}: References non-existent dependency "${depId}"`);
|
|
180
|
+
}
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
// Check for circular dependencies (simple check)
|
|
186
|
+
const hasCycle = (taskId, visited = new Set()) => {
|
|
187
|
+
if (visited.has(taskId)) return true;
|
|
188
|
+
visited.add(taskId);
|
|
189
|
+
|
|
190
|
+
const task = tasks.tasks.find(t => t.id === taskId);
|
|
191
|
+
if (!task || !task.dependencies) return false;
|
|
192
|
+
|
|
193
|
+
return task.dependencies.some(depId => hasCycle(depId, new Set(visited)));
|
|
194
|
+
};
|
|
195
|
+
|
|
196
|
+
tasks.tasks.forEach(task => {
|
|
197
|
+
if (task.id && hasCycle(task.id)) {
|
|
198
|
+
errors.push(`Circular dependency detected involving task ${task.id}`);
|
|
199
|
+
}
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
return {
|
|
203
|
+
valid: errors.length === 0,
|
|
204
|
+
errors,
|
|
205
|
+
warnings,
|
|
206
|
+
stats: {
|
|
207
|
+
totalTasks: tasks.tasks.length,
|
|
208
|
+
regularTasks: tasks.tasks.filter(t => t.id?.startsWith('T')).length,
|
|
209
|
+
checkpoints: tasks.tasks.filter(t => t.id?.startsWith('CHECKPOINT')).length
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
/**
|
|
215
|
+
* Validate contracts file structure (C# contracts)
|
|
216
|
+
* @param {string} contractsPath - Path to contracts file
|
|
217
|
+
* @returns {Object} Validation result
|
|
218
|
+
*/
|
|
219
|
+
export function validateContractsContent(contractsPath) {
|
|
220
|
+
if (!existsSync(contractsPath)) {
|
|
221
|
+
return {
|
|
222
|
+
valid: false,
|
|
223
|
+
errors: ['Contracts file not found at: ' + contractsPath]
|
|
224
|
+
};
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
const content = readFileSync(contractsPath, 'utf8');
|
|
228
|
+
const errors = [];
|
|
229
|
+
const warnings = [];
|
|
230
|
+
|
|
231
|
+
// Check for basic C# structure
|
|
232
|
+
if (!content.includes('namespace')) {
|
|
233
|
+
errors.push('Contracts file missing namespace declaration');
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
// Check for at least one interface or record
|
|
237
|
+
const hasInterface = /interface\s+\w+/.test(content);
|
|
238
|
+
const hasRecord = /record\s+\w+/.test(content);
|
|
239
|
+
const hasClass = /class\s+\w+/.test(content);
|
|
240
|
+
|
|
241
|
+
if (!hasInterface && !hasRecord && !hasClass) {
|
|
242
|
+
errors.push('Contracts file should define at least one interface, record, or class');
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// Check for common anti-patterns
|
|
246
|
+
if (content.includes('// TODO') || content.includes('// PLACEHOLDER')) {
|
|
247
|
+
warnings.push('Contracts contain TODO/PLACEHOLDER comments - ensure they are completed');
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
return {
|
|
251
|
+
valid: errors.length === 0,
|
|
252
|
+
errors,
|
|
253
|
+
warnings,
|
|
254
|
+
found: {
|
|
255
|
+
hasNamespace: content.includes('namespace'),
|
|
256
|
+
hasInterfaces: hasInterface,
|
|
257
|
+
hasRecords: hasRecord,
|
|
258
|
+
hasClasses: hasClass
|
|
259
|
+
}
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* Validate proposal.md structure
|
|
265
|
+
* @param {string} proposalPath - Path to proposal.md file
|
|
266
|
+
* @returns {Object} Validation result
|
|
267
|
+
*/
|
|
268
|
+
export function validateProposalContent(proposalPath) {
|
|
269
|
+
if (!existsSync(proposalPath)) {
|
|
270
|
+
return {
|
|
271
|
+
valid: false,
|
|
272
|
+
errors: ['Proposal file not found at: ' + proposalPath]
|
|
273
|
+
};
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
const content = readFileSync(proposalPath, 'utf8');
|
|
277
|
+
const errors = [];
|
|
278
|
+
const warnings = [];
|
|
279
|
+
|
|
280
|
+
// Required sections
|
|
281
|
+
const requiredSections = [
|
|
282
|
+
'## Problem Statement',
|
|
283
|
+
'## Proposed Solution'
|
|
284
|
+
];
|
|
285
|
+
|
|
286
|
+
const missing = requiredSections.filter(section => !content.includes(section));
|
|
287
|
+
missing.forEach(section => errors.push(`Missing required section: ${section}`));
|
|
288
|
+
|
|
289
|
+
// Check for minimum content
|
|
290
|
+
if (content.length < 300) {
|
|
291
|
+
warnings.push('Proposal seems very brief (< 300 characters)');
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
return {
|
|
295
|
+
valid: errors.length === 0,
|
|
296
|
+
errors,
|
|
297
|
+
warnings,
|
|
298
|
+
sections: {
|
|
299
|
+
required: requiredSections.length,
|
|
300
|
+
found: requiredSections.length - missing.length
|
|
301
|
+
}
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
/**
|
|
306
|
+
* Validate all outputs for a feature comprehensively
|
|
307
|
+
* @param {Object} featureState - Feature state object
|
|
308
|
+
* @param {string} targetPhase - Target phase for validation
|
|
309
|
+
* @returns {Object} Comprehensive validation result
|
|
310
|
+
*/
|
|
311
|
+
export function validateFeatureOutputs(featureState, targetPhase) {
|
|
312
|
+
const results = {
|
|
313
|
+
valid: true,
|
|
314
|
+
validations: []
|
|
315
|
+
};
|
|
316
|
+
|
|
317
|
+
// Validate based on target phase
|
|
318
|
+
if (targetPhase === 'clarify' || targetPhase === 'tasks' || targetPhase === 'implement') {
|
|
319
|
+
// Spec is required
|
|
320
|
+
if (featureState.outputs?.spec?.created) {
|
|
321
|
+
const specResult = validateSpecContent(featureState.outputs.spec.path);
|
|
322
|
+
results.validations.push({ type: 'spec', ...specResult });
|
|
323
|
+
if (!specResult.valid) results.valid = false;
|
|
324
|
+
} else {
|
|
325
|
+
results.valid = false;
|
|
326
|
+
results.validations.push({
|
|
327
|
+
type: 'spec',
|
|
328
|
+
valid: false,
|
|
329
|
+
errors: ['Spec file not created']
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
if (targetPhase === 'implement') {
|
|
335
|
+
// Tasks are required
|
|
336
|
+
if (featureState.outputs?.tasks?.created) {
|
|
337
|
+
const tasksResult = validateTasksContent(featureState.outputs.tasks.path);
|
|
338
|
+
results.validations.push({ type: 'tasks', ...tasksResult });
|
|
339
|
+
if (!tasksResult.valid) results.valid = false;
|
|
340
|
+
} else {
|
|
341
|
+
results.valid = false;
|
|
342
|
+
results.validations.push({
|
|
343
|
+
type: 'tasks',
|
|
344
|
+
valid: false,
|
|
345
|
+
errors: ['Tasks file not created']
|
|
346
|
+
});
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
return results;
|
|
351
|
+
}
|
package/src/llm/.gitkeep
ADDED
|
File without changes
|
|
@@ -0,0 +1,215 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview LLMAnalyzer - Analyzes project context using Claude Code's LLM
|
|
3
|
+
* @module morph-spec/llm/analyzer
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { spawn } from 'child_process';
|
|
7
|
+
import { detectClaudeCode, isInteractive } from './environment-detector.js';
|
|
8
|
+
import { buildPrompt } from './prompt-builder.js';
|
|
9
|
+
import Ajv from 'ajv';
|
|
10
|
+
import projectConfigSchema from './project-config-schema.json' with { type: 'json' };
|
|
11
|
+
|
|
12
|
+
/**
|
|
13
|
+
* @typedef {import('../types/index.js').SanitizedContext} SanitizedContext
|
|
14
|
+
* @typedef {import('../types/index.js').ProjectConfig} ProjectConfig
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* LLM Analysis Error
|
|
19
|
+
*/
|
|
20
|
+
export class LLMAnalysisError extends Error {
|
|
21
|
+
constructor(message, statusCode = 500, attempts = 0, originalError = null) {
|
|
22
|
+
super(message);
|
|
23
|
+
this.name = 'LLMAnalysisError';
|
|
24
|
+
this.statusCode = statusCode;
|
|
25
|
+
this.attempts = attempts;
|
|
26
|
+
this.originalError = originalError;
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Validation Error
|
|
32
|
+
*/
|
|
33
|
+
export class ValidationError extends Error {
|
|
34
|
+
constructor(message, field, value, schema = null) {
|
|
35
|
+
super(message);
|
|
36
|
+
this.name = 'ValidationError';
|
|
37
|
+
this.field = field;
|
|
38
|
+
this.value = value;
|
|
39
|
+
this.schema = schema;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* LLMAnalyzer - Analyzes project context using Claude Code's embedded LLM
|
|
45
|
+
* @class
|
|
46
|
+
*/
|
|
47
|
+
export class LLMAnalyzer {
|
|
48
|
+
constructor() {
|
|
49
|
+
this.ajv = new Ajv({ allErrors: true });
|
|
50
|
+
this.validateSchema = this.ajv.compile(projectConfigSchema);
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Analyze project context using Claude Code's LLM
|
|
55
|
+
* @param {SanitizedContext} context - Sanitized project context
|
|
56
|
+
* @param {Object} [options] - Analysis options
|
|
57
|
+
* @param {number} [options.timeout] - Timeout in ms (default: 60000 for user, no timeout for LLM)
|
|
58
|
+
* @returns {Promise<ProjectConfig>}
|
|
59
|
+
* @throws {LLMAnalysisError} If analysis fails
|
|
60
|
+
*/
|
|
61
|
+
async analyze(context, options = {}) {
|
|
62
|
+
// Check if Claude Code is available
|
|
63
|
+
if (!detectClaudeCode()) {
|
|
64
|
+
throw new LLMAnalysisError(
|
|
65
|
+
'Claude Code environment not detected. This feature requires Claude Code CLI.',
|
|
66
|
+
503,
|
|
67
|
+
0
|
|
68
|
+
);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// Build the prompt
|
|
72
|
+
const prompt = buildPrompt(context);
|
|
73
|
+
|
|
74
|
+
// Determine timeout: 60s for interactive (user present), no timeout for LLM-only
|
|
75
|
+
const timeout = options.timeout || (isInteractive() ? 60000 : 0);
|
|
76
|
+
|
|
77
|
+
// Invoke Claude Code LLM
|
|
78
|
+
const response = await this.invokeLLM(prompt, { timeout });
|
|
79
|
+
|
|
80
|
+
// Parse and validate JSON response
|
|
81
|
+
const projectConfig = this.parseJsonResponse(response);
|
|
82
|
+
|
|
83
|
+
return projectConfig;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Build structured prompt for LLM
|
|
88
|
+
* @param {SanitizedContext} context - Sanitized context
|
|
89
|
+
* @returns {string} Prompt text
|
|
90
|
+
*/
|
|
91
|
+
buildPrompt(context) {
|
|
92
|
+
return buildPrompt(context);
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Invoke Claude Code's LLM via spawning a subprocess
|
|
97
|
+
*
|
|
98
|
+
* IMPLEMENTATION NOTE: This is a placeholder implementation.
|
|
99
|
+
* The actual Claude Code invocation mechanism depends on how Claude Code exposes its LLM.
|
|
100
|
+
*
|
|
101
|
+
* Possible approaches:
|
|
102
|
+
* 1. Spawn `claude` CLI with a special flag for JSON mode
|
|
103
|
+
* 2. Use an internal API/IPC mechanism
|
|
104
|
+
* 3. Write prompt to a temp file and invoke Claude Code agent
|
|
105
|
+
*
|
|
106
|
+
* For now, we'll simulate the response for testing purposes.
|
|
107
|
+
*
|
|
108
|
+
* @param {string} prompt - Prompt text
|
|
109
|
+
* @param {Object} options - Invocation options
|
|
110
|
+
* @param {number} options.timeout - Timeout in ms (0 = no timeout)
|
|
111
|
+
* @returns {Promise<string>} JSON response from LLM
|
|
112
|
+
* @throws {LLMAnalysisError}
|
|
113
|
+
*/
|
|
114
|
+
async invokeLLM(prompt, options = {}) {
|
|
115
|
+
return new Promise((resolve, reject) => {
|
|
116
|
+
// TODO: Replace with actual Claude Code invocation
|
|
117
|
+
// For now, this is a placeholder that would need to be implemented
|
|
118
|
+
// based on how Claude Code exposes its LLM for programmatic access
|
|
119
|
+
|
|
120
|
+
// Example approach (pseudo-code):
|
|
121
|
+
// const claudeProcess = spawn('claude', ['--mode=json', '--prompt', prompt]);
|
|
122
|
+
|
|
123
|
+
// Placeholder: Simulate a timeout if needed
|
|
124
|
+
const { timeout } = options;
|
|
125
|
+
|
|
126
|
+
let output = '';
|
|
127
|
+
let errorOutput = '';
|
|
128
|
+
|
|
129
|
+
// Simulated response for testing (to be replaced with actual Claude Code invocation)
|
|
130
|
+
// In real implementation, this would spawn Claude Code process:
|
|
131
|
+
/*
|
|
132
|
+
const claudeProcess = spawn('claude-code', [
|
|
133
|
+
'--json',
|
|
134
|
+
'--prompt-from-stdin'
|
|
135
|
+
]);
|
|
136
|
+
|
|
137
|
+
claudeProcess.stdin.write(prompt);
|
|
138
|
+
claudeProcess.stdin.end();
|
|
139
|
+
|
|
140
|
+
claudeProcess.stdout.on('data', (data) => {
|
|
141
|
+
output += data.toString();
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
claudeProcess.stderr.on('data', (data) => {
|
|
145
|
+
errorOutput += data.toString();
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
claudeProcess.on('close', (code) => {
|
|
149
|
+
if (code !== 0) {
|
|
150
|
+
reject(new LLMAnalysisError(
|
|
151
|
+
`Claude Code exited with code ${code}: ${errorOutput}`,
|
|
152
|
+
500,
|
|
153
|
+
1
|
|
154
|
+
));
|
|
155
|
+
return;
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
resolve(output);
|
|
159
|
+
});
|
|
160
|
+
|
|
161
|
+
if (timeout > 0) {
|
|
162
|
+
setTimeout(() => {
|
|
163
|
+
claudeProcess.kill();
|
|
164
|
+
reject(new LLMAnalysisError('LLM analysis timeout', 504, 1));
|
|
165
|
+
}, timeout);
|
|
166
|
+
}
|
|
167
|
+
*/
|
|
168
|
+
|
|
169
|
+
// For now, reject with a "not implemented" error
|
|
170
|
+
// This will be replaced when we know the exact Claude Code invocation mechanism
|
|
171
|
+
reject(new LLMAnalysisError(
|
|
172
|
+
'LLM invocation not yet implemented. Awaiting Claude Code API documentation.',
|
|
173
|
+
501,
|
|
174
|
+
0
|
|
175
|
+
));
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Parse and validate LLM JSON response
|
|
181
|
+
* @param {string} response - Raw LLM response
|
|
182
|
+
* @returns {ProjectConfig}
|
|
183
|
+
* @throws {ValidationError} If response doesn't match schema
|
|
184
|
+
*/
|
|
185
|
+
parseJsonResponse(response) {
|
|
186
|
+
// Try to parse JSON
|
|
187
|
+
let parsed;
|
|
188
|
+
try {
|
|
189
|
+
parsed = JSON.parse(response);
|
|
190
|
+
} catch (error) {
|
|
191
|
+
throw new ValidationError(
|
|
192
|
+
'LLM response is not valid JSON',
|
|
193
|
+
'response',
|
|
194
|
+
response,
|
|
195
|
+
null
|
|
196
|
+
);
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
// Validate against schema
|
|
200
|
+
const valid = this.validateSchema(parsed);
|
|
201
|
+
if (!valid) {
|
|
202
|
+
const errors = this.validateSchema.errors || [];
|
|
203
|
+
const errorMessages = errors.map(err => `${err.instancePath} ${err.message}`).join(', ');
|
|
204
|
+
|
|
205
|
+
throw new ValidationError(
|
|
206
|
+
`Schema validation failed: ${errorMessages}`,
|
|
207
|
+
'schema',
|
|
208
|
+
parsed,
|
|
209
|
+
projectConfigSchema
|
|
210
|
+
);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
return parsed;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @fileoverview Environment Detector - Detects if running in Claude Code
|
|
3
|
+
* @module morph-spec/llm/environment-detector
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Detect if running inside Claude Code environment
|
|
8
|
+
* @returns {boolean} True if Claude Code is available
|
|
9
|
+
*/
|
|
10
|
+
export function detectClaudeCode() {
|
|
11
|
+
// Check for environment variables that Claude Code sets
|
|
12
|
+
const claudeEnvVars = [
|
|
13
|
+
'CLAUDE_CODE_SESSION',
|
|
14
|
+
'CLAUDE_API_KEY',
|
|
15
|
+
'ANTHROPIC_API_KEY'
|
|
16
|
+
];
|
|
17
|
+
|
|
18
|
+
// Check if any Claude-specific env var is set
|
|
19
|
+
const hasClaudeEnv = claudeEnvVars.some(varName => process.env[varName]);
|
|
20
|
+
|
|
21
|
+
// Check if we can detect Claude Code CLI
|
|
22
|
+
// (This is a heuristic - Claude Code sets these when running)
|
|
23
|
+
const hasClaudeCLI = process.env.TERM_PROGRAM === 'claude' ||
|
|
24
|
+
process.env.CLAUDE_CODE_VERSION;
|
|
25
|
+
|
|
26
|
+
return hasClaudeEnv || hasClaudeCLI;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Get Claude Code version if available
|
|
31
|
+
* @returns {string|null} Version string or null
|
|
32
|
+
*/
|
|
33
|
+
export function getClaudeCodeVersion() {
|
|
34
|
+
return process.env.CLAUDE_CODE_VERSION || null;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Check if running in interactive mode (user is present)
|
|
39
|
+
* @returns {boolean} True if interactive
|
|
40
|
+
*/
|
|
41
|
+
export function isInteractive() {
|
|
42
|
+
return process.stdout.isTTY && process.stdin.isTTY;
|
|
43
|
+
}
|