@polymorphism-tech/morph-spec 3.2.0 → 4.3.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/README.md +1 -14
- package/bin/detect-agents.js +1 -1
- package/bin/morph-spec.js +403 -40
- package/bin/validate.js +5 -5
- package/docs/getting-started.md +0 -5
- package/docs/next-generation/AGENTS.md +521 -0
- package/docs/next-generation/ANALYSIS.md +555 -0
- package/docs/next-generation/ARCHITECTURE.md +436 -0
- package/docs/next-generation/CONTEXT-OPTIMIZATION.md +267 -0
- package/docs/next-generation/EXECUTION-FLOW.md +274 -0
- package/docs/next-generation/FEATURES.md +688 -0
- package/docs/next-generation/META-PROMPTS.md +235 -0
- package/docs/next-generation/MIGRATION-GUIDE.md +253 -0
- package/docs/next-generation/README.md +231 -0
- package/docs/next-generation/ROADMAP.md +801 -0
- package/docs/next-generation/THREAD-MANAGEMENT.md +240 -0
- package/docs/validation-checklist.md +0 -1
- package/package.json +5 -5
- package/src/commands/agents/agents-fuse.js +96 -0
- package/src/commands/agents/index.js +4 -0
- package/src/commands/agents/micro-agent.js +112 -0
- package/src/commands/{spawn-team.js → agents/spawn-team.js} +237 -172
- 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/{create-story.js → feature/create-story.js} +357 -354
- package/src/commands/feature/index.js +6 -0
- package/src/commands/{shard-spec.js → feature/shard-spec.js} +2 -2
- package/src/commands/{sprint-status.js → feature/sprint-status.js} +1 -1
- package/src/commands/{generate-context.js → generation/generate-context.js} +40 -40
- package/src/commands/{generate.js → generation/generate.js} +4 -4
- package/src/commands/generation/index.js +5 -0
- package/src/commands/index.js +16 -0
- package/src/commands/{capture-pattern.js → learning/capture-pattern.js} +121 -121
- package/src/commands/learning/index.js +5 -0
- package/src/commands/mcp/mcp.js +102 -0
- package/src/commands/{detect-agents.js → project/detect-agents.js} +178 -178
- package/src/commands/project/detect-workflow.js +174 -0
- package/src/commands/{detect.js → project/detect.js} +104 -104
- package/src/commands/{doctor.js → project/doctor.js} +221 -4
- package/src/commands/project/index.js +10 -0
- package/src/commands/{init.js → project/init.js} +305 -295
- package/src/commands/{sync.js → project/sync.js} +167 -167
- package/src/commands/{update.js → project/update.js} +240 -240
- package/src/commands/{advance-phase.js → state/advance-phase.js} +101 -25
- package/src/commands/{approve.js → state/approve.js} +221 -221
- package/src/commands/state/index.js +8 -0
- package/src/commands/{rollback-phase.js → state/rollback-phase.js} +185 -185
- package/src/commands/{state.js → state/state.js} +334 -334
- package/src/commands/{validate-phase.js → state/validate-phase.js} +221 -221
- package/src/commands/tasks/index.js +4 -0
- package/src/commands/{task.js → tasks/task.js} +78 -78
- package/src/commands/templates/index.js +8 -0
- package/src/commands/templates/template-customize.js +101 -0
- package/src/commands/templates/template-list.js +128 -0
- package/src/commands/templates/template-render.js +174 -0
- package/src/commands/templates/template-show.js +131 -0
- package/src/commands/templates/template-validate.js +91 -0
- 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/commands/utils/index.js +7 -0
- package/src/commands/{session-summary.js → utils/session-summary.js} +291 -291
- package/src/commands/{troubleshoot.js → utils/troubleshoot.js} +222 -222
- package/src/commands/{analyze-blazor-concurrency.js → validation/analyze-blazor-concurrency.js} +193 -193
- package/src/commands/validation/index.js +8 -0
- package/src/commands/{lint-fluent.js → validation/lint-fluent.js} +352 -352
- package/src/commands/{validate-blazor-state.js → validation/validate-blazor-state.js} +210 -210
- package/src/commands/{validate-blazor.js → validation/validate-blazor.js} +156 -156
- package/src/commands/{validate-css.js → validation/validate-css.js} +84 -84
- package/src/core/index.js +10 -0
- package/src/{orchestrator.js → core/orchestrator.js} +8 -8
- package/src/core/registry/command-registry.js +302 -0
- package/src/core/registry/index.js +8 -0
- package/src/core/registry/validator-registry.js +204 -0
- package/src/core/state/index.js +8 -0
- package/src/{lib → core/state}/phase-state-machine.js +214 -214
- package/src/{lib → core/state}/state-manager.js +588 -534
- package/src/core/templates/index.js +9 -0
- package/src/core/templates/template-registry.js +335 -0
- package/src/core/templates/template-renderer.js +477 -0
- package/src/core/templates/template-validator.js +296 -0
- package/src/core/workflows/index.js +7 -0
- package/src/core/workflows/workflow-detector.js +452 -0
- package/src/lib/agents/micro-agent-factory.js +161 -0
- package/src/lib/{complexity-analyzer.js → analysis/complexity-analyzer.js} +441 -441
- package/src/lib/analysis/index.js +7 -0
- package/src/lib/analytics/analytics-engine.js +345 -0
- package/src/lib/{checkpoint-hooks.js → checkpoints/checkpoint-hooks.js} +35 -0
- package/src/lib/checkpoints/index.js +7 -0
- 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/config-detector.js +223 -223
- package/src/lib/detectors/conversation-analyzer.js +163 -163
- package/src/lib/{design-system-detector.js → detectors/design-system-detector.js} +187 -187
- package/src/lib/detectors/index.js +87 -84
- package/src/lib/detectors/standards-generator.js +275 -275
- package/src/lib/detectors/structure-detector.js +245 -245
- package/src/lib/execution/fusion-executor.js +304 -0
- package/src/lib/execution/parallel-executor.js +270 -0
- package/src/lib/{context-generator.js → generators/context-generator.js} +526 -516
- package/src/lib/generators/index.js +10 -0
- package/src/lib/{metadata-extractor.js → generators/metadata-extractor.js} +387 -380
- package/src/lib/{recap-generator.js → generators/recap-generator.js} +205 -205
- 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/learning/index.js +7 -0
- package/src/lib/orchestration/index.js +7 -0
- package/src/lib/{team-orchestrator.js → orchestration/team-orchestrator.js} +323 -323
- package/src/lib/stacks/index.js +7 -0
- package/src/lib/{stack-resolver.js → stacks/stack-resolver.js} +180 -148
- package/src/lib/standards/index.js +7 -0
- package/src/lib/{standards-context-injector.js → standards/standards-context-injector.js} +298 -288
- 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/troubleshooting/index.js +8 -0
- package/src/lib/{troubleshoot-grep.js → troubleshooting/troubleshoot-grep.js} +204 -204
- package/src/lib/{troubleshoot-index.js → troubleshooting/troubleshoot-index.js} +144 -144
- package/src/lib/trust/trust-manager.js +269 -0
- package/src/lib/validators/{architecture-validator.js → architecture/architecture-validator.js} +8 -8
- package/src/lib/validators/architecture/index.js +7 -0
- package/src/lib/{blazor-concurrency-analyzer.js → validators/blazor/blazor-concurrency-analyzer.js} +277 -288
- package/src/lib/{blazor-state-validator.js → validators/blazor/blazor-state-validator.js} +279 -291
- package/src/lib/{blazor-validator.js → validators/blazor/blazor-validator.js} +369 -374
- package/src/lib/validators/blazor/index.js +9 -0
- package/src/lib/validators/{content-validator.js → content/content-validator.js} +351 -351
- package/src/lib/validators/content/index.js +7 -0
- package/src/lib/validators/{contract-compliance-validator.js → contracts/contract-compliance-validator.js} +273 -273
- package/src/lib/validators/contracts/index.js +7 -0
- package/src/lib/{css-validator.js → validators/css/css-validator.js} +352 -352
- package/src/lib/validators/css/index.js +7 -0
- package/src/lib/validators/{design-system-validator.js → design-system/design-system-validator.js} +231 -231
- package/src/lib/validators/design-system/index.js +7 -0
- package/src/lib/validators/packages/index.js +7 -0
- package/src/lib/validators/shared/index.js +12 -0
- package/src/lib/validators/shared/issue-counter.js +18 -0
- package/src/lib/validators/shared/result-formatter.js +124 -0
- package/src/lib/{spec-validator.js → validators/spec-validator.js} +258 -258
- package/src/lib/validators/ui/index.js +7 -0
- package/src/lib/{validation-runner.js → validators/validation-runner.js} +286 -284
- package/src/ui/wizard-questions.js +0 -2
- package/src/utils/color-utils.js +70 -0
- package/src/utils/file-copier.js +188 -189
- package/src/utils/process-handler.js +97 -0
- package/stacks/blazor-azure/.morph/config/agents.json +948 -764
- package/stacks/blazor-azure/.morph/hooks/{pre-commit-tests.sh → pre-commit/tests-csharp.sh} +3 -2
- package/stacks/blazor-azure/.morph/templates/infrastructure/github/workflows/cd-prod.yml.hbs +41 -0
- package/stacks/blazor-azure/.morph/templates/infrastructure/github/workflows/cd-staging.yml.hbs +24 -0
- package/stacks/blazor-azure/.morph/templates/infrastructure/github/workflows/ci-build.yml.hbs +23 -0
- package/stacks/nextjs-supabase/.morph/config/agents.json +345 -345
- package/stacks/nextjs-supabase/.morph/hooks/pre-commit/tests-typescript.sh +61 -0
- package/stacks/nextjs-supabase/.morph/templates/infrastructure/github/workflows/cd-prod.yml.hbs +22 -0
- package/stacks/nextjs-supabase/.morph/templates/infrastructure/github/workflows/cd-staging.yml.hbs +22 -0
- package/stacks/nextjs-supabase/.morph/templates/infrastructure/github/workflows/ci-build.yml.hbs +35 -0
- package/stacks/nextjs-supabase/README.md +6 -15
- package/CLAUDE.md +0 -648
- package/bin/render-template.js +0 -349
- package/bin/semantic-detect-agents.js +0 -247
- package/bin/validate-agents-skills.js +0 -257
- package/bin/validate-agents.js +0 -70
- package/bin/validate-phase.js +0 -263
- package/docs/examples.md +0 -328
- package/docs/llm-interaction-config.md +0 -735
- package/scripts/reorganize-skills.cjs +0 -175
- package/scripts/validate-agents-structure.cjs +0 -52
- package/scripts/validate-skills.cjs +0 -180
- package/src/commands/deploy.js +0 -780
- package/src/commands/migrate-state.js +0 -158
- package/src/commands/upgrade.js +0 -346
- package/src/lib/continuous-validator.js +0 -421
- package/src/lib/decision-constraint-loader.js +0 -109
- package/src/lib/design-system-scaffolder.js +0 -299
- package/src/lib/hook-executor.js +0 -257
- package/src/lib/mockup-generator.js +0 -366
- package/src/lib/ui-detector.js +0 -350
- package/src/llm/schema-validator.js +0 -121
- package/src/sanitizer/.gitkeep +0 -0
- package/src/scanner/.gitkeep +0 -0
- package/src/types/index.js +0 -477
- package/src/ui/.gitkeep +0 -0
- package/src/writer/.gitkeep +0 -0
- package/stacks/blazor-azure/.azure/README.md +0 -293
- package/stacks/blazor-azure/.azure/docs/azure-devops-setup.md +0 -454
- package/stacks/blazor-azure/.azure/docs/branch-strategy.md +0 -398
- package/stacks/blazor-azure/.azure/docs/local-development.md +0 -515
- package/stacks/blazor-azure/.azure/pipelines/pipeline-variables.yml +0 -34
- package/stacks/blazor-azure/.azure/pipelines/prod-pipeline.yml +0 -319
- package/stacks/blazor-azure/.azure/pipelines/staging-pipeline.yml +0 -234
- package/stacks/blazor-azure/.azure/pipelines/templates/build-dotnet.yml +0 -75
- package/stacks/blazor-azure/.azure/pipelines/templates/deploy-app-service.yml +0 -94
- package/stacks/blazor-azure/.azure/pipelines/templates/deploy-container-app.yml +0 -120
- package/stacks/blazor-azure/.azure/pipelines/templates/infra-deploy.yml +0 -90
- 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/settings.local.json +0 -15
- 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/dotnet-senior.md +0 -287
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ef-modeler.md +0 -113
- 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 -109
- 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/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/docs/STORY-DRIVEN-DEVELOPMENT.md +0 -392
- package/stacks/blazor-azure/.morph/docs/workflows/design-impl.md +0 -37
- package/stacks/blazor-azure/.morph/docs/workflows/enforcement-pipeline.md +0 -668
- package/stacks/blazor-azure/.morph/docs/workflows/fast-track.md +0 -29
- package/stacks/blazor-azure/.morph/docs/workflows/full-morph.md +0 -76
- package/stacks/blazor-azure/.morph/docs/workflows/standard.md +0 -44
- package/stacks/blazor-azure/.morph/docs/workflows/ui-refresh.md +0 -39
- package/stacks/blazor-azure/.morph/examples/api-nextjs/README.md +0 -241
- package/stacks/blazor-azure/.morph/examples/api-nextjs/contracts.ts +0 -307
- package/stacks/blazor-azure/.morph/examples/api-nextjs/spec.md +0 -399
- package/stacks/blazor-azure/.morph/examples/api-nextjs/tasks.md +0 -168
- package/stacks/blazor-azure/.morph/examples/micro-saas/README.md +0 -125
- package/stacks/blazor-azure/.morph/examples/micro-saas/contracts.cs +0 -358
- package/stacks/blazor-azure/.morph/examples/micro-saas/decisions.md +0 -246
- package/stacks/blazor-azure/.morph/examples/micro-saas/spec.md +0 -236
- package/stacks/blazor-azure/.morph/examples/micro-saas/tasks.md +0 -150
- package/stacks/blazor-azure/.morph/examples/multi-agent/README.md +0 -309
- package/stacks/blazor-azure/.morph/examples/multi-agent/contracts.cs +0 -433
- package/stacks/blazor-azure/.morph/examples/multi-agent/spec.md +0 -479
- package/stacks/blazor-azure/.morph/examples/multi-agent/tasks.md +0 -185
- package/stacks/blazor-azure/.morph/examples/scheduled-reports/decisions.md +0 -158
- package/stacks/blazor-azure/.morph/examples/scheduled-reports/proposal.md +0 -95
- package/stacks/blazor-azure/.morph/examples/scheduled-reports/spec.md +0 -267
- package/stacks/blazor-azure/.morph/examples/state-v3.json +0 -188
- package/stacks/blazor-azure/.morph/features/.gitkeep +0 -25
- package/stacks/blazor-azure/.morph/hooks/README.md +0 -348
- package/stacks/blazor-azure/.morph/hooks/pre-commit-agents.sh +0 -24
- package/stacks/blazor-azure/.morph/hooks/pre-commit-all.sh +0 -48
- package/stacks/blazor-azure/.morph/hooks/pre-commit-specs.sh +0 -49
- package/stacks/blazor-azure/.morph/hooks/task-completed.js +0 -73
- package/stacks/blazor-azure/.morph/hooks/teammate-idle.js +0 -68
- 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/standards/agent-framework-blazor-ui.md +0 -359
- package/stacks/blazor-azure/.morph/standards/agent-framework-production.md +0 -410
- package/stacks/blazor-azure/.morph/standards/agent-framework-setup.md +0 -413
- package/stacks/blazor-azure/.morph/standards/agent-framework-workflows.md +0 -349
- package/stacks/blazor-azure/.morph/standards/agent-teams-workflow.md +0 -474
- package/stacks/blazor-azure/.morph/standards/architecture.md +0 -325
- package/stacks/blazor-azure/.morph/standards/azure.md +0 -605
- package/stacks/blazor-azure/.morph/standards/coding.md +0 -377
- package/stacks/blazor-azure/.morph/standards/dotnet10-migration.md +0 -520
- package/stacks/blazor-azure/.morph/standards/fluent-ui-setup.md +0 -590
- package/stacks/blazor-azure/.morph/standards/migration-guide.md +0 -514
- package/stacks/blazor-azure/.morph/standards/passkeys-auth.md +0 -423
- package/stacks/blazor-azure/.morph/standards/vector-search-rag.md +0 -536
- package/stacks/blazor-azure/.morph/templates/CONTEXT-FEATURE.md +0 -276
- package/stacks/blazor-azure/.morph/templates/CONTEXT.md +0 -170
- package/stacks/blazor-azure/.morph/templates/FluentDesignTheme.cs +0 -149
- package/stacks/blazor-azure/.morph/templates/MudTheme.cs +0 -281
- package/stacks/blazor-azure/.morph/templates/agent.cs +0 -163
- package/stacks/blazor-azure/.morph/templates/clarify-questions.md +0 -159
- package/stacks/blazor-azure/.morph/templates/component.razor +0 -239
- package/stacks/blazor-azure/.morph/templates/contracts/Commands.cs +0 -74
- package/stacks/blazor-azure/.morph/templates/contracts/Entities.cs +0 -25
- package/stacks/blazor-azure/.morph/templates/contracts/Queries.cs +0 -74
- package/stacks/blazor-azure/.morph/templates/contracts/README.md +0 -74
- package/stacks/blazor-azure/.morph/templates/contracts.cs +0 -217
- package/stacks/blazor-azure/.morph/templates/decisions.md +0 -123
- package/stacks/blazor-azure/.morph/templates/design-system.css +0 -226
- package/stacks/blazor-azure/.morph/templates/infra/.dockerignore.example +0 -89
- package/stacks/blazor-azure/.morph/templates/infra/Dockerfile.example +0 -82
- package/stacks/blazor-azure/.morph/templates/infra/README.md +0 -286
- package/stacks/blazor-azure/.morph/templates/infra/app-insights.bicep +0 -63
- package/stacks/blazor-azure/.morph/templates/infra/app-service.bicep +0 -164
- package/stacks/blazor-azure/.morph/templates/infra/azure-pipelines-deploy.yml +0 -480
- package/stacks/blazor-azure/.morph/templates/infra/container-app-env.bicep +0 -49
- package/stacks/blazor-azure/.morph/templates/infra/container-app.bicep +0 -156
- package/stacks/blazor-azure/.morph/templates/infra/deploy-checklist.md +0 -426
- package/stacks/blazor-azure/.morph/templates/infra/deploy.ps1 +0 -229
- package/stacks/blazor-azure/.morph/templates/infra/deploy.sh +0 -208
- package/stacks/blazor-azure/.morph/templates/infra/key-vault.bicep +0 -91
- package/stacks/blazor-azure/.morph/templates/infra/main.bicep +0 -189
- package/stacks/blazor-azure/.morph/templates/infra/parameters.dev.json +0 -29
- package/stacks/blazor-azure/.morph/templates/infra/parameters.prod.json +0 -29
- package/stacks/blazor-azure/.morph/templates/infra/parameters.staging.json +0 -29
- package/stacks/blazor-azure/.morph/templates/infra/sql-database.bicep +0 -103
- package/stacks/blazor-azure/.morph/templates/infra/storage.bicep +0 -106
- package/stacks/blazor-azure/.morph/templates/integrations/asaas-client.cs +0 -387
- package/stacks/blazor-azure/.morph/templates/integrations/asaas-webhook.cs +0 -351
- package/stacks/blazor-azure/.morph/templates/integrations/azure-identity-config.cs +0 -288
- package/stacks/blazor-azure/.morph/templates/integrations/clerk-config.cs +0 -258
- package/stacks/blazor-azure/.morph/templates/job.cs +0 -171
- package/stacks/blazor-azure/.morph/templates/migration.cs +0 -83
- package/stacks/blazor-azure/.morph/templates/proposal.md +0 -141
- package/stacks/blazor-azure/.morph/templates/recap.md +0 -94
- package/stacks/blazor-azure/.morph/templates/repository.cs +0 -141
- package/stacks/blazor-azure/.morph/templates/saas/subscription.cs +0 -347
- package/stacks/blazor-azure/.morph/templates/saas/tenant.cs +0 -338
- package/stacks/blazor-azure/.morph/templates/service.cs +0 -139
- package/stacks/blazor-azure/.morph/templates/simulation.md +0 -353
- package/stacks/blazor-azure/.morph/templates/spec.md +0 -149
- package/stacks/blazor-azure/.morph/templates/sprint-status.yaml +0 -68
- package/stacks/blazor-azure/.morph/templates/state.template.json +0 -222
- package/stacks/blazor-azure/.morph/templates/story.md +0 -143
- package/stacks/blazor-azure/.morph/templates/tasks.md +0 -257
- package/stacks/blazor-azure/.morph/templates/test.cs +0 -239
- package/stacks/blazor-azure/.morph/templates/ui-components.md +0 -362
- package/stacks/blazor-azure/.morph/templates/ui-design-system.md +0 -286
- package/stacks/blazor-azure/.morph/templates/ui-flows.md +0 -336
- package/stacks/blazor-azure/.morph/templates/ui-mockups.md +0 -133
- package/stacks/blazor-azure/.morph/test-infra/example.bicep +0 -59
- 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 -170
- package/stacks/nextjs-supabase/.morph/docs/easypanel-setup.md +0 -169
- package/stacks/nextjs-supabase/.morph/docs/supabase-mcp-setup.md +0 -247
- package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/README.md +0 -697
- package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/spec.md +0 -85
- package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/tasks.md +0 -86
- package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/README.md +0 -498
- package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/decisions.md +0 -121
- package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/spec.md +0 -138
- package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/tasks.md +0 -162
- package/stacks/nextjs-supabase/.morph/standards/easypanel-deploy.md +0 -191
- package/stacks/nextjs-supabase/.morph/standards/nextjs-patterns.md +0 -193
- package/stacks/nextjs-supabase/.morph/standards/supabase-auth.md +0 -171
- package/stacks/nextjs-supabase/.morph/standards/supabase-pgvector.md +0 -164
- package/stacks/nextjs-supabase/.morph/standards/supabase-rls.md +0 -179
- package/stacks/nextjs-supabase/.morph/standards/supabase-storage.md +0 -148
- package/stacks/nextjs-supabase/.morph/templates/contracts.cs +0 -173
- package/stacks/nextjs-supabase/.morph/templates/contracts.ts +0 -168
- package/stacks/nextjs-supabase/.morph/templates/decisions.md +0 -115
- package/stacks/nextjs-supabase/.morph/templates/dockerfile-api.dockerfile +0 -38
- package/stacks/nextjs-supabase/.morph/templates/dockerfile-web.dockerfile +0 -48
- package/stacks/nextjs-supabase/.morph/templates/proposal.md +0 -145
- package/stacks/nextjs-supabase/.morph/templates/recap.md +0 -134
- package/stacks/nextjs-supabase/.morph/templates/rls-policy.sql +0 -57
- package/stacks/nextjs-supabase/.morph/templates/spec.md +0 -231
- package/stacks/nextjs-supabase/.morph/templates/supabase-migration.sql +0 -100
- package/stacks/nextjs-supabase/.morph/templates/tasks.md +0 -257
- /package/src/commands/{search-patterns.js → learning/search-patterns.js} +0 -0
- /package/src/{lib → core/templates}/template-data-sources.js +0 -0
- /package/src/lib/{design-system-generator.js → generators/design-system-generator.js} +0 -0
- /package/src/lib/{learning-system.js → learning/learning-system.js} +0 -0
- /package/src/lib/validators/{package-validator.js → packages/package-validator.js} +0 -0
- /package/src/lib/validators/{ui-contrast-validator.js → ui/ui-contrast-validator.js} +0 -0
- /package/{src/generator → stacks/blazor-azure/.morph/templates}/.gitkeep +0 -0
- /package/{src/llm → stacks/nextjs-supabase/.morph/templates}/.gitkeep +0 -0
package/src/lib/validators/{design-system-validator.js → design-system/design-system-validator.js}
RENAMED
|
@@ -1,231 +1,231 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Design System Validator
|
|
3
|
-
*
|
|
4
|
-
* Validates that CSS and Razor files only use design system tokens
|
|
5
|
-
* instead of hardcoded colors, fonts, or spacing values.
|
|
6
|
-
*
|
|
7
|
-
* MORPH-SPEC 3.0 - Phase 3: Design System Enforcement
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
import { existsSync, readFileSync, readdirSync, statSync } from 'fs';
|
|
11
|
-
import { join, extname } from 'path';
|
|
12
|
-
import { parseColors, parseTypography, parseSpacing } from '
|
|
13
|
-
import { detectDesignSystem } from '
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Validate design system compliance
|
|
17
|
-
* @param {string} projectPath - Root path of the project
|
|
18
|
-
* @param {string} featureName - Feature name (optional)
|
|
19
|
-
* @param {Object} options - Validation options
|
|
20
|
-
* @returns {Object} { errors, warnings, issues }
|
|
21
|
-
*/
|
|
22
|
-
export async function validateDesignSystem(projectPath, featureName = null, options = {}) {
|
|
23
|
-
const result = {
|
|
24
|
-
errors: 0,
|
|
25
|
-
warnings: 0,
|
|
26
|
-
issues: []
|
|
27
|
-
};
|
|
28
|
-
|
|
29
|
-
// Detect design system
|
|
30
|
-
const dsDetection = detectDesignSystem(projectPath, featureName);
|
|
31
|
-
|
|
32
|
-
if (!dsDetection.hasDesignSystem) {
|
|
33
|
-
result.errors++;
|
|
34
|
-
result.issues.push({
|
|
35
|
-
level: 'error',
|
|
36
|
-
message: 'No design system found',
|
|
37
|
-
solution: 'Create a design system at .morph/project/design-system.md or run design system generator'
|
|
38
|
-
});
|
|
39
|
-
return result;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
// Parse design system
|
|
43
|
-
let designSystem;
|
|
44
|
-
try {
|
|
45
|
-
const markdown = readFileSync(dsDetection.path, 'utf8');
|
|
46
|
-
designSystem = {
|
|
47
|
-
colors: parseColors(markdown),
|
|
48
|
-
typography: parseTypography(markdown),
|
|
49
|
-
spacing: parseSpacing(markdown)
|
|
50
|
-
};
|
|
51
|
-
} catch (err) {
|
|
52
|
-
result.errors++;
|
|
53
|
-
result.issues.push({
|
|
54
|
-
level: 'error',
|
|
55
|
-
message: `Failed to parse design system: ${err.message}`,
|
|
56
|
-
file: dsDetection.path
|
|
57
|
-
});
|
|
58
|
-
return result;
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
// Build allowed color set
|
|
62
|
-
const allowedColors = new Set();
|
|
63
|
-
Object.values(designSystem.colors.primary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
64
|
-
Object.values(designSystem.colors.secondary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
65
|
-
Object.values(designSystem.colors.neutral).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
66
|
-
Object.values(designSystem.colors.semantic).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
67
|
-
|
|
68
|
-
// Common system colors (allowed)
|
|
69
|
-
const systemColors = new Set([
|
|
70
|
-
'#ffffff', '#fff', '#000000', '#000', 'transparent', 'inherit', 'currentcolor'
|
|
71
|
-
]);
|
|
72
|
-
|
|
73
|
-
// Scan CSS files
|
|
74
|
-
const cssFiles = findFiles(projectPath, ['.css'], options.scanPaths || ['wwwroot/css', 'styles']);
|
|
75
|
-
for (const cssFile of cssFiles) {
|
|
76
|
-
const cssResult = validateCSSFile(cssFile, allowedColors, systemColors);
|
|
77
|
-
result.warnings += cssResult.warnings;
|
|
78
|
-
result.issues.push(...cssResult.issues);
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
// Scan Razor files
|
|
82
|
-
const razorFiles = findFiles(projectPath, ['.razor'], options.scanPaths || ['Components', 'Pages']);
|
|
83
|
-
for (const razorFile of razorFiles) {
|
|
84
|
-
const razorResult = validateRazorFile(razorFile, allowedColors, systemColors);
|
|
85
|
-
result.warnings += razorResult.warnings;
|
|
86
|
-
result.issues.push(...razorResult.issues);
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
return result;
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
/**
|
|
93
|
-
* Find files with specific extensions in directories
|
|
94
|
-
*/
|
|
95
|
-
function findFiles(projectPath, extensions, scanPaths) {
|
|
96
|
-
const files = [];
|
|
97
|
-
|
|
98
|
-
for (const scanPath of scanPaths) {
|
|
99
|
-
const fullPath = join(projectPath, scanPath);
|
|
100
|
-
if (!existsSync(fullPath)) continue;
|
|
101
|
-
|
|
102
|
-
try {
|
|
103
|
-
walkDirectory(fullPath, extensions, files);
|
|
104
|
-
} catch (err) {
|
|
105
|
-
// Ignore permission errors
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
return files;
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
/**
|
|
113
|
-
* Recursively walk directory
|
|
114
|
-
*/
|
|
115
|
-
function walkDirectory(dir, extensions, files) {
|
|
116
|
-
const items = readdirSync(dir);
|
|
117
|
-
|
|
118
|
-
for (const item of items) {
|
|
119
|
-
const fullPath = join(dir, item);
|
|
120
|
-
const stat = statSync(fullPath);
|
|
121
|
-
|
|
122
|
-
if (stat.isDirectory()) {
|
|
123
|
-
walkDirectory(fullPath, extensions, files);
|
|
124
|
-
} else if (stat.isFile()) {
|
|
125
|
-
const ext = extname(fullPath);
|
|
126
|
-
if (extensions.includes(ext)) {
|
|
127
|
-
files.push(fullPath);
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
/**
|
|
134
|
-
* Validate CSS file for hardcoded colors
|
|
135
|
-
*/
|
|
136
|
-
function validateCSSFile(filePath, allowedColors, systemColors) {
|
|
137
|
-
const result = {
|
|
138
|
-
warnings: 0,
|
|
139
|
-
issues: []
|
|
140
|
-
};
|
|
141
|
-
|
|
142
|
-
try {
|
|
143
|
-
const content = readFileSync(filePath, 'utf8');
|
|
144
|
-
|
|
145
|
-
// Match hex colors: #RGB or #RRGGBB
|
|
146
|
-
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
147
|
-
|
|
148
|
-
let match;
|
|
149
|
-
let lineNumber = 1;
|
|
150
|
-
let lastIndex = 0;
|
|
151
|
-
|
|
152
|
-
while ((match = hexRegex.exec(content)) !== null) {
|
|
153
|
-
const hex = match[0].toLowerCase();
|
|
154
|
-
|
|
155
|
-
// Count line number
|
|
156
|
-
const upToMatch = content.substring(lastIndex, match.index);
|
|
157
|
-
lineNumber += (upToMatch.match(/\n/g) || []).length;
|
|
158
|
-
lastIndex = match.index;
|
|
159
|
-
|
|
160
|
-
// Skip if in allowed or system colors
|
|
161
|
-
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
162
|
-
continue;
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
result.warnings++;
|
|
166
|
-
result.issues.push({
|
|
167
|
-
level: 'warning',
|
|
168
|
-
message: `Hardcoded color ${hex} not in design system palette`,
|
|
169
|
-
file: filePath,
|
|
170
|
-
line: lineNumber,
|
|
171
|
-
solution: 'Use CSS variable like var(--color-primary-500) instead'
|
|
172
|
-
});
|
|
173
|
-
}
|
|
174
|
-
} catch (err) {
|
|
175
|
-
// Ignore read errors
|
|
176
|
-
}
|
|
177
|
-
|
|
178
|
-
return result;
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
/**
|
|
182
|
-
* Validate Razor file for inline styles with hardcoded colors
|
|
183
|
-
*/
|
|
184
|
-
function validateRazorFile(filePath, allowedColors, systemColors) {
|
|
185
|
-
const result = {
|
|
186
|
-
warnings: 0,
|
|
187
|
-
issues: []
|
|
188
|
-
};
|
|
189
|
-
|
|
190
|
-
try {
|
|
191
|
-
const content = readFileSync(filePath, 'utf8');
|
|
192
|
-
|
|
193
|
-
// Match style="" attributes with hex colors
|
|
194
|
-
const styleRegex = /style\s*=\s*"([^"]*)"/gi;
|
|
195
|
-
|
|
196
|
-
let match;
|
|
197
|
-
while ((match = styleRegex.exec(content)) !== null) {
|
|
198
|
-
const styleContent = match[1];
|
|
199
|
-
|
|
200
|
-
// Find hex colors in style attribute
|
|
201
|
-
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
202
|
-
let hexMatch;
|
|
203
|
-
|
|
204
|
-
while ((hexMatch = hexRegex.exec(styleContent)) !== null) {
|
|
205
|
-
const hex = hexMatch[0].toLowerCase();
|
|
206
|
-
|
|
207
|
-
// Skip if in allowed or system colors
|
|
208
|
-
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
209
|
-
continue;
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
// Count line number
|
|
213
|
-
const upToMatch = content.substring(0, match.index);
|
|
214
|
-
const lineNumber = (upToMatch.match(/\n/g) || []).length + 1;
|
|
215
|
-
|
|
216
|
-
result.warnings++;
|
|
217
|
-
result.issues.push({
|
|
218
|
-
level: 'warning',
|
|
219
|
-
message: `Inline style with hardcoded color ${hex} not in design system`,
|
|
220
|
-
file: filePath,
|
|
221
|
-
line: lineNumber,
|
|
222
|
-
solution: 'Use CSS class with design system variables or component props instead of inline styles'
|
|
223
|
-
});
|
|
224
|
-
}
|
|
225
|
-
}
|
|
226
|
-
} catch (err) {
|
|
227
|
-
// Ignore read errors
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
return result;
|
|
231
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* Design System Validator
|
|
3
|
+
*
|
|
4
|
+
* Validates that CSS and Razor files only use design system tokens
|
|
5
|
+
* instead of hardcoded colors, fonts, or spacing values.
|
|
6
|
+
*
|
|
7
|
+
* MORPH-SPEC 3.0 - Phase 3: Design System Enforcement
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import { existsSync, readFileSync, readdirSync, statSync } from 'fs';
|
|
11
|
+
import { join, extname } from 'path';
|
|
12
|
+
import { parseColors, parseTypography, parseSpacing } from '../../generators/design-system-generator.js';
|
|
13
|
+
import { detectDesignSystem } from '../../detectors/design-system-detector.js';
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Validate design system compliance
|
|
17
|
+
* @param {string} projectPath - Root path of the project
|
|
18
|
+
* @param {string} featureName - Feature name (optional)
|
|
19
|
+
* @param {Object} options - Validation options
|
|
20
|
+
* @returns {Object} { errors, warnings, issues }
|
|
21
|
+
*/
|
|
22
|
+
export async function validateDesignSystem(projectPath, featureName = null, options = {}) {
|
|
23
|
+
const result = {
|
|
24
|
+
errors: 0,
|
|
25
|
+
warnings: 0,
|
|
26
|
+
issues: []
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
// Detect design system
|
|
30
|
+
const dsDetection = detectDesignSystem(projectPath, featureName);
|
|
31
|
+
|
|
32
|
+
if (!dsDetection.hasDesignSystem) {
|
|
33
|
+
result.errors++;
|
|
34
|
+
result.issues.push({
|
|
35
|
+
level: 'error',
|
|
36
|
+
message: 'No design system found',
|
|
37
|
+
solution: 'Create a design system at .morph/project/design-system.md or run design system generator'
|
|
38
|
+
});
|
|
39
|
+
return result;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Parse design system
|
|
43
|
+
let designSystem;
|
|
44
|
+
try {
|
|
45
|
+
const markdown = readFileSync(dsDetection.path, 'utf8');
|
|
46
|
+
designSystem = {
|
|
47
|
+
colors: parseColors(markdown),
|
|
48
|
+
typography: parseTypography(markdown),
|
|
49
|
+
spacing: parseSpacing(markdown)
|
|
50
|
+
};
|
|
51
|
+
} catch (err) {
|
|
52
|
+
result.errors++;
|
|
53
|
+
result.issues.push({
|
|
54
|
+
level: 'error',
|
|
55
|
+
message: `Failed to parse design system: ${err.message}`,
|
|
56
|
+
file: dsDetection.path
|
|
57
|
+
});
|
|
58
|
+
return result;
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
// Build allowed color set
|
|
62
|
+
const allowedColors = new Set();
|
|
63
|
+
Object.values(designSystem.colors.primary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
64
|
+
Object.values(designSystem.colors.secondary).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
65
|
+
Object.values(designSystem.colors.neutral).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
66
|
+
Object.values(designSystem.colors.semantic).forEach(hex => allowedColors.add(hex.toLowerCase()));
|
|
67
|
+
|
|
68
|
+
// Common system colors (allowed)
|
|
69
|
+
const systemColors = new Set([
|
|
70
|
+
'#ffffff', '#fff', '#000000', '#000', 'transparent', 'inherit', 'currentcolor'
|
|
71
|
+
]);
|
|
72
|
+
|
|
73
|
+
// Scan CSS files
|
|
74
|
+
const cssFiles = findFiles(projectPath, ['.css'], options.scanPaths || ['wwwroot/css', 'styles']);
|
|
75
|
+
for (const cssFile of cssFiles) {
|
|
76
|
+
const cssResult = validateCSSFile(cssFile, allowedColors, systemColors);
|
|
77
|
+
result.warnings += cssResult.warnings;
|
|
78
|
+
result.issues.push(...cssResult.issues);
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// Scan Razor files
|
|
82
|
+
const razorFiles = findFiles(projectPath, ['.razor'], options.scanPaths || ['Components', 'Pages']);
|
|
83
|
+
for (const razorFile of razorFiles) {
|
|
84
|
+
const razorResult = validateRazorFile(razorFile, allowedColors, systemColors);
|
|
85
|
+
result.warnings += razorResult.warnings;
|
|
86
|
+
result.issues.push(...razorResult.issues);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
return result;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Find files with specific extensions in directories
|
|
94
|
+
*/
|
|
95
|
+
function findFiles(projectPath, extensions, scanPaths) {
|
|
96
|
+
const files = [];
|
|
97
|
+
|
|
98
|
+
for (const scanPath of scanPaths) {
|
|
99
|
+
const fullPath = join(projectPath, scanPath);
|
|
100
|
+
if (!existsSync(fullPath)) continue;
|
|
101
|
+
|
|
102
|
+
try {
|
|
103
|
+
walkDirectory(fullPath, extensions, files);
|
|
104
|
+
} catch (err) {
|
|
105
|
+
// Ignore permission errors
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
return files;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Recursively walk directory
|
|
114
|
+
*/
|
|
115
|
+
function walkDirectory(dir, extensions, files) {
|
|
116
|
+
const items = readdirSync(dir);
|
|
117
|
+
|
|
118
|
+
for (const item of items) {
|
|
119
|
+
const fullPath = join(dir, item);
|
|
120
|
+
const stat = statSync(fullPath);
|
|
121
|
+
|
|
122
|
+
if (stat.isDirectory()) {
|
|
123
|
+
walkDirectory(fullPath, extensions, files);
|
|
124
|
+
} else if (stat.isFile()) {
|
|
125
|
+
const ext = extname(fullPath);
|
|
126
|
+
if (extensions.includes(ext)) {
|
|
127
|
+
files.push(fullPath);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
/**
|
|
134
|
+
* Validate CSS file for hardcoded colors
|
|
135
|
+
*/
|
|
136
|
+
function validateCSSFile(filePath, allowedColors, systemColors) {
|
|
137
|
+
const result = {
|
|
138
|
+
warnings: 0,
|
|
139
|
+
issues: []
|
|
140
|
+
};
|
|
141
|
+
|
|
142
|
+
try {
|
|
143
|
+
const content = readFileSync(filePath, 'utf8');
|
|
144
|
+
|
|
145
|
+
// Match hex colors: #RGB or #RRGGBB
|
|
146
|
+
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
147
|
+
|
|
148
|
+
let match;
|
|
149
|
+
let lineNumber = 1;
|
|
150
|
+
let lastIndex = 0;
|
|
151
|
+
|
|
152
|
+
while ((match = hexRegex.exec(content)) !== null) {
|
|
153
|
+
const hex = match[0].toLowerCase();
|
|
154
|
+
|
|
155
|
+
// Count line number
|
|
156
|
+
const upToMatch = content.substring(lastIndex, match.index);
|
|
157
|
+
lineNumber += (upToMatch.match(/\n/g) || []).length;
|
|
158
|
+
lastIndex = match.index;
|
|
159
|
+
|
|
160
|
+
// Skip if in allowed or system colors
|
|
161
|
+
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
162
|
+
continue;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
result.warnings++;
|
|
166
|
+
result.issues.push({
|
|
167
|
+
level: 'warning',
|
|
168
|
+
message: `Hardcoded color ${hex} not in design system palette`,
|
|
169
|
+
file: filePath,
|
|
170
|
+
line: lineNumber,
|
|
171
|
+
solution: 'Use CSS variable like var(--color-primary-500) instead'
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
} catch (err) {
|
|
175
|
+
// Ignore read errors
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
return result;
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* Validate Razor file for inline styles with hardcoded colors
|
|
183
|
+
*/
|
|
184
|
+
function validateRazorFile(filePath, allowedColors, systemColors) {
|
|
185
|
+
const result = {
|
|
186
|
+
warnings: 0,
|
|
187
|
+
issues: []
|
|
188
|
+
};
|
|
189
|
+
|
|
190
|
+
try {
|
|
191
|
+
const content = readFileSync(filePath, 'utf8');
|
|
192
|
+
|
|
193
|
+
// Match style="" attributes with hex colors
|
|
194
|
+
const styleRegex = /style\s*=\s*"([^"]*)"/gi;
|
|
195
|
+
|
|
196
|
+
let match;
|
|
197
|
+
while ((match = styleRegex.exec(content)) !== null) {
|
|
198
|
+
const styleContent = match[1];
|
|
199
|
+
|
|
200
|
+
// Find hex colors in style attribute
|
|
201
|
+
const hexRegex = /#([0-9A-Fa-f]{3}|[0-9A-Fa-f]{6})\b/g;
|
|
202
|
+
let hexMatch;
|
|
203
|
+
|
|
204
|
+
while ((hexMatch = hexRegex.exec(styleContent)) !== null) {
|
|
205
|
+
const hex = hexMatch[0].toLowerCase();
|
|
206
|
+
|
|
207
|
+
// Skip if in allowed or system colors
|
|
208
|
+
if (allowedColors.has(hex) || systemColors.has(hex)) {
|
|
209
|
+
continue;
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
// Count line number
|
|
213
|
+
const upToMatch = content.substring(0, match.index);
|
|
214
|
+
const lineNumber = (upToMatch.match(/\n/g) || []).length + 1;
|
|
215
|
+
|
|
216
|
+
result.warnings++;
|
|
217
|
+
result.issues.push({
|
|
218
|
+
level: 'warning',
|
|
219
|
+
message: `Inline style with hardcoded color ${hex} not in design system`,
|
|
220
|
+
file: filePath,
|
|
221
|
+
line: lineNumber,
|
|
222
|
+
solution: 'Use CSS class with design system variables or component props instead of inline styles'
|
|
223
|
+
});
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
} catch (err) {
|
|
227
|
+
// Ignore read errors
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
return result;
|
|
231
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared validator utilities
|
|
3
|
+
*
|
|
4
|
+
* Common functionality used across multiple validators.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
export { countIssues } from './issue-counter.js';
|
|
8
|
+
export {
|
|
9
|
+
formatValidationSummary,
|
|
10
|
+
formatValidationIssues,
|
|
11
|
+
formatValidationStatus,
|
|
12
|
+
} from './result-formatter.js';
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared issue counting utility for validators
|
|
3
|
+
*
|
|
4
|
+
* Consolidates duplicate implementations from:
|
|
5
|
+
* - blazor-validator.js
|
|
6
|
+
* - blazor-state-validator.js
|
|
7
|
+
* - blazor-concurrency-analyzer.js
|
|
8
|
+
*
|
|
9
|
+
* @param {Array} issues - Array of validation issues with type property
|
|
10
|
+
* @returns {{errors: number, warnings: number, infos: number}} Count by type
|
|
11
|
+
*/
|
|
12
|
+
export function countIssues(issues) {
|
|
13
|
+
return {
|
|
14
|
+
errors: issues.filter((i) => i.type === 'error').length,
|
|
15
|
+
warnings: issues.filter((i) => i.type === 'warning').length,
|
|
16
|
+
infos: issues.filter((i) => i.type === 'info').length,
|
|
17
|
+
};
|
|
18
|
+
}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared validation result formatting utilities
|
|
3
|
+
*
|
|
4
|
+
* Provides common formatting patterns for validation results.
|
|
5
|
+
* Domain-specific formatters (CSS, template, etc.) can extend these.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import chalk from 'chalk';
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Format a generic validation result summary
|
|
12
|
+
*
|
|
13
|
+
* @param {Object} summary - Validation summary object
|
|
14
|
+
* @param {string} title - Title for the results section
|
|
15
|
+
* @returns {string[]} Array of formatted lines
|
|
16
|
+
*/
|
|
17
|
+
export function formatValidationSummary(summary, title = 'Validation Results') {
|
|
18
|
+
const lines = [];
|
|
19
|
+
|
|
20
|
+
lines.push('');
|
|
21
|
+
lines.push(title);
|
|
22
|
+
lines.push('='.repeat(50));
|
|
23
|
+
lines.push('');
|
|
24
|
+
|
|
25
|
+
// Format summary key-value pairs
|
|
26
|
+
for (const [key, value] of Object.entries(summary)) {
|
|
27
|
+
const label = key
|
|
28
|
+
.replace(/([A-Z])/g, ' $1')
|
|
29
|
+
.replace(/^./, str => str.toUpperCase())
|
|
30
|
+
.trim();
|
|
31
|
+
lines.push(`${label}: ${value}`);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
lines.push('');
|
|
35
|
+
|
|
36
|
+
return lines;
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
/**
|
|
40
|
+
* Format validation issues with consistent styling
|
|
41
|
+
*
|
|
42
|
+
* @param {Array} issues - Array of validation issues
|
|
43
|
+
* @param {boolean} verbose - Include detailed output
|
|
44
|
+
* @returns {string[]} Array of formatted issue lines
|
|
45
|
+
*/
|
|
46
|
+
export function formatValidationIssues(issues, verbose = false) {
|
|
47
|
+
const lines = [];
|
|
48
|
+
|
|
49
|
+
if (issues.length === 0) {
|
|
50
|
+
return lines;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// Group by type
|
|
54
|
+
const errors = issues.filter(i => i.type === 'error');
|
|
55
|
+
const warnings = issues.filter(i => i.type === 'warning');
|
|
56
|
+
|
|
57
|
+
if (errors.length > 0) {
|
|
58
|
+
lines.push(chalk.red.bold(`Errors (${errors.length}):`));
|
|
59
|
+
lines.push('-'.repeat(50));
|
|
60
|
+
for (const issue of errors) {
|
|
61
|
+
lines.push(formatIssue(issue, verbose));
|
|
62
|
+
}
|
|
63
|
+
lines.push('');
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
if (warnings.length > 0) {
|
|
67
|
+
lines.push(chalk.yellow.bold(`Warnings (${warnings.length}):`));
|
|
68
|
+
lines.push('-'.repeat(50));
|
|
69
|
+
for (const issue of warnings) {
|
|
70
|
+
lines.push(formatIssue(issue, verbose));
|
|
71
|
+
}
|
|
72
|
+
lines.push('');
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
return lines;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* Format a single validation issue
|
|
80
|
+
*
|
|
81
|
+
* @param {Object} issue - Validation issue
|
|
82
|
+
* @param {boolean} verbose - Include additional details
|
|
83
|
+
* @returns {string} Formatted issue string
|
|
84
|
+
*/
|
|
85
|
+
function formatIssue(issue, verbose = false) {
|
|
86
|
+
const color = issue.type === 'error' ? chalk.red : chalk.yellow;
|
|
87
|
+
const icon = issue.type === 'error' ? '✗' : '⚠';
|
|
88
|
+
|
|
89
|
+
let formatted = ` ${icon} ${issue.message}`;
|
|
90
|
+
|
|
91
|
+
if (issue.file) {
|
|
92
|
+
formatted += chalk.dim(` (${issue.file}`);
|
|
93
|
+
if (issue.line) {
|
|
94
|
+
formatted += `:${issue.line}`;
|
|
95
|
+
}
|
|
96
|
+
formatted += chalk.dim(')');
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
if (verbose && issue.suggestion) {
|
|
100
|
+
formatted += `\n ${chalk.cyan('→')} ${issue.suggestion}`;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
return formatted;
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
/**
|
|
107
|
+
* Format final validation status
|
|
108
|
+
*
|
|
109
|
+
* @param {boolean} valid - Whether validation passed
|
|
110
|
+
* @param {number} errorCount - Number of errors
|
|
111
|
+
* @param {number} warningCount - Number of warnings
|
|
112
|
+
* @returns {string} Status message with appropriate styling
|
|
113
|
+
*/
|
|
114
|
+
export function formatValidationStatus(valid, errorCount = 0, warningCount = 0) {
|
|
115
|
+
if (valid && errorCount === 0 && warningCount === 0) {
|
|
116
|
+
return chalk.green('✓ All validations passed');
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
if (errorCount > 0) {
|
|
120
|
+
return chalk.red(`✗ Validation failed with ${errorCount} error(s) and ${warningCount} warning(s)`);
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
return chalk.yellow(`⚠ Validation passed with ${warningCount} warning(s)`);
|
|
124
|
+
}
|