@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,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;
@@ -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) => {
@@ -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