@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,172 @@
|
|
|
1
|
+
import { Command } from 'commander';
|
|
2
|
+
import chalk from 'chalk';
|
|
3
|
+
import { getFeature } from '../lib/state-manager.js';
|
|
4
|
+
import { orchestrateTeam } from '../lib/team-orchestrator.js';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Spawn Team Command - Generate agent team configurations
|
|
8
|
+
*
|
|
9
|
+
* Reads active agents from feature state and generates ready-to-use
|
|
10
|
+
* Task tool configurations for spawning agent teams.
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const program = new Command();
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Spawn team action function
|
|
17
|
+
*/
|
|
18
|
+
export async function spawnTeamCommand(featureName, options = {}) {
|
|
19
|
+
try {
|
|
20
|
+
// Get feature state
|
|
21
|
+
const feature = getFeature(featureName);
|
|
22
|
+
if (!feature) {
|
|
23
|
+
console.error(chalk.red(`\n❌ Feature not found: ${featureName}\n`));
|
|
24
|
+
process.exit(1);
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
if (!feature.activeAgents || feature.activeAgents.length === 0) {
|
|
28
|
+
console.error(chalk.yellow(`\n⚠️ No active agents found for feature: ${featureName}`));
|
|
29
|
+
console.log(chalk.gray('Run "morph-spec detect-agents" first to identify agents.\n'));
|
|
30
|
+
process.exit(1);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
// Determine if multi-domain
|
|
34
|
+
const domains = new Set();
|
|
35
|
+
feature.activeAgents.forEach(agentId => {
|
|
36
|
+
if (agentId.includes('dotnet') || agentId.includes('ef') || agentId.includes('api')) {
|
|
37
|
+
domains.add('backend');
|
|
38
|
+
}
|
|
39
|
+
if (agentId.includes('blazor') || agentId.includes('nextjs') || agentId.includes('ui')) {
|
|
40
|
+
domains.add('frontend');
|
|
41
|
+
}
|
|
42
|
+
if (agentId.includes('azure') || agentId.includes('bicep') || agentId.includes('container')) {
|
|
43
|
+
domains.add('infrastructure');
|
|
44
|
+
}
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
const multiDomain = domains.size > 1;
|
|
48
|
+
|
|
49
|
+
// Orchestrate team
|
|
50
|
+
const teamConfig = orchestrateTeam({
|
|
51
|
+
activeAgents: feature.activeAgents,
|
|
52
|
+
complexity: options.complexity,
|
|
53
|
+
estimatedFiles: parseInt(options.estimatedFiles, 10),
|
|
54
|
+
multiDomain
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
if (options.json) {
|
|
58
|
+
// Output as JSON for programmatic use
|
|
59
|
+
console.log(JSON.stringify(teamConfig, null, 2));
|
|
60
|
+
return;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// Pretty output for human reading
|
|
64
|
+
console.log(chalk.bold(`\n🤖 Agent Team Configuration for: ${featureName}`));
|
|
65
|
+
console.log('━'.repeat(60));
|
|
66
|
+
|
|
67
|
+
console.log(chalk.cyan(`\nShould Spawn Agents: ${teamConfig.useAgentTeams ? chalk.green('YES') : chalk.yellow('NO')}`));
|
|
68
|
+
console.log(chalk.gray(`Reason: ${teamConfig.useAgentTeams ? 'Meets spawn threshold' : 'Below threshold - implement directly'}`));
|
|
69
|
+
|
|
70
|
+
if (!teamConfig.useAgentTeams) {
|
|
71
|
+
console.log(chalk.yellow('\n⚠️ Team spawning not recommended for this feature.'));
|
|
72
|
+
console.log(chalk.gray('Implement directly in main session.\n'));
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
console.log(chalk.cyan(`\nDisplay Mode: ${teamConfig.displayMode}`));
|
|
77
|
+
console.log(chalk.cyan(`Total Teammates: ${teamConfig.teamHierarchy.totalTeammates}`));
|
|
78
|
+
|
|
79
|
+
// Show team hierarchy
|
|
80
|
+
console.log(chalk.bold('\nTeam Hierarchy:'));
|
|
81
|
+
console.log(chalk.green(` Team Lead: ${teamConfig.teamHierarchy.teamLead}`));
|
|
82
|
+
|
|
83
|
+
if (teamConfig.teamHierarchy.squads && teamConfig.teamHierarchy.squads.length > 0) {
|
|
84
|
+
console.log(chalk.cyan('\n Squad Leaders:'));
|
|
85
|
+
teamConfig.teamHierarchy.squads.forEach(squad => {
|
|
86
|
+
console.log(chalk.cyan(` • ${squad}`));
|
|
87
|
+
});
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (teamConfig.teamHierarchy.validators && teamConfig.teamHierarchy.validators.length > 0) {
|
|
91
|
+
console.log(chalk.gray('\n Validators (run in hooks):'));
|
|
92
|
+
teamConfig.teamHierarchy.validators.forEach(validator => {
|
|
93
|
+
console.log(chalk.gray(` • ${validator}`));
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
// Show Task tool usage
|
|
98
|
+
console.log(chalk.bold('\n📝 Task Tool Usage:'));
|
|
99
|
+
console.log(chalk.gray('\nUse the following pattern to spawn the team:\n'));
|
|
100
|
+
|
|
101
|
+
console.log(chalk.cyan('```typescript'));
|
|
102
|
+
console.log(chalk.white(`{
|
|
103
|
+
subagent_type: "general-purpose",
|
|
104
|
+
description: "Implement ${featureName}",
|
|
105
|
+
prompt: \`You are the Team Lead (${teamConfig.teamHierarchy.teamLead})...
|
|
106
|
+
|
|
107
|
+
Your Mission: Implement ${featureName}
|
|
108
|
+
|
|
109
|
+
Your Domain Leaders:
|
|
110
|
+
${teamConfig.teammates
|
|
111
|
+
.filter(t => t.tier === 2)
|
|
112
|
+
.map(t => `- ${t.id}: ${t.role}`)
|
|
113
|
+
.join('\n')}
|
|
114
|
+
|
|
115
|
+
Your Specialists:
|
|
116
|
+
${teamConfig.teammates
|
|
117
|
+
.filter(t => t.tier === 3)
|
|
118
|
+
.map(t => `- ${t.id}: ${t.role}`)
|
|
119
|
+
.join('\n')}
|
|
120
|
+
|
|
121
|
+
Standards to follow: [coding.md, architecture.md]
|
|
122
|
+
|
|
123
|
+
Deliverables: [list specific outputs]
|
|
124
|
+
\`
|
|
125
|
+
}`));
|
|
126
|
+
console.log(chalk.cyan('```\n'));
|
|
127
|
+
|
|
128
|
+
// Show individual teammates
|
|
129
|
+
console.log(chalk.bold('Individual Teammates:'));
|
|
130
|
+
teamConfig.teammates.forEach(teammate => {
|
|
131
|
+
console.log(chalk.cyan(`\n ${teammate.icon} ${teammate.id} (Tier ${teammate.tier})`));
|
|
132
|
+
console.log(chalk.gray(` Role: ${teammate.role}`));
|
|
133
|
+
|
|
134
|
+
if (teammate.spawnPrompt && teammate.spawnPrompt.length < 200) {
|
|
135
|
+
console.log(chalk.gray(` Prompt: ${teammate.spawnPrompt.substring(0, 100)}...`));
|
|
136
|
+
}
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
console.log('\n' + '━'.repeat(60));
|
|
140
|
+
console.log(chalk.green('\n✅ Team configuration generated successfully\n'));
|
|
141
|
+
|
|
142
|
+
} catch (error) {
|
|
143
|
+
console.error(chalk.red(`\n❌ Error generating team configuration: ${error.message}\n`));
|
|
144
|
+
if (error.stack) {
|
|
145
|
+
console.error(chalk.gray(error.stack));
|
|
146
|
+
}
|
|
147
|
+
process.exit(1);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Commander program command
|
|
153
|
+
*/
|
|
154
|
+
program
|
|
155
|
+
.name('spawn-team')
|
|
156
|
+
.description('Generate agent team configuration for Task tool spawning')
|
|
157
|
+
.argument('<feature>', 'Feature name')
|
|
158
|
+
.option('--json', 'Output as JSON')
|
|
159
|
+
.option('--complexity <level>', 'Override complexity level (low/medium/high/critical)', 'high')
|
|
160
|
+
.option('--estimated-files <count>', 'Override estimated file count', '20')
|
|
161
|
+
.action(spawnTeamCommand);
|
|
162
|
+
|
|
163
|
+
// Only parse if run directly (not imported as module)
|
|
164
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
165
|
+
if (process.argv.length > 2) {
|
|
166
|
+
program.parse(process.argv);
|
|
167
|
+
} else {
|
|
168
|
+
program.help();
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export default program;
|
package/src/commands/task.js
CHANGED
|
@@ -16,9 +16,9 @@ const taskManagerPath = join(__dirname, '..', '..', 'bin', 'task-manager.cjs');
|
|
|
16
16
|
*/
|
|
17
17
|
function executeTaskManager(args) {
|
|
18
18
|
return new Promise((resolve, reject) => {
|
|
19
|
+
// Note: shell: false (default) is intentional to preserve arguments with spaces
|
|
19
20
|
const child = spawn('node', [taskManagerPath, ...args], {
|
|
20
|
-
stdio: 'inherit'
|
|
21
|
-
shell: true
|
|
21
|
+
stdio: 'inherit'
|
|
22
22
|
});
|
|
23
23
|
|
|
24
24
|
child.on('close', (code) => {
|
package/src/commands/update.js
CHANGED
|
@@ -19,6 +19,8 @@ import {
|
|
|
19
19
|
getUpdateInstructions,
|
|
20
20
|
detectInstallMethod
|
|
21
21
|
} from '../utils/version-checker.js';
|
|
22
|
+
import { AutoContextOrchestrator } from '../orchestrator.js';
|
|
23
|
+
import { detectClaudeCode } from '../llm/environment-detector.js';
|
|
22
24
|
|
|
23
25
|
export async function updateCommand(options) {
|
|
24
26
|
const targetPath = process.cwd();
|
|
@@ -196,6 +198,40 @@ export async function updateCommand(options) {
|
|
|
196
198
|
logger.dim('Review the updated files for any new features.');
|
|
197
199
|
logger.blank();
|
|
198
200
|
|
|
201
|
+
// Re-analyze project context (unless --skip-detection)
|
|
202
|
+
if (!options.skipDetection && detectClaudeCode()) {
|
|
203
|
+
logger.header('Re-Analyzing Project Context');
|
|
204
|
+
logger.dim('Using Claude Code LLM to update project detection...');
|
|
205
|
+
logger.blank();
|
|
206
|
+
|
|
207
|
+
try {
|
|
208
|
+
const orchestrator = new AutoContextOrchestrator();
|
|
209
|
+
const result = await orchestrator.execute(targetPath, {
|
|
210
|
+
skipReview: false,
|
|
211
|
+
fallbackOnError: true,
|
|
212
|
+
wizardMode: options.wizard || false
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
if (result.success) {
|
|
216
|
+
logger.success('Project context re-analyzed and updated');
|
|
217
|
+
logger.dim(' ✓ .morph/project.md updated with latest detection');
|
|
218
|
+
logger.dim(' ✓ .morph/config/config.json updated with stack info');
|
|
219
|
+
} else {
|
|
220
|
+
logger.warn('Context re-analysis incomplete');
|
|
221
|
+
}
|
|
222
|
+
} catch (error) {
|
|
223
|
+
logger.warn(`Auto-detection failed: ${error.message}`);
|
|
224
|
+
logger.dim('Your existing config.json was preserved.');
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
logger.blank();
|
|
228
|
+
} else if (options.skipDetection) {
|
|
229
|
+
logger.dim('Skipped auto-detection (--skip-detection flag)');
|
|
230
|
+
} else {
|
|
231
|
+
logger.warn('⚠️ Claude Code not detected - skipping auto-detection');
|
|
232
|
+
logger.dim('Run with --wizard to configure manually.');
|
|
233
|
+
}
|
|
234
|
+
|
|
199
235
|
} catch (error) {
|
|
200
236
|
updateSpinner.fail('Update failed');
|
|
201
237
|
logger.error(error.message);
|
|
@@ -0,0 +1,346 @@
|
|
|
1
|
+
import { Command } from 'commander';
|
|
2
|
+
import chalk from 'chalk';
|
|
3
|
+
import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'fs';
|
|
4
|
+
import { join } from 'path';
|
|
5
|
+
import { execSync } from 'child_process';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Upgrade Command - Upgrade MORPH-SPEC project to v3.0
|
|
9
|
+
*
|
|
10
|
+
* Orchestrates:
|
|
11
|
+
* - State migration (add approval gates)
|
|
12
|
+
* - llm-interaction.json creation
|
|
13
|
+
* - patterns-learned.md creation
|
|
14
|
+
* - Documentation updates
|
|
15
|
+
*/
|
|
16
|
+
|
|
17
|
+
const program = new Command();
|
|
18
|
+
|
|
19
|
+
program
|
|
20
|
+
.name('upgrade')
|
|
21
|
+
.description('Upgrade MORPH-SPEC project to v3.0')
|
|
22
|
+
.option('--to-version <version>', 'Target version (default: 3.0.0)', '3.0.0')
|
|
23
|
+
.option('--preset <preset>', 'Configuration preset (strict | lightweight | no-guardrails)', 'strict')
|
|
24
|
+
.option('--dry-run', 'Preview changes without writing')
|
|
25
|
+
.option('--force', 'Skip confirmations')
|
|
26
|
+
.action(async (options) => {
|
|
27
|
+
try {
|
|
28
|
+
console.log(chalk.cyan(`
|
|
29
|
+
███████╗ MORPH-SPEC Upgrade
|
|
30
|
+
╚══════╝ Upgrading to v${options.toVersion}
|
|
31
|
+
`));
|
|
32
|
+
|
|
33
|
+
// Validate project
|
|
34
|
+
if (!existsSync(join(process.cwd(), '.morph'))) {
|
|
35
|
+
console.error(chalk.red('❌ No .morph directory found'));
|
|
36
|
+
console.log(chalk.yellow(' This does not appear to be a MORPH-SPEC project.\n'));
|
|
37
|
+
process.exit(1);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
// Read current state
|
|
41
|
+
const statePath = join(process.cwd(), '.morph/state.json');
|
|
42
|
+
if (!existsSync(statePath)) {
|
|
43
|
+
console.error(chalk.red('❌ No state.json found'));
|
|
44
|
+
console.log(chalk.yellow(' Run "morph-spec init" first.\n'));
|
|
45
|
+
process.exit(1);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
const state = JSON.parse(readFileSync(statePath, 'utf8'));
|
|
49
|
+
const currentVersion = state.version || '2.0.0';
|
|
50
|
+
|
|
51
|
+
console.log(chalk.gray(`Current version: ${currentVersion}`));
|
|
52
|
+
console.log(chalk.gray(`Target version: ${options.toVersion}\n`));
|
|
53
|
+
|
|
54
|
+
// Check if upgrade needed
|
|
55
|
+
if (currentVersion === options.toVersion) {
|
|
56
|
+
console.log(chalk.green(`✅ Already at v${options.toVersion}\n`));
|
|
57
|
+
process.exit(0);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Upgrade steps
|
|
61
|
+
const steps = [
|
|
62
|
+
{ name: 'Migrate state.json', fn: migrateState },
|
|
63
|
+
{ name: 'Create llm-interaction.json', fn: createLLMConfig },
|
|
64
|
+
{ name: 'Create patterns-learned.md', fn: createPatternsLibrary },
|
|
65
|
+
{ name: 'Update CLAUDE.md', fn: updateClaudeMd },
|
|
66
|
+
{ name: 'Create metadata directory', fn: createMetadataDir }
|
|
67
|
+
];
|
|
68
|
+
|
|
69
|
+
console.log(chalk.bold('📋 Upgrade Plan:\n'));
|
|
70
|
+
steps.forEach((step, i) => {
|
|
71
|
+
console.log(chalk.cyan(` ${i + 1}. ${step.name}`));
|
|
72
|
+
});
|
|
73
|
+
console.log('');
|
|
74
|
+
|
|
75
|
+
if (options.dryRun) {
|
|
76
|
+
console.log(chalk.yellow('🔍 DRY RUN - No changes will be written\n'));
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
if (!options.force && !options.dryRun) {
|
|
80
|
+
console.log(chalk.yellow('⚠️ This will modify your project'));
|
|
81
|
+
console.log(chalk.gray(' Backups will be created for modified files\n'));
|
|
82
|
+
console.log(chalk.gray(' Press Ctrl+C to cancel, or run with --force to skip confirmation.\n'));
|
|
83
|
+
// In real implementation, use inquirer for confirmation
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Execute steps
|
|
87
|
+
const results = [];
|
|
88
|
+
for (const [index, step] of steps.entries()) {
|
|
89
|
+
console.log(chalk.cyan(`\n[${index + 1}/${steps.length}] ${step.name}...`));
|
|
90
|
+
|
|
91
|
+
try {
|
|
92
|
+
const result = await step.fn(options);
|
|
93
|
+
results.push({ step: step.name, status: 'success', result });
|
|
94
|
+
console.log(chalk.green(`✅ ${step.name} complete`));
|
|
95
|
+
} catch (error) {
|
|
96
|
+
results.push({ step: step.name, status: 'failed', error: error.message });
|
|
97
|
+
console.log(chalk.red(`❌ ${step.name} failed: ${error.message}`));
|
|
98
|
+
|
|
99
|
+
if (!options.force) {
|
|
100
|
+
console.log(chalk.yellow('\nUpgrade halted due to error.'));
|
|
101
|
+
console.log(chalk.gray('Run with --force to continue despite errors.\n'));
|
|
102
|
+
process.exit(1);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Summary
|
|
108
|
+
console.log(chalk.bold('\n\n📊 Upgrade Summary:\n'));
|
|
109
|
+
|
|
110
|
+
const succeeded = results.filter(r => r.status === 'success').length;
|
|
111
|
+
const failed = results.filter(r => r.status === 'failed').length;
|
|
112
|
+
|
|
113
|
+
results.forEach(result => {
|
|
114
|
+
const icon = result.status === 'success' ? '✅' : '❌';
|
|
115
|
+
const color = result.status === 'success' ? chalk.green : chalk.red;
|
|
116
|
+
console.log(color(` ${icon} ${result.step}`));
|
|
117
|
+
|
|
118
|
+
if (result.status === 'failed') {
|
|
119
|
+
console.log(chalk.gray(` Error: ${result.error}`));
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
|
|
123
|
+
console.log('');
|
|
124
|
+
|
|
125
|
+
if (failed === 0) {
|
|
126
|
+
console.log(chalk.green.bold('🎉 Upgrade to v3.0 complete!\n'));
|
|
127
|
+
|
|
128
|
+
console.log(chalk.bold('Next steps:\n'));
|
|
129
|
+
console.log(chalk.gray('1. Review .morph/config/llm-interaction.json'));
|
|
130
|
+
console.log(chalk.gray('2. Read docs/llm-interaction-config.md for configuration options'));
|
|
131
|
+
console.log(chalk.gray('3. Try new features:'));
|
|
132
|
+
console.log(chalk.gray(' - npx morph-spec search-patterns "{keyword}"'));
|
|
133
|
+
console.log(chalk.gray(' - npx morph-spec approve {feature} design'));
|
|
134
|
+
console.log(chalk.gray(' - npx morph-spec spawn-team {feature}'));
|
|
135
|
+
console.log(chalk.gray('4. Continue development with v3.0 workflow\n'));
|
|
136
|
+
} else {
|
|
137
|
+
console.log(chalk.yellow(`⚠️ Upgrade completed with ${failed} error(s)\n`));
|
|
138
|
+
console.log(chalk.gray('Review errors above and fix manually.\n'));
|
|
139
|
+
process.exit(1);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
} catch (error) {
|
|
143
|
+
console.error(chalk.red(`\n❌ Upgrade failed: ${error.message}\n`));
|
|
144
|
+
process.exit(1);
|
|
145
|
+
}
|
|
146
|
+
});
|
|
147
|
+
|
|
148
|
+
/**
|
|
149
|
+
* Step 1: Migrate state.json
|
|
150
|
+
*/
|
|
151
|
+
async function migrateState(options) {
|
|
152
|
+
const migrateCommand = options.dryRun
|
|
153
|
+
? 'npx morph-spec migrate-state --dry-run --force'
|
|
154
|
+
: 'npx morph-spec migrate-state --force';
|
|
155
|
+
|
|
156
|
+
try {
|
|
157
|
+
execSync(migrateCommand, { cwd: process.cwd(), stdio: 'pipe' });
|
|
158
|
+
return { featuresUpdated: 'N/A' };
|
|
159
|
+
} catch (error) {
|
|
160
|
+
// If migrate-state doesn't exist yet, do inline migration
|
|
161
|
+
const statePath = join(process.cwd(), '.morph/state.json');
|
|
162
|
+
const state = JSON.parse(readFileSync(statePath, 'utf8'));
|
|
163
|
+
|
|
164
|
+
let featuresUpdated = 0;
|
|
165
|
+
|
|
166
|
+
Object.values(state.features || {}).forEach(feature => {
|
|
167
|
+
if (!feature.approvalGates) {
|
|
168
|
+
feature.approvalGates = {
|
|
169
|
+
proposal: { approved: true, timestamp: new Date().toISOString(), approvedBy: 'auto-migration' },
|
|
170
|
+
uiux: { approved: false, timestamp: null, approvedBy: null },
|
|
171
|
+
design: { approved: true, timestamp: new Date().toISOString(), approvedBy: 'auto-migration' },
|
|
172
|
+
tasks: { approved: true, timestamp: new Date().toISOString(), approvedBy: 'auto-migration' }
|
|
173
|
+
};
|
|
174
|
+
featuresUpdated++;
|
|
175
|
+
}
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
state.version = '3.0.0';
|
|
179
|
+
|
|
180
|
+
if (!options.dryRun) {
|
|
181
|
+
writeFileSync(`${statePath}.backup`, readFileSync(statePath));
|
|
182
|
+
writeFileSync(statePath, JSON.stringify(state, null, 2));
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
return { featuresUpdated };
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
/**
|
|
190
|
+
* Step 2: Create llm-interaction.json
|
|
191
|
+
*/
|
|
192
|
+
async function createLLMConfig(options) {
|
|
193
|
+
const configPath = join(process.cwd(), '.morph/config/llm-interaction.json');
|
|
194
|
+
|
|
195
|
+
// Choose preset
|
|
196
|
+
const presets = {
|
|
197
|
+
strict: {
|
|
198
|
+
approvalGates: { enabled: true, required: ['design', 'tasks'], allowSkip: false },
|
|
199
|
+
checkpoints: { frequency: 3, autoValidate: true, onFailure: { blockProgress: true } },
|
|
200
|
+
phaseValidation: { strictTransitions: true, requireContentValidation: true },
|
|
201
|
+
agentSpawning: { enabled: true, teamMode: 'auto' },
|
|
202
|
+
metadata: { autoGenerate: true },
|
|
203
|
+
patterns: { searchBeforeStart: true, captureOnComplete: true },
|
|
204
|
+
interactiveDecisions: { enabled: true }
|
|
205
|
+
},
|
|
206
|
+
lightweight: {
|
|
207
|
+
approvalGates: { enabled: true, required: ['design'], allowSkip: true },
|
|
208
|
+
checkpoints: { frequency: 5, autoValidate: true, onFailure: { blockProgress: false } },
|
|
209
|
+
phaseValidation: { strictTransitions: true, allowPhaseSkip: true },
|
|
210
|
+
agentSpawning: { enabled: false },
|
|
211
|
+
metadata: { autoGenerate: false },
|
|
212
|
+
patterns: { searchBeforeStart: true, captureOnComplete: false },
|
|
213
|
+
interactiveDecisions: { enabled: true }
|
|
214
|
+
},
|
|
215
|
+
'no-guardrails': {
|
|
216
|
+
approvalGates: { enabled: false },
|
|
217
|
+
checkpoints: { autoValidate: false },
|
|
218
|
+
phaseValidation: { strictTransitions: false, allowPhaseSkip: true },
|
|
219
|
+
agentSpawning: { enabled: false },
|
|
220
|
+
metadata: { autoGenerate: false },
|
|
221
|
+
patterns: { searchBeforeStart: false, captureOnComplete: false },
|
|
222
|
+
interactiveDecisions: { enabled: false }
|
|
223
|
+
}
|
|
224
|
+
};
|
|
225
|
+
|
|
226
|
+
const preset = presets[options.preset] || presets.strict;
|
|
227
|
+
|
|
228
|
+
const config = {
|
|
229
|
+
version: '1.0.0',
|
|
230
|
+
enabled: true,
|
|
231
|
+
...preset,
|
|
232
|
+
checkpoints: {
|
|
233
|
+
...preset.checkpoints,
|
|
234
|
+
validators: { enabled: ['architecture', 'packages', 'design-system', 'security'] },
|
|
235
|
+
hooks: { runTests: false, runLinters: true, buildCheck: false },
|
|
236
|
+
onFailure: { ...preset.checkpoints.onFailure, maxRetries: 3 }
|
|
237
|
+
},
|
|
238
|
+
agentSpawning: {
|
|
239
|
+
...preset.agentSpawning,
|
|
240
|
+
autoDetect: true,
|
|
241
|
+
spawnThreshold: { complexity: 'high', fileCount: 15, multiDomain: true, agentCount: 5 }
|
|
242
|
+
},
|
|
243
|
+
metadata: { ...preset.metadata, updateFrequency: 'on_task_done' },
|
|
244
|
+
patterns: { ...preset.patterns, location: '.morph/memory/patterns-learned.md' },
|
|
245
|
+
templates: { mcpDataInjection: true, dataSources: ['projectStructure', 'complianceStatus', 'recentActivity'] },
|
|
246
|
+
interactiveDecisions: {
|
|
247
|
+
...preset.interactiveDecisions,
|
|
248
|
+
usePromptTemplates: true,
|
|
249
|
+
requireUserInput: ['architecture-choice', 'scope-change', 'validator-override']
|
|
250
|
+
}
|
|
251
|
+
};
|
|
252
|
+
|
|
253
|
+
if (!options.dryRun) {
|
|
254
|
+
mkdirSync(join(process.cwd(), '.morph/config'), { recursive: true });
|
|
255
|
+
writeFileSync(configPath, JSON.stringify(config, null, 2));
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
return { preset: options.preset, path: configPath };
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
/**
|
|
262
|
+
* Step 3: Create patterns-learned.md
|
|
263
|
+
*/
|
|
264
|
+
async function createPatternsLibrary(options) {
|
|
265
|
+
const patternsPath = join(process.cwd(), '.morph/memory/patterns-learned.md');
|
|
266
|
+
|
|
267
|
+
if (existsSync(patternsPath)) {
|
|
268
|
+
return { status: 'already exists' };
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
const initialContent = `# Patterns Learned
|
|
272
|
+
|
|
273
|
+
This file tracks reusable patterns, lessons learned, and anti-patterns discovered during development.
|
|
274
|
+
|
|
275
|
+
**Usage:**
|
|
276
|
+
- Search: \`npx morph-spec search-patterns "{keyword}"\`
|
|
277
|
+
- Capture: \`npx morph-spec capture-pattern {feature} {category} "Description"\`
|
|
278
|
+
|
|
279
|
+
---
|
|
280
|
+
|
|
281
|
+
## Pre-Seeded Framework Patterns
|
|
282
|
+
|
|
283
|
+
See framework/.morph/memory/patterns-learned.md for standard patterns.
|
|
284
|
+
|
|
285
|
+
---
|
|
286
|
+
`;
|
|
287
|
+
|
|
288
|
+
if (!options.dryRun) {
|
|
289
|
+
mkdirSync(join(process.cwd(), '.morph/memory'), { recursive: true });
|
|
290
|
+
writeFileSync(patternsPath, initialContent);
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
return { path: patternsPath };
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
/**
|
|
297
|
+
* Step 4: Update CLAUDE.md
|
|
298
|
+
*/
|
|
299
|
+
async function updateClaudeMd(options) {
|
|
300
|
+
const claudePath = join(process.cwd(), 'CLAUDE.md');
|
|
301
|
+
|
|
302
|
+
if (!existsSync(claudePath)) {
|
|
303
|
+
return { status: 'not found, skipped' };
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
const content = readFileSync(claudePath, 'utf8');
|
|
307
|
+
|
|
308
|
+
// Check if already updated
|
|
309
|
+
if (content.includes('## APPROVAL GATES')) {
|
|
310
|
+
return { status: 'already up-to-date' };
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
// Add note about v3.0 features
|
|
314
|
+
const note = `\n---\n\n> **MORPH-SPEC v3.0 Features:** This project now supports approval gates, checkpoint automation, pattern learning, and metadata generation. See \`.morph/config/llm-interaction.json\` for configuration.\n\n---\n`;
|
|
315
|
+
|
|
316
|
+
if (!options.dryRun) {
|
|
317
|
+
writeFileSync(`${claudePath}.backup`, content);
|
|
318
|
+
writeFileSync(claudePath, content + note);
|
|
319
|
+
}
|
|
320
|
+
|
|
321
|
+
return { status: 'updated with v3.0 note' };
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
/**
|
|
325
|
+
* Step 5: Create metadata output directory
|
|
326
|
+
*/
|
|
327
|
+
async function createMetadataDir(options) {
|
|
328
|
+
const metadataDir = join(process.cwd(), '.morph/project/outputs');
|
|
329
|
+
|
|
330
|
+
if (!options.dryRun) {
|
|
331
|
+
mkdirSync(metadataDir, { recursive: true });
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
return { path: metadataDir };
|
|
335
|
+
}
|
|
336
|
+
|
|
337
|
+
// Only parse if run directly (not imported as module)
|
|
338
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
339
|
+
if (process.argv.length > 2) {
|
|
340
|
+
program.parse(process.argv);
|
|
341
|
+
} else {
|
|
342
|
+
program.help();
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
export default program;
|
|
File without changes
|