@polymorphism-tech/morph-spec 4.2.0 → 4.3.1

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 (140) hide show
  1. package/CLAUDE.md +108 -946
  2. package/bin/morph-spec.js +284 -9
  3. package/bin/task-manager.cjs +102 -14
  4. package/bin/validate.js +4 -4
  5. package/docs/{v3.0 → next-generation}/AGENTS.md +1 -1
  6. package/docs/next-generation/CONTEXT-OPTIMIZATION.md +267 -0
  7. package/docs/next-generation/EXECUTION-FLOW.md +274 -0
  8. package/docs/next-generation/META-PROMPTS.md +235 -0
  9. package/docs/next-generation/MIGRATION-GUIDE.md +253 -0
  10. package/docs/next-generation/THREAD-MANAGEMENT.md +240 -0
  11. package/package.json +5 -5
  12. package/src/commands/agents/agents-fuse.js +97 -0
  13. package/src/commands/agents/micro-agent.js +112 -0
  14. package/src/commands/agents/spawn-team.js +69 -4
  15. package/src/commands/agents/squad-template.js +146 -0
  16. package/src/commands/analytics/analytics.js +176 -0
  17. package/src/commands/context/context-prime.js +63 -0
  18. package/src/commands/context/core-four.js +54 -0
  19. package/src/commands/mcp/mcp.js +102 -0
  20. package/src/commands/project/detect-agents.js +32 -2
  21. package/src/commands/project/detect.js +11 -1
  22. package/src/commands/project/doctor.js +573 -356
  23. package/src/commands/project/init.js +9 -2
  24. package/src/commands/project/update.js +13 -3
  25. package/src/commands/state/advance-phase.js +448 -416
  26. package/src/commands/state/state.js +14 -12
  27. package/src/commands/tasks/task.js +1 -1
  28. package/src/commands/templates/template-render.js +80 -1
  29. package/src/commands/threads/thread-template.js +103 -0
  30. package/src/commands/threads/threads.js +261 -0
  31. package/src/commands/trust/trust.js +205 -0
  32. package/src/{orchestrator.js → core/orchestrator.js} +8 -8
  33. package/src/core/state/state-manager.js +37 -17
  34. package/src/core/workflows/workflow-detector.js +114 -3
  35. package/src/lib/agents/micro-agent-factory.js +161 -0
  36. package/src/lib/analytics/analytics-engine.js +345 -0
  37. package/src/lib/checkpoints/checkpoint-hooks.js +298 -258
  38. package/src/lib/context/context-bundler.js +240 -0
  39. package/src/lib/context/context-optimizer.js +212 -0
  40. package/src/lib/context/context-tracker.js +273 -0
  41. package/src/lib/context/core-four-tracker.js +201 -0
  42. package/src/lib/context/mcp-optimizer.js +200 -0
  43. package/src/lib/detectors/index.js +1 -1
  44. package/src/lib/detectors/standards-generator.js +77 -17
  45. package/src/lib/detectors/structure-detector.js +67 -39
  46. package/src/lib/execution/fusion-executor.js +304 -0
  47. package/src/lib/execution/parallel-executor.js +270 -0
  48. package/src/lib/generators/context-generator.js +3 -3
  49. package/src/lib/generators/recap-generator.js +32 -12
  50. package/src/lib/hooks/hook-executor.js +169 -0
  51. package/src/lib/hooks/stop-hook-executor.js +286 -0
  52. package/src/lib/hops/hop-composer.js +221 -0
  53. package/src/lib/threads/thread-coordinator.js +238 -0
  54. package/src/lib/threads/thread-manager.js +317 -0
  55. package/src/lib/tracking/artifact-trail.js +202 -0
  56. package/src/lib/trust/trust-manager.js +269 -0
  57. package/src/lib/validators/design-system/design-system-validator.js +2 -2
  58. package/src/lib/validators/validation-runner.js +14 -30
  59. package/src/utils/hooks-installer.js +69 -0
  60. package/stacks/blazor-azure/.morph/config/agents.json +72 -3
  61. package/stacks/nextjs-supabase/.morph/config/agents.json +3 -3
  62. package/docs/llm-interaction-config.md +0 -735
  63. package/docs/v3.0/EXECUTION-FLOW.md +0 -1304
  64. package/src/commands/utils/migrate-state.js +0 -158
  65. package/src/commands/utils/upgrade.js +0 -346
  66. package/src/lib/validators/architecture-validator.js +0 -60
  67. package/src/lib/validators/content-validator.js +0 -164
  68. package/src/lib/validators/package-validator.js +0 -61
  69. package/src/lib/validators/ui-contrast-validator.js +0 -44
  70. package/stacks/blazor-azure/.claude/commands/morph-apply.md +0 -221
  71. package/stacks/blazor-azure/.claude/commands/morph-archive.md +0 -79
  72. package/stacks/blazor-azure/.claude/commands/morph-deploy.md +0 -529
  73. package/stacks/blazor-azure/.claude/commands/morph-infra.md +0 -209
  74. package/stacks/blazor-azure/.claude/commands/morph-preflight.md +0 -227
  75. package/stacks/blazor-azure/.claude/commands/morph-proposal.md +0 -122
  76. package/stacks/blazor-azure/.claude/commands/morph-status.md +0 -86
  77. package/stacks/blazor-azure/.claude/commands/morph-troubleshoot.md +0 -122
  78. package/stacks/blazor-azure/.claude/skills/level-0-meta/README.md +0 -7
  79. package/stacks/blazor-azure/.claude/skills/level-0-meta/code-review.md +0 -226
  80. package/stacks/blazor-azure/.claude/skills/level-0-meta/morph-checklist.md +0 -117
  81. package/stacks/blazor-azure/.claude/skills/level-0-meta/simulation-checklist.md +0 -77
  82. package/stacks/blazor-azure/.claude/skills/level-1-workflows/README.md +0 -7
  83. package/stacks/blazor-azure/.claude/skills/level-1-workflows/morph-replicate.md +0 -213
  84. package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-clarify.md +0 -131
  85. package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-design.md +0 -213
  86. package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-setup.md +0 -106
  87. package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-tasks.md +0 -164
  88. package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-uiux.md +0 -169
  89. package/stacks/blazor-azure/.claude/skills/level-2-domains/README.md +0 -14
  90. package/stacks/blazor-azure/.claude/skills/level-2-domains/ai-agents/ai-system-architect.md +0 -192
  91. package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/po-pm-advisor.md +0 -197
  92. package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/prompt-engineer.md +0 -189
  93. package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/seo-growth-hacker.md +0 -320
  94. package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/standards-architect.md +0 -156
  95. package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/api-designer.md +0 -59
  96. package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/dotnet-senior.md +0 -77
  97. package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ef-modeler.md +0 -58
  98. package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/hangfire-orchestrator.md +0 -126
  99. package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ms-agent-expert.md +0 -45
  100. package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/blazor-builder.md +0 -210
  101. package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/nextjs-expert.md +0 -154
  102. package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/ui-ux-designer.md +0 -191
  103. package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/azure-architect.md +0 -142
  104. package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/azure-deploy-specialist.md +0 -699
  105. package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/bicep-architect.md +0 -126
  106. package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/container-specialist.md +0 -131
  107. package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/devops-engineer.md +0 -119
  108. package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/asaas-financial.md +0 -130
  109. package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/azure-identity.md +0 -142
  110. package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/clerk-auth.md +0 -108
  111. package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/hangfire-orchestrator.md +0 -64
  112. package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/resend-email.md +0 -119
  113. package/stacks/blazor-azure/.claude/skills/level-2-domains/quality/code-analyzer.md +0 -235
  114. package/stacks/blazor-azure/.claude/skills/level-2-domains/quality/testing-specialist.md +0 -126
  115. package/stacks/blazor-azure/.claude/skills/level-3-technologies/README.md +0 -7
  116. package/stacks/blazor-azure/.claude/skills/level-4-patterns/README.md +0 -7
  117. package/stacks/blazor-azure/.morph/archive/.gitkeep +0 -25
  118. package/stacks/blazor-azure/.morph/features/.gitkeep +0 -25
  119. package/stacks/blazor-azure/.morph/schemas/agent.schema.json +0 -296
  120. package/stacks/blazor-azure/.morph/schemas/tasks.schema.json +0 -220
  121. package/stacks/blazor-azure/.morph/specs/.gitkeep +0 -20
  122. package/stacks/blazor-azure/.morph/test-infra/example.bicep +0 -59
  123. package/stacks/nextjs-supabase/.claude/commands/morph-apply.md +0 -221
  124. package/stacks/nextjs-supabase/.claude/commands/morph-archive.md +0 -79
  125. package/stacks/nextjs-supabase/.claude/commands/morph-deploy.md +0 -529
  126. package/stacks/nextjs-supabase/.claude/commands/morph-infra.md +0 -209
  127. package/stacks/nextjs-supabase/.claude/commands/morph-preflight.md +0 -227
  128. package/stacks/nextjs-supabase/.claude/commands/morph-proposal.md +0 -122
  129. package/stacks/nextjs-supabase/.claude/commands/morph-status.md +0 -86
  130. package/stacks/nextjs-supabase/.claude/commands/morph-troubleshoot.md +0 -122
  131. package/stacks/nextjs-supabase/.claude/settings.local.json +0 -6
  132. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/backend/dotnet-supabase.md +0 -244
  133. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/frontend/nextjs-supabase.md +0 -335
  134. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/infrastructure/easypanel-deployer.md +0 -189
  135. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/integrations/supabase-expert.md +0 -50
  136. /package/docs/{v3.0 → next-generation}/ANALYSIS.md +0 -0
  137. /package/docs/{v3.0 → next-generation}/ARCHITECTURE.md +0 -0
  138. /package/docs/{v3.0 → next-generation}/FEATURES.md +0 -0
  139. /package/docs/{v3.0 → next-generation}/README.md +0 -0
  140. /package/docs/{v3.0 → next-generation}/ROADMAP.md +0 -0
