@polymorphism-tech/morph-spec 2.2.0 → 2.4.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 +314 -1673
- package/LICENSE +72 -72
- package/README.md +515 -516
- package/bin/detect-agents.js +225 -225
- package/bin/morph-spec.js +358 -173
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/task-manager.js +429 -0
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/bin/validate.js +369 -0
- package/content/.azure/README.md +293 -293
- package/content/.azure/docs/azure-devops-setup.md +454 -454
- package/content/.azure/docs/branch-strategy.md +398 -398
- package/content/.azure/docs/local-development.md +515 -515
- package/content/.azure/pipelines/pipeline-variables.yml +34 -34
- package/content/.azure/pipelines/prod-pipeline.yml +319 -319
- package/content/.azure/pipelines/staging-pipeline.yml +234 -234
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
- package/content/.claude/commands/morph-apply.md +221 -158
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -0
- package/content/.claude/commands/morph-proposal.md +122 -101
- package/content/.claude/commands/morph-status.md +86 -86
- package/content/.claude/commands/morph-troubleshoot.md +122 -0
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/checklists/code-review.md +226 -0
- package/content/.claude/skills/checklists/morph-checklist.md +117 -0
- package/content/.claude/skills/checklists/simulation-checklist.md +77 -0
- package/content/.claude/skills/infra/bicep-architect.md +126 -419
- package/content/.claude/skills/infra/container-specialist.md +131 -437
- package/content/.claude/skills/infra/devops-engineer.md +119 -405
- package/content/.claude/skills/integrations/asaas-financial.md +130 -333
- package/content/.claude/skills/integrations/azure-identity.md +142 -309
- package/content/.claude/skills/integrations/clerk-auth.md +108 -290
- package/content/.claude/skills/integrations/resend-email.md +119 -0
- package/content/.claude/skills/specialists/ai-system-architect.md +192 -604
- package/content/.claude/skills/specialists/azure-architect.md +142 -142
- package/content/.claude/skills/specialists/code-analyzer.md +235 -0
- package/content/.claude/skills/specialists/dotnet-senior.md +287 -0
- package/content/.claude/skills/specialists/ef-modeler.md +113 -200
- package/content/.claude/skills/specialists/hangfire-orchestrator.md +126 -245
- package/content/.claude/skills/specialists/ms-agent-expert.md +109 -263
- package/content/.claude/skills/specialists/po-pm-advisor.md +197 -197
- package/content/.claude/skills/specialists/standards-architect.md +156 -78
- package/content/.claude/skills/specialists/testing-specialist.md +126 -0
- package/content/.claude/skills/specialists/ui-ux-designer.md +191 -1060
- package/content/.claude/skills/stacks/dotnet-blazor.md +210 -588
- package/content/.claude/skills/stacks/dotnet-nextjs.md +154 -402
- package/content/.claude/skills/workflows/morph-replicate.md +213 -0
- package/content/.claude/{commands/morph-clarify.md → skills/workflows/phase-clarify.md} +5 -58
- package/content/.claude/{commands/morph-design.md → skills/workflows/phase-design.md} +16 -86
- package/content/.claude/{commands/morph-setup.md → skills/workflows/phase-setup.md} +9 -17
- package/content/.claude/skills/workflows/phase-tasks.md +164 -0
- package/content/.claude/{commands/morph-uiux.md → skills/workflows/phase-uiux.md} +15 -88
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +378 -242
- package/content/.morph/config/config.template.json +89 -108
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/docs/workflows/design-impl.md +37 -0
- package/content/.morph/docs/workflows/fast-track.md +29 -0
- package/content/.morph/docs/workflows/full-morph.md +76 -0
- package/content/.morph/docs/workflows/standard.md +44 -0
- package/content/.morph/docs/workflows/ui-refresh.md +39 -0
- package/content/.morph/examples/api-nextjs/README.md +241 -241
- package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
- package/content/.morph/examples/api-nextjs/spec.md +399 -399
- package/content/.morph/examples/api-nextjs/tasks.md +168 -168
- package/content/.morph/examples/micro-saas/README.md +125 -125
- package/content/.morph/examples/micro-saas/contracts.cs +358 -358
- package/content/.morph/examples/micro-saas/decisions.md +246 -246
- package/content/.morph/examples/micro-saas/spec.md +236 -236
- package/content/.morph/examples/micro-saas/tasks.md +150 -150
- package/content/.morph/examples/multi-agent/README.md +309 -309
- package/content/.morph/examples/multi-agent/contracts.cs +433 -433
- package/content/.morph/examples/multi-agent/spec.md +479 -479
- package/content/.morph/examples/multi-agent/tasks.md +185 -185
- package/content/.morph/examples/scheduled-reports/decisions.md +158 -0
- package/content/.morph/examples/scheduled-reports/proposal.md +95 -0
- package/content/.morph/examples/scheduled-reports/spec.md +267 -0
- package/content/.morph/examples/state-v3.json +188 -0
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/README.md +190 -239
- package/content/.morph/hooks/pre-commit-agents.sh +24 -24
- package/content/.morph/hooks/pre-commit-all.sh +48 -48
- package/content/.morph/hooks/pre-commit-specs.sh +49 -49
- package/content/.morph/hooks/pre-commit-tests.sh +60 -60
- package/content/.morph/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/schemas/tasks.schema.json +220 -0
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/agent-framework-blazor-ui.md +359 -0
- package/content/.morph/standards/agent-framework-production.md +410 -0
- package/content/.morph/standards/agent-framework-setup.md +413 -453
- package/content/.morph/standards/agent-framework-workflows.md +349 -0
- package/content/.morph/standards/architecture.md +325 -325
- package/content/.morph/standards/azure.md +605 -379
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/dotnet10-migration.md +520 -494
- package/content/.morph/standards/fluent-ui-setup.md +590 -590
- package/content/.morph/standards/migration-guide.md +514 -514
- package/content/.morph/standards/passkeys-auth.md +423 -423
- package/content/.morph/standards/vector-search-rag.md +536 -536
- package/content/.morph/state.json +17 -17
- package/content/.morph/templates/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/agent.cs +163 -172
- package/content/.morph/templates/clarify-questions.md +159 -0
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts/Commands.cs +74 -0
- package/content/.morph/templates/contracts/Entities.cs +25 -0
- package/content/.morph/templates/contracts/Queries.cs +74 -0
- package/content/.morph/templates/contracts/README.md +74 -0
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/decisions.md +123 -106
- package/content/.morph/templates/design-system.css +226 -226
- package/content/.morph/templates/infra/.dockerignore.example +89 -89
- package/content/.morph/templates/infra/Dockerfile.example +82 -82
- package/content/.morph/templates/infra/README.md +286 -286
- package/content/.morph/templates/infra/app-insights.bicep +63 -63
- package/content/.morph/templates/infra/app-service.bicep +164 -164
- package/content/.morph/templates/infra/container-app-env.bicep +49 -49
- package/content/.morph/templates/infra/container-app.bicep +156 -156
- package/content/.morph/templates/infra/deploy-checklist.md +426 -0
- package/content/.morph/templates/infra/deploy.ps1 +229 -229
- package/content/.morph/templates/infra/deploy.sh +208 -208
- package/content/.morph/templates/infra/key-vault.bicep +91 -91
- package/content/.morph/templates/infra/main.bicep +189 -189
- package/content/.morph/templates/infra/parameters.dev.json +29 -29
- package/content/.morph/templates/infra/parameters.prod.json +29 -29
- package/content/.morph/templates/infra/parameters.staging.json +29 -29
- package/content/.morph/templates/infra/sql-database.bicep +103 -103
- package/content/.morph/templates/infra/storage.bicep +106 -106
- package/content/.morph/templates/integrations/asaas-client.cs +387 -387
- package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
- package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
- package/content/.morph/templates/integrations/clerk-config.cs +258 -258
- package/content/.morph/templates/job.cs +171 -171
- package/content/.morph/templates/migration.cs +83 -83
- package/content/.morph/templates/proposal.md +141 -155
- package/content/.morph/templates/recap.md +94 -105
- package/content/.morph/templates/repository.cs +141 -141
- package/content/.morph/templates/saas/subscription.cs +347 -347
- package/content/.morph/templates/saas/tenant.cs +338 -338
- package/content/.morph/templates/service.cs +139 -139
- package/content/.morph/templates/simulation.md +353 -0
- package/content/.morph/templates/spec.md +149 -148
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/state.template.json +222 -222
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/tasks.md +257 -235
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-components.md +362 -276
- package/content/.morph/templates/ui-design-system.md +286 -286
- package/content/.morph/templates/ui-flows.md +336 -336
- package/content/.morph/templates/ui-mockups.md +133 -133
- package/content/.morph/test-infra/example.bicep +59 -59
- package/content/CLAUDE.md +150 -442
- package/content/README.md +79 -79
- package/detectors/config-detector.js +223 -223
- package/detectors/conversation-analyzer.js +163 -163
- package/detectors/index.js +84 -84
- package/detectors/standards-generator.js +275 -275
- package/detectors/structure-detector.js +245 -250
- package/docs/README.md +144 -149
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
- package/docs/api/scripts/collapse.js +38 -38
- package/docs/api/scripts/commonNav.js +28 -28
- package/docs/api/scripts/linenumber.js +25 -25
- package/docs/api/scripts/nav.js +12 -12
- package/docs/api/scripts/polyfill.js +3 -3
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
- package/docs/api/scripts/prettify/lang-css.js +2 -2
- package/docs/api/scripts/prettify/prettify.js +28 -28
- package/docs/api/scripts/search.js +98 -98
- package/docs/api/styles/jsdoc.css +776 -776
- package/docs/api/styles/prettify.css +80 -80
- package/docs/examples.md +328 -328
- package/docs/getting-started.md +301 -302
- package/docs/installation.md +361 -361
- package/docs/templates.md +418 -418
- package/docs/validation-checklist.md +265 -266
- package/package.json +80 -80
- package/scripts/postinstall.js +132 -132
- package/src/commands/advance-phase.js +183 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -0
- package/src/commands/create-story.js +351 -351
- package/src/commands/detect-agents.js +139 -0
- package/src/commands/detect.js +104 -104
- package/src/commands/doctor.js +356 -280
- package/src/commands/generate.js +149 -149
- package/src/commands/init.js +258 -245
- package/src/commands/lint-fluent.js +352 -0
- package/src/commands/rollback-phase.js +185 -0
- package/src/commands/session-summary.js +291 -0
- package/src/commands/shard-spec.js +224 -224
- package/src/commands/sprint-status.js +250 -250
- package/src/commands/state.js +333 -333
- package/src/commands/sync.js +167 -167
- package/src/commands/task.js +78 -0
- package/src/commands/troubleshoot.js +222 -0
- package/src/commands/update.js +192 -159
- package/src/commands/validate-blazor-state.js +210 -0
- package/src/commands/validate-blazor.js +156 -0
- package/src/commands/validate-css.js +84 -0
- package/src/commands/validate-phase.js +221 -0
- package/src/lib/blazor-concurrency-analyzer.js +288 -0
- package/src/lib/blazor-state-validator.js +291 -0
- package/src/lib/blazor-validator.js +374 -0
- package/src/lib/complexity-analyzer.js +441 -292
- package/src/lib/continuous-validator.js +421 -0
- package/src/lib/css-validator.js +352 -0
- package/src/lib/decision-constraint-loader.js +109 -0
- package/src/lib/design-system-generator.js +298 -298
- package/src/lib/learning-system.js +520 -0
- package/src/lib/mockup-generator.js +366 -0
- package/src/lib/recap-generator.js +205 -0
- package/src/lib/state-manager.js +397 -340
- package/src/lib/troubleshoot-grep.js +194 -0
- package/src/lib/troubleshoot-index.js +144 -0
- package/src/lib/ui-detector.js +350 -0
- package/src/lib/validation-runner.js +231 -0
- package/src/lib/validators/architecture-validator.js +387 -0
- package/src/lib/validators/contract-compliance-validator.js +273 -0
- package/src/lib/validators/package-validator.js +360 -0
- package/src/lib/validators/ui-contrast-validator.js +422 -0
- package/src/utils/file-copier.js +179 -139
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- package/content/.claude/commands/morph-costs.md +0 -206
- package/content/.claude/commands/morph-tasks.md +0 -319
- package/content/.claude/skills/specialists/cost-guardian.md +0 -110
- package/content/.claude/skills/stacks/shopify.md +0 -445
- package/content/.morph/config/azure-pricing.json +0 -70
- package/content/.morph/config/azure-pricing.schema.json +0 -50
- package/content/.morph/hooks/pre-commit-costs.sh +0 -91
- package/docs/api/cost-calculator.js.html +0 -513
- package/docs/api/design-system-generator.js.html +0 -382
- package/docs/api/global.html +0 -5263
- package/docs/api/index.html +0 -96
- package/docs/api/state-manager.js.html +0 -423
- package/src/commands/cost.js +0 -181
- package/src/commands/update-pricing.js +0 -206
- package/src/lib/cost-calculator.js +0 -429
package/bin/validate.js
ADDED
|
@@ -0,0 +1,369 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Validation CLI
|
|
5
|
+
*
|
|
6
|
+
* Runs all validators (package, architecture, contrast) on the project.
|
|
7
|
+
* Part of Sprint 4: Continuous Validation + Learning System
|
|
8
|
+
*
|
|
9
|
+
* Usage:
|
|
10
|
+
* morph-spec validate [options]
|
|
11
|
+
* morph-spec validate packages
|
|
12
|
+
* morph-spec validate architecture
|
|
13
|
+
* morph-spec validate contrast
|
|
14
|
+
* morph-spec validate all
|
|
15
|
+
*
|
|
16
|
+
* MORPH-SPEC 3.0
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import chalk from 'chalk';
|
|
20
|
+
import { validatePackages } from '../src/lib/validators/package-validator.js';
|
|
21
|
+
import { validateArchitecture } from '../src/lib/validators/architecture-validator.js';
|
|
22
|
+
import { validateContrast } from '../src/lib/validators/ui-contrast-validator.js';
|
|
23
|
+
import { LearningSystem } from '../src/lib/learning-system.js';
|
|
24
|
+
|
|
25
|
+
const VALIDATORS = {
|
|
26
|
+
packages: {
|
|
27
|
+
name: 'Package Compatibility',
|
|
28
|
+
description: 'Validates NuGet package versions against .NET compatibility matrix',
|
|
29
|
+
run: validatePackages
|
|
30
|
+
},
|
|
31
|
+
architecture: {
|
|
32
|
+
name: 'Architecture Patterns',
|
|
33
|
+
description: 'Detects DI anti-patterns and lifecycle issues in Blazor',
|
|
34
|
+
run: validateArchitecture
|
|
35
|
+
},
|
|
36
|
+
contrast: {
|
|
37
|
+
name: 'UI Contrast (WCAG)',
|
|
38
|
+
description: 'Validates color contrast ratios for accessibility',
|
|
39
|
+
run: validateContrast
|
|
40
|
+
},
|
|
41
|
+
'spec-tasks': {
|
|
42
|
+
name: 'Spec vs Tasks Coverage',
|
|
43
|
+
description: 'Validates that tasks.json covers all requirements from spec.md',
|
|
44
|
+
run: validateSpecVsTasks
|
|
45
|
+
}
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Validate spec.md requirements are covered by tasks
|
|
50
|
+
*/
|
|
51
|
+
async function validateSpecVsTasks(projectPath, options = {}) {
|
|
52
|
+
const path = await import('path');
|
|
53
|
+
const fs = await import('fs');
|
|
54
|
+
const { loadState } = await import('../src/lib/state-manager.js');
|
|
55
|
+
|
|
56
|
+
const results = [];
|
|
57
|
+
let errors = 0;
|
|
58
|
+
let warnings = 0;
|
|
59
|
+
|
|
60
|
+
try {
|
|
61
|
+
const state = loadState(false);
|
|
62
|
+
if (!state || !state.features) {
|
|
63
|
+
return { status: 'ok', errors: 0, warnings: 0, message: 'No features to validate' };
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
for (const [featureName, feature] of Object.entries(state.features)) {
|
|
67
|
+
const outputsPath = path.join(projectPath, '.morph/project/outputs', featureName);
|
|
68
|
+
const specPath = path.join(outputsPath, 'spec.md');
|
|
69
|
+
|
|
70
|
+
if (!fs.existsSync(specPath)) {
|
|
71
|
+
continue; // No spec to validate
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// Read spec.md and extract requirements
|
|
75
|
+
const specContent = fs.readFileSync(specPath, 'utf-8');
|
|
76
|
+
|
|
77
|
+
// Extract requirement patterns (## Requisitos, - [ ], numbered lists, etc.)
|
|
78
|
+
const requirementPatterns = [
|
|
79
|
+
/^#+\s*Requisitos?\s*$/gim,
|
|
80
|
+
/^-\s*\[.\]\s*(.+)$/gm,
|
|
81
|
+
/^\d+\.\s+(.+)$/gm
|
|
82
|
+
];
|
|
83
|
+
|
|
84
|
+
const requirements = [];
|
|
85
|
+
for (const pattern of requirementPatterns) {
|
|
86
|
+
const matches = specContent.matchAll(pattern);
|
|
87
|
+
for (const match of matches) {
|
|
88
|
+
if (match[1]) {
|
|
89
|
+
requirements.push(match[1].trim());
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// Check if feature has tasks
|
|
95
|
+
if (!feature.tasks || (Array.isArray(feature.tasks) && feature.tasks.length === 0)) {
|
|
96
|
+
if (requirements.length > 0) {
|
|
97
|
+
warnings++;
|
|
98
|
+
results.push({
|
|
99
|
+
feature: featureName,
|
|
100
|
+
level: 'warning',
|
|
101
|
+
message: `Spec has ${requirements.length} requirements but no tasks defined`
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
continue;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Get task titles
|
|
108
|
+
const taskTitles = Array.isArray(feature.tasks)
|
|
109
|
+
? feature.tasks.map(t => t.title?.toLowerCase() || '')
|
|
110
|
+
: [];
|
|
111
|
+
|
|
112
|
+
// Check each requirement has a matching task (fuzzy match)
|
|
113
|
+
for (const req of requirements.slice(0, 20)) { // Limit to first 20
|
|
114
|
+
const reqLower = req.toLowerCase();
|
|
115
|
+
const hasMatch = taskTitles.some(title =>
|
|
116
|
+
title.includes(reqLower.slice(0, 20)) ||
|
|
117
|
+
reqLower.includes(title.slice(0, 20))
|
|
118
|
+
);
|
|
119
|
+
|
|
120
|
+
if (!hasMatch && options.verbose) {
|
|
121
|
+
console.log(chalk.gray(` ? Requirement may not be covered: "${req.slice(0, 50)}..."`));
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
if (options.verbose) {
|
|
126
|
+
console.log(chalk.green(` ✓ ${featureName}: ${requirements.length} requirements, ${taskTitles.length} tasks`));
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
} catch (error) {
|
|
131
|
+
if (options.verbose) {
|
|
132
|
+
console.log(chalk.yellow(` ⚠ Could not validate: ${error.message}`));
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
return {
|
|
137
|
+
status: errors > 0 ? 'error' : warnings > 0 ? 'warning' : 'ok',
|
|
138
|
+
errors,
|
|
139
|
+
warnings,
|
|
140
|
+
results
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Main validation command
|
|
146
|
+
*/
|
|
147
|
+
export async function validateCommand(args = []) {
|
|
148
|
+
const validatorName = args[0] || 'all';
|
|
149
|
+
const options = parseOptions(args);
|
|
150
|
+
|
|
151
|
+
console.log(chalk.bold.cyan('\n🔍 MORPH-SPEC Validator\n'));
|
|
152
|
+
|
|
153
|
+
if (validatorName === 'all') {
|
|
154
|
+
await runAllValidators(options);
|
|
155
|
+
} else if (VALIDATORS[validatorName]) {
|
|
156
|
+
await runValidator(validatorName, options);
|
|
157
|
+
} else {
|
|
158
|
+
console.error(chalk.red(`❌ Unknown validator: ${validatorName}`));
|
|
159
|
+
console.log(chalk.gray('\nAvailable validators:'));
|
|
160
|
+
for (const [name, validator] of Object.entries(VALIDATORS)) {
|
|
161
|
+
console.log(chalk.gray(` - ${name}: ${validator.description}`));
|
|
162
|
+
}
|
|
163
|
+
console.log(chalk.gray(' - all: Run all validators'));
|
|
164
|
+
process.exit(1);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
// Show learning insights if --insights flag
|
|
168
|
+
if (options.insights) {
|
|
169
|
+
await showLearningInsights(options);
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
/**
|
|
174
|
+
* Run all validators
|
|
175
|
+
*/
|
|
176
|
+
async function runAllValidators(options) {
|
|
177
|
+
const results = {};
|
|
178
|
+
let totalErrors = 0;
|
|
179
|
+
let totalWarnings = 0;
|
|
180
|
+
|
|
181
|
+
console.log(chalk.gray('Running all validators...\n'));
|
|
182
|
+
|
|
183
|
+
for (const [name, validator] of Object.entries(VALIDATORS)) {
|
|
184
|
+
console.log(chalk.cyan(`\n▸ ${validator.name}`));
|
|
185
|
+
console.log(chalk.gray(` ${validator.description}\n`));
|
|
186
|
+
|
|
187
|
+
try {
|
|
188
|
+
const result = await validator.run('.', {
|
|
189
|
+
verbose: options.verbose,
|
|
190
|
+
autoFix: options.autoFix
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
results[name] = result;
|
|
194
|
+
|
|
195
|
+
// Count errors/warnings
|
|
196
|
+
if (result.errors !== undefined) {
|
|
197
|
+
totalErrors += result.errors;
|
|
198
|
+
totalWarnings += result.warnings || 0;
|
|
199
|
+
} else if (result.results) {
|
|
200
|
+
// Count issues in results array
|
|
201
|
+
const issues = result.results.flatMap(r => r.issues || []);
|
|
202
|
+
totalErrors += issues.filter(i => i.level === 'error').length;
|
|
203
|
+
totalWarnings += issues.filter(i => i.level === 'warning').length;
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
} catch (error) {
|
|
207
|
+
console.error(chalk.red(` ❌ Failed: ${error.message}`));
|
|
208
|
+
results[name] = { status: 'error', error: error.message };
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// Summary
|
|
213
|
+
console.log(chalk.bold.cyan('\n━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\n'));
|
|
214
|
+
console.log(chalk.bold('📊 Validation Summary\n'));
|
|
215
|
+
|
|
216
|
+
if (totalErrors === 0 && totalWarnings === 0) {
|
|
217
|
+
console.log(chalk.green('✅ All validations passed!'));
|
|
218
|
+
} else {
|
|
219
|
+
if (totalErrors > 0) {
|
|
220
|
+
console.log(chalk.red(`❌ ${totalErrors} error(s) found`));
|
|
221
|
+
}
|
|
222
|
+
if (totalWarnings > 0) {
|
|
223
|
+
console.log(chalk.yellow(`⚠️ ${totalWarnings} warning(s) found`));
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
console.log('');
|
|
228
|
+
|
|
229
|
+
// Exit with error code if errors found
|
|
230
|
+
if (totalErrors > 0 && !options.noFail) {
|
|
231
|
+
process.exit(1);
|
|
232
|
+
}
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
/**
|
|
236
|
+
* Run single validator
|
|
237
|
+
*/
|
|
238
|
+
async function runValidator(name, options) {
|
|
239
|
+
const validator = VALIDATORS[name];
|
|
240
|
+
|
|
241
|
+
console.log(chalk.cyan(`▸ ${validator.name}`));
|
|
242
|
+
console.log(chalk.gray(` ${validator.description}\n`));
|
|
243
|
+
|
|
244
|
+
try {
|
|
245
|
+
const result = await validator.run('.', {
|
|
246
|
+
verbose: true, // Always verbose for single validator
|
|
247
|
+
autoFix: options.autoFix,
|
|
248
|
+
wcagLevel: options.wcagLevel
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
if (result.status === 'ok') {
|
|
252
|
+
console.log(chalk.green('\n✅ Validation passed!\n'));
|
|
253
|
+
} else {
|
|
254
|
+
console.log(chalk.yellow('\n⚠️ Validation completed with issues\n'));
|
|
255
|
+
|
|
256
|
+
if (result.errors > 0 && !options.noFail) {
|
|
257
|
+
process.exit(1);
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
} catch (error) {
|
|
262
|
+
console.error(chalk.red(`\n❌ Validation failed: ${error.message}\n`));
|
|
263
|
+
if (!options.noFail) {
|
|
264
|
+
process.exit(1);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
|
|
269
|
+
/**
|
|
270
|
+
* Show learning insights
|
|
271
|
+
*/
|
|
272
|
+
async function showLearningInsights(options) {
|
|
273
|
+
console.log(chalk.bold.cyan('\n━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━\n'));
|
|
274
|
+
|
|
275
|
+
const learner = new LearningSystem('.');
|
|
276
|
+
learner.formatInsights();
|
|
277
|
+
|
|
278
|
+
if (options.verbose) {
|
|
279
|
+
const suggestions = learner.getAllSuggestions();
|
|
280
|
+
learner.formatSuggestions(suggestions);
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
/**
|
|
285
|
+
* Parse command-line options
|
|
286
|
+
*/
|
|
287
|
+
function parseOptions(args) {
|
|
288
|
+
const options = {
|
|
289
|
+
verbose: false,
|
|
290
|
+
autoFix: false,
|
|
291
|
+
noFail: false,
|
|
292
|
+
insights: false,
|
|
293
|
+
wcagLevel: 'AA'
|
|
294
|
+
};
|
|
295
|
+
|
|
296
|
+
for (const arg of args) {
|
|
297
|
+
if (arg === '--verbose' || arg === '-v') {
|
|
298
|
+
options.verbose = true;
|
|
299
|
+
} else if (arg === '--auto-fix' || arg === '--fix') {
|
|
300
|
+
options.autoFix = true;
|
|
301
|
+
} else if (arg === '--no-fail') {
|
|
302
|
+
options.noFail = true;
|
|
303
|
+
} else if (arg === '--insights' || arg === '-i') {
|
|
304
|
+
options.insights = true;
|
|
305
|
+
} else if (arg === '--wcag-aaa') {
|
|
306
|
+
options.wcagLevel = 'AAA';
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
return options;
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
/**
|
|
314
|
+
* Show help
|
|
315
|
+
*/
|
|
316
|
+
export function showValidateHelp() {
|
|
317
|
+
console.log(`
|
|
318
|
+
${chalk.bold.cyan('morph-spec validate')} - Run project validations
|
|
319
|
+
|
|
320
|
+
${chalk.bold('USAGE')}
|
|
321
|
+
morph-spec validate [validator] [options]
|
|
322
|
+
|
|
323
|
+
${chalk.bold('VALIDATORS')}
|
|
324
|
+
all Run all validators (default)
|
|
325
|
+
packages Validate NuGet package compatibility (.NET 10)
|
|
326
|
+
architecture Validate architecture patterns (DI, lifecycle)
|
|
327
|
+
contrast Validate UI contrast ratios (WCAG 2.1)
|
|
328
|
+
|
|
329
|
+
${chalk.bold('OPTIONS')}
|
|
330
|
+
--verbose, -v Show detailed output
|
|
331
|
+
--auto-fix, --fix Auto-fix issues where possible
|
|
332
|
+
--no-fail Don't exit with error code
|
|
333
|
+
--insights, -i Show learning insights
|
|
334
|
+
--wcag-aaa Use WCAG AAA standard (stricter)
|
|
335
|
+
|
|
336
|
+
${chalk.bold('EXAMPLES')}
|
|
337
|
+
morph-spec validate # Run all validators
|
|
338
|
+
morph-spec validate packages --fix # Validate packages and auto-fix
|
|
339
|
+
morph-spec validate architecture -v # Verbose architecture validation
|
|
340
|
+
morph-spec validate contrast --wcag-aaa # Stricter contrast validation
|
|
341
|
+
morph-spec validate all --insights # All validators + AI insights
|
|
342
|
+
|
|
343
|
+
${chalk.bold('EXIT CODES')}
|
|
344
|
+
0 All validations passed
|
|
345
|
+
1 Validation errors found (unless --no-fail)
|
|
346
|
+
|
|
347
|
+
${chalk.bold('INTEGRATION')}
|
|
348
|
+
Add to pre-commit hook:
|
|
349
|
+
morph-spec validate --no-fail
|
|
350
|
+
|
|
351
|
+
CI/CD pipeline:
|
|
352
|
+
morph-spec validate || exit 1
|
|
353
|
+
`);
|
|
354
|
+
}
|
|
355
|
+
|
|
356
|
+
// Run if called directly
|
|
357
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
358
|
+
const args = process.argv.slice(2);
|
|
359
|
+
|
|
360
|
+
if (args.includes('--help') || args.includes('-h')) {
|
|
361
|
+
showValidateHelp();
|
|
362
|
+
process.exit(0);
|
|
363
|
+
}
|
|
364
|
+
|
|
365
|
+
validateCommand(args).catch(error => {
|
|
366
|
+
console.error(chalk.red(`\n❌ Error: ${error.message}\n`));
|
|
367
|
+
process.exit(1);
|
|
368
|
+
});
|
|
369
|
+
}
|