@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.
Files changed (51) hide show
  1. package/CLAUDE.md +534 -0
  2. package/README.md +78 -4
  3. package/bin/morph-spec.js +50 -1
  4. package/bin/render-template.js +56 -10
  5. package/bin/task-manager.cjs +101 -7
  6. package/docs/cli-auto-detection.md +219 -0
  7. package/docs/llm-interaction-config.md +735 -0
  8. package/docs/troubleshooting.md +269 -0
  9. package/package.json +5 -1
  10. package/src/commands/advance-phase.js +93 -2
  11. package/src/commands/approve.js +221 -0
  12. package/src/commands/capture-pattern.js +121 -0
  13. package/src/commands/generate.js +128 -1
  14. package/src/commands/init.js +37 -0
  15. package/src/commands/migrate-state.js +158 -0
  16. package/src/commands/search-patterns.js +126 -0
  17. package/src/commands/spawn-team.js +172 -0
  18. package/src/commands/task.js +2 -2
  19. package/src/commands/update.js +36 -0
  20. package/src/commands/upgrade.js +346 -0
  21. package/src/generator/.gitkeep +0 -0
  22. package/src/generator/config-generator.js +206 -0
  23. package/src/generator/templates/config.json.template +40 -0
  24. package/src/generator/templates/project.md.template +67 -0
  25. package/src/lib/checkpoint-hooks.js +258 -0
  26. package/src/lib/metadata-extractor.js +380 -0
  27. package/src/lib/phase-state-machine.js +214 -0
  28. package/src/lib/state-manager.js +120 -0
  29. package/src/lib/template-data-sources.js +325 -0
  30. package/src/lib/validators/content-validator.js +351 -0
  31. package/src/llm/.gitkeep +0 -0
  32. package/src/llm/analyzer.js +215 -0
  33. package/src/llm/environment-detector.js +43 -0
  34. package/src/llm/few-shot-examples.js +216 -0
  35. package/src/llm/project-config-schema.json +188 -0
  36. package/src/llm/prompt-builder.js +96 -0
  37. package/src/llm/schema-validator.js +121 -0
  38. package/src/orchestrator.js +206 -0
  39. package/src/sanitizer/.gitkeep +0 -0
  40. package/src/sanitizer/context-sanitizer.js +221 -0
  41. package/src/sanitizer/patterns.js +163 -0
  42. package/src/scanner/.gitkeep +0 -0
  43. package/src/scanner/project-scanner.js +242 -0
  44. package/src/types/index.js +477 -0
  45. package/src/ui/.gitkeep +0 -0
  46. package/src/ui/diff-display.js +91 -0
  47. package/src/ui/interactive-wizard.js +96 -0
  48. package/src/ui/user-review.js +211 -0
  49. package/src/ui/wizard-questions.js +190 -0
  50. package/src/writer/.gitkeep +0 -0
  51. 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
+ }
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
+ }