@@ -1,158 +0,0 @@
1
- import { Command } from 'commander';
2
- import chalk from 'chalk';
3
- import { readFileSync, writeFileSync, existsSync } from 'fs';
4
- import { join } from 'path';
5
-
6
- /**
7
- * Migrate State Command - Upgrade state.json to v3.0 schema
8
- *
9
- * Adds approval gates to existing features
10
- */
11
-
12
- const program = new Command();
13
-
14
- program
15
- .name('migrate-state')
16
- .description('Migrate state.json to v3.0 schema (add approval gates)')
17
- .option('--dry-run', 'Preview changes without writing')
18
- .option('--force', 'Skip confirmation')
19
- .action(async (options) => {
20
- try {
21
- const statePath = join(process.cwd(), '.morph/state.json');
22
-
23
- if (!existsSync(statePath)) {
24
- console.error(chalk.red(`\n❌ No state.json found at ${statePath}`));
25
- console.log(chalk.yellow('This project may not be initialized with MORPH-SPEC.\n'));
26
- process.exit(1);
27
- }
28
-
29
- // Read current state
30
- const state = JSON.parse(readFileSync(statePath, 'utf8'));
31
-
32
- console.log(chalk.cyan('\n🔄 Migrating state.json to v3.0 schema...\n'));
33
-
34
- // Check if already migrated
35
- if (state.version === '3.0.0') {
36
- console.log(chalk.green('✅ State already at v3.0.0 - no migration needed\n'));
37
- process.exit(0);
38
- }
39
-
40
- // Track changes
41
- let featuresUpdated = 0;
42
- const changes = [];
43
-
44
- // Migrate each feature
45
- Object.entries(state.features || {}).forEach(([featureName, feature]) => {
46
- // Skip if already has approvalGates
47
- if (feature.approvalGates) {
48
- return;
49
- }
50
-
51
- // Determine which gates to auto-approve based on current phase
52
- const currentPhase = feature.phase || 'proposal';
53
- const approvalGates = {
54
- proposal: { approved: false, timestamp: null, approvedBy: null },
55
- uiux: { approved: false, timestamp: null, approvedBy: null },
56
- design: { approved: false, timestamp: null, approvedBy: null },
57
- tasks: { approved: false, timestamp: null, approvedBy: null }
58
- };
59
-
60
- // Auto-approve past gates
61
- const phaseOrder = ['proposal', 'setup', 'uiux', 'design', 'clarify', 'tasks', 'implement', 'sync', 'archived'];
62
- const currentPhaseIndex = phaseOrder.indexOf(currentPhase);
63
-
64
- if (currentPhaseIndex >= phaseOrder.indexOf('proposal')) {
65
- approvalGates.proposal.approved = true;
66
- approvalGates.proposal.timestamp = feature.createdAt || new Date().toISOString();
67
- approvalGates.proposal.approvedBy = 'auto-migration';
68
- }
69
-
70
- if (currentPhaseIndex >= phaseOrder.indexOf('design')) {
71
- approvalGates.design.approved = true;
72
- approvalGates.design.timestamp = feature.updatedAt || new Date().toISOString();
73
- approvalGates.design.approvedBy = 'auto-migration';
74
- }
75
-
76
- if (currentPhaseIndex >= phaseOrder.indexOf('tasks')) {
77
- approvalGates.tasks.approved = true;
78
- approvalGates.tasks.timestamp = feature.updatedAt || new Date().toISOString();
79
- approvalGates.tasks.approvedBy = 'auto-migration';
80
- }
81
-
82
- // Add approval gates to feature
83
- feature.approvalGates = approvalGates;
84
-
85
- featuresUpdated++;
86
- changes.push({
87
- feature: featureName,
88
- phase: currentPhase,
89
- approvedGates: Object.entries(approvalGates)
90
- .filter(([_, gate]) => gate.approved)
91
- .map(([name]) => name)
92
- });
93
- });
94
-
95
- // Update state version
96
- state.version = '3.0.0';
97
-
98
- // Show changes
99
- console.log(chalk.bold('📊 Migration Summary:\n'));
100
- console.log(chalk.cyan(` Features to update: ${featuresUpdated}`));
101
-
102
- if (changes.length > 0) {
103
- console.log(chalk.gray('\n Changes:'));
104
- changes.forEach(change => {
105
- console.log(chalk.white(` - ${change.feature} (${change.phase})`));
106
- console.log(chalk.gray(` Auto-approved gates: ${change.approvedGates.join(', ') || 'none'}`));
107
- });
108
- }
109
-
110
- console.log('');
111
-
112
- // Dry run
113
- if (options.dryRun) {
114
- console.log(chalk.yellow('🔍 DRY RUN - No changes written\n'));
115
- console.log(chalk.gray('Run without --dry-run to apply migration.\n'));
116
- return;
117
- }
118
-
119
- // Confirmation
120
- if (!options.force) {
121
- console.log(chalk.yellow('⚠️ This will modify .morph/state.json'));
122
- console.log(chalk.gray(' A backup will be created at .morph/state.json.backup\n'));
123
- console.log(chalk.gray(' Run with --force to skip this confirmation, or Ctrl+C to cancel.\n'));
124
-
125
- // Wait for user confirmation (in CLI context, assume proceed for now)
126
- // In real implementation, use inquirer for interactive confirmation
127
- }
128
-
129
- // Backup original
130
- const backupPath = `${statePath}.backup`;
131
- writeFileSync(backupPath, JSON.stringify(state, null, 2));
132
- console.log(chalk.gray(`📦 Backup created: ${backupPath}\n`));
133
-
134
- // Write migrated state
135
- writeFileSync(statePath, JSON.stringify(state, null, 2));
136
-
137
- console.log(chalk.green('✅ Migration complete!\n'));
138
- console.log(chalk.bold('Next steps:\n'));
139
- console.log(chalk.gray('1. Review .morph/config/llm-interaction.json (created on next init)'));
140
- console.log(chalk.gray('2. Configure approval gates, checkpoints, patterns'));
141
- console.log(chalk.gray('3. Continue working with v3.0 features\n'));
142
-
143
- } catch (error) {
144
- console.error(chalk.red(`\n❌ Migration failed: ${error.message}\n`));
145
- process.exit(1);
146
- }
147
- });
148
-
149
- // Only parse if run directly (not imported as module)
150
- if (import.meta.url === `file://${process.argv[1]}`) {
151
- if (process.argv.length > 2) {
152
- program.parse(process.argv);
153
- } else {
154
- program.help();
155
- }
156
- }
157
-
158
- export default program;
@@ -1,346 +0,0 @@
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;
@@ -1,60 +0,0 @@
1
- /**
2
- * Architecture Validator
3
- *
4
- * Validates architectural patterns and standards
5
- *
6
- * MORPH-SPEC 3.0 - Continuous Validation
7
- */
8
-
9
- import { readFileSync } from 'fs';
10
- import { execSync } from 'child_process';
11
-
12
- /**
13
- * Validate architecture patterns
14
- *
15
- * @param {string} projectPath - Project root path
16
- * @param {Object} options - Validation options
17
- * @returns {Object} { passed, errors, warnings, issues }
18
- */
19
- export async function validateArchitecture(projectPath, options = {}) {
20
- const issues = [];
21
-
22
- // Find C# files
23
- try {
24
- const csFiles = execSync('find . -name "*.cs" -type f 2>/dev/null || true', {
25
- cwd: projectPath,
26
- encoding: 'utf8'
27
- }).trim().split('\n').filter(Boolean);
28
-
29
- if (csFiles.length === 0) {
30
- return {
31
- passed: true,
32
- errors: 0,
33
- warnings: 0,
34
- issues: [],
35
- message: 'No C# files found to validate'
36
- };
37
- }
38
-
39
- // Basic architecture validation (stub implementation)
40
- // TODO: Implement actual DI pattern validation, async/await checks, etc.
41
-
42
- } catch (err) {
43
- issues.push({
44
- level: 'warning',
45
- type: 'architecture',
46
- message: `Failed to validate architecture: ${err.message}`,
47
- solution: 'Check architecture patterns manually'
48
- });
49
- }
50
-
51
- const errors = issues.filter(i => i.level === 'error').length;
52
- const warnings = issues.filter(i => i.level === 'warning').length;
53
-
54
- return {
55
- passed: errors === 0,
56
- errors,
57
- warnings,
58
- issues
59
- };
60
- }