@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.
- package/CLAUDE.md +108 -946
- package/bin/morph-spec.js +284 -9
- package/bin/task-manager.cjs +102 -14
- package/bin/validate.js +4 -4
- package/docs/{v3.0 → next-generation}/AGENTS.md +1 -1
- package/docs/next-generation/CONTEXT-OPTIMIZATION.md +267 -0
- package/docs/next-generation/EXECUTION-FLOW.md +274 -0
- package/docs/next-generation/META-PROMPTS.md +235 -0
- package/docs/next-generation/MIGRATION-GUIDE.md +253 -0
- package/docs/next-generation/THREAD-MANAGEMENT.md +240 -0
- package/package.json +5 -5
- package/src/commands/agents/agents-fuse.js +97 -0
- package/src/commands/agents/micro-agent.js +112 -0
- package/src/commands/agents/spawn-team.js +69 -4
- package/src/commands/agents/squad-template.js +146 -0
- package/src/commands/analytics/analytics.js +176 -0
- package/src/commands/context/context-prime.js +63 -0
- package/src/commands/context/core-four.js +54 -0
- package/src/commands/mcp/mcp.js +102 -0
- package/src/commands/project/detect-agents.js +32 -2
- package/src/commands/project/detect.js +11 -1
- package/src/commands/project/doctor.js +573 -356
- package/src/commands/project/init.js +9 -2
- package/src/commands/project/update.js +13 -3
- package/src/commands/state/advance-phase.js +448 -416
- package/src/commands/state/state.js +14 -12
- package/src/commands/tasks/task.js +1 -1
- package/src/commands/templates/template-render.js +80 -1
- package/src/commands/threads/thread-template.js +103 -0
- package/src/commands/threads/threads.js +261 -0
- package/src/commands/trust/trust.js +205 -0
- package/src/{orchestrator.js → core/orchestrator.js} +8 -8
- package/src/core/state/state-manager.js +37 -17
- package/src/core/workflows/workflow-detector.js +114 -3
- package/src/lib/agents/micro-agent-factory.js +161 -0
- package/src/lib/analytics/analytics-engine.js +345 -0
- package/src/lib/checkpoints/checkpoint-hooks.js +298 -258
- package/src/lib/context/context-bundler.js +240 -0
- package/src/lib/context/context-optimizer.js +212 -0
- package/src/lib/context/context-tracker.js +273 -0
- package/src/lib/context/core-four-tracker.js +201 -0
- package/src/lib/context/mcp-optimizer.js +200 -0
- package/src/lib/detectors/index.js +1 -1
- package/src/lib/detectors/standards-generator.js +77 -17
- package/src/lib/detectors/structure-detector.js +67 -39
- package/src/lib/execution/fusion-executor.js +304 -0
- package/src/lib/execution/parallel-executor.js +270 -0
- package/src/lib/generators/context-generator.js +3 -3
- package/src/lib/generators/recap-generator.js +32 -12
- package/src/lib/hooks/hook-executor.js +169 -0
- package/src/lib/hooks/stop-hook-executor.js +286 -0
- package/src/lib/hops/hop-composer.js +221 -0
- package/src/lib/threads/thread-coordinator.js +238 -0
- package/src/lib/threads/thread-manager.js +317 -0
- package/src/lib/tracking/artifact-trail.js +202 -0
- package/src/lib/trust/trust-manager.js +269 -0
- package/src/lib/validators/design-system/design-system-validator.js +2 -2
- package/src/lib/validators/validation-runner.js +14 -30
- package/src/utils/hooks-installer.js +69 -0
- package/stacks/blazor-azure/.morph/config/agents.json +72 -3
- package/stacks/nextjs-supabase/.morph/config/agents.json +3 -3
- package/docs/llm-interaction-config.md +0 -735
- package/docs/v3.0/EXECUTION-FLOW.md +0 -1304
- package/src/commands/utils/migrate-state.js +0 -158
- package/src/commands/utils/upgrade.js +0 -346
- package/src/lib/validators/architecture-validator.js +0 -60
- package/src/lib/validators/content-validator.js +0 -164
- package/src/lib/validators/package-validator.js +0 -61
- package/src/lib/validators/ui-contrast-validator.js +0 -44
- package/stacks/blazor-azure/.claude/commands/morph-apply.md +0 -221
- package/stacks/blazor-azure/.claude/commands/morph-archive.md +0 -79
- package/stacks/blazor-azure/.claude/commands/morph-deploy.md +0 -529
- package/stacks/blazor-azure/.claude/commands/morph-infra.md +0 -209
- package/stacks/blazor-azure/.claude/commands/morph-preflight.md +0 -227
- package/stacks/blazor-azure/.claude/commands/morph-proposal.md +0 -122
- package/stacks/blazor-azure/.claude/commands/morph-status.md +0 -86
- package/stacks/blazor-azure/.claude/commands/morph-troubleshoot.md +0 -122
- package/stacks/blazor-azure/.claude/skills/level-0-meta/README.md +0 -7
- package/stacks/blazor-azure/.claude/skills/level-0-meta/code-review.md +0 -226
- package/stacks/blazor-azure/.claude/skills/level-0-meta/morph-checklist.md +0 -117
- package/stacks/blazor-azure/.claude/skills/level-0-meta/simulation-checklist.md +0 -77
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/README.md +0 -7
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/morph-replicate.md +0 -213
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-clarify.md +0 -131
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-design.md +0 -213
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-setup.md +0 -106
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-tasks.md +0 -164
- package/stacks/blazor-azure/.claude/skills/level-1-workflows/phase-uiux.md +0 -169
- package/stacks/blazor-azure/.claude/skills/level-2-domains/README.md +0 -14
- package/stacks/blazor-azure/.claude/skills/level-2-domains/ai-agents/ai-system-architect.md +0 -192
- package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/po-pm-advisor.md +0 -197
- package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/prompt-engineer.md +0 -189
- package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/seo-growth-hacker.md +0 -320
- package/stacks/blazor-azure/.claude/skills/level-2-domains/architecture/standards-architect.md +0 -156
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/api-designer.md +0 -59
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/dotnet-senior.md +0 -77
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ef-modeler.md +0 -58
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/hangfire-orchestrator.md +0 -126
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ms-agent-expert.md +0 -45
- package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/blazor-builder.md +0 -210
- package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/nextjs-expert.md +0 -154
- package/stacks/blazor-azure/.claude/skills/level-2-domains/frontend/ui-ux-designer.md +0 -191
- package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/azure-architect.md +0 -142
- package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/azure-deploy-specialist.md +0 -699
- package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/bicep-architect.md +0 -126
- package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/container-specialist.md +0 -131
- package/stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure/devops-engineer.md +0 -119
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/asaas-financial.md +0 -130
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/azure-identity.md +0 -142
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/clerk-auth.md +0 -108
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/hangfire-orchestrator.md +0 -64
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/resend-email.md +0 -119
- package/stacks/blazor-azure/.claude/skills/level-2-domains/quality/code-analyzer.md +0 -235
- package/stacks/blazor-azure/.claude/skills/level-2-domains/quality/testing-specialist.md +0 -126
- package/stacks/blazor-azure/.claude/skills/level-3-technologies/README.md +0 -7
- package/stacks/blazor-azure/.claude/skills/level-4-patterns/README.md +0 -7
- package/stacks/blazor-azure/.morph/archive/.gitkeep +0 -25
- package/stacks/blazor-azure/.morph/features/.gitkeep +0 -25
- package/stacks/blazor-azure/.morph/schemas/agent.schema.json +0 -296
- package/stacks/blazor-azure/.morph/schemas/tasks.schema.json +0 -220
- package/stacks/blazor-azure/.morph/specs/.gitkeep +0 -20
- package/stacks/blazor-azure/.morph/test-infra/example.bicep +0 -59
- package/stacks/nextjs-supabase/.claude/commands/morph-apply.md +0 -221
- package/stacks/nextjs-supabase/.claude/commands/morph-archive.md +0 -79
- package/stacks/nextjs-supabase/.claude/commands/morph-deploy.md +0 -529
- package/stacks/nextjs-supabase/.claude/commands/morph-infra.md +0 -209
- package/stacks/nextjs-supabase/.claude/commands/morph-preflight.md +0 -227
- package/stacks/nextjs-supabase/.claude/commands/morph-proposal.md +0 -122
- package/stacks/nextjs-supabase/.claude/commands/morph-status.md +0 -86
- package/stacks/nextjs-supabase/.claude/commands/morph-troubleshoot.md +0 -122
- package/stacks/nextjs-supabase/.claude/settings.local.json +0 -6
- package/stacks/nextjs-supabase/.claude/skills/level-2-domains/backend/dotnet-supabase.md +0 -244
- package/stacks/nextjs-supabase/.claude/skills/level-2-domains/frontend/nextjs-supabase.md +0 -335
- package/stacks/nextjs-supabase/.claude/skills/level-2-domains/infrastructure/easypanel-deployer.md +0 -189
- package/stacks/nextjs-supabase/.claude/skills/level-2-domains/integrations/supabase-expert.md +0 -50
- /package/docs/{v3.0 → next-generation}/ANALYSIS.md +0 -0
- /package/docs/{v3.0 → next-generation}/ARCHITECTURE.md +0 -0
- /package/docs/{v3.0 → next-generation}/FEATURES.md +0 -0
- /package/docs/{v3.0 → next-generation}/README.md +0 -0
- /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
|
-
}
|