@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
|
@@ -1,351 +1,351 @@
|
|
|
1
|
-
import { readFileSync, existsSync } from 'fs';
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Content Validator - Validates structure and content of output files
|
|
5
|
-
*
|
|
6
|
-
* Ensures spec.md, tasks.json, and other outputs have required sections
|
|
7
|
-
* and proper structure before allowing phase transitions.
|
|
8
|
-
*/
|
|
9
|
-
|
|
10
|
-
/**
|
|
11
|
-
* Validate spec.md structure
|
|
12
|
-
* @param {string} specPath - Path to spec.md file
|
|
13
|
-
* @returns {Object} Validation result
|
|
14
|
-
*/
|
|
15
|
-
export function validateSpecContent(specPath) {
|
|
16
|
-
if (!existsSync(specPath)) {
|
|
17
|
-
return {
|
|
18
|
-
valid: false,
|
|
19
|
-
missing: ['File does not exist'],
|
|
20
|
-
errors: ['Spec file not found at: ' + specPath]
|
|
21
|
-
};
|
|
22
|
-
}
|
|
23
|
-
|
|
24
|
-
const content = readFileSync(specPath, 'utf8');
|
|
25
|
-
|
|
26
|
-
// Required sections for a complete spec
|
|
27
|
-
const requiredSections = [
|
|
28
|
-
'## Overview',
|
|
29
|
-
'## Requirements',
|
|
30
|
-
'## Technical Design',
|
|
31
|
-
'## Data Model',
|
|
32
|
-
'## API Contracts'
|
|
33
|
-
];
|
|
34
|
-
|
|
35
|
-
const missing = requiredSections.filter(section => !content.includes(section));
|
|
36
|
-
|
|
37
|
-
// Additional quality checks
|
|
38
|
-
const errors = [];
|
|
39
|
-
const warnings = [];
|
|
40
|
-
|
|
41
|
-
// Check minimum content length
|
|
42
|
-
if (content.length < 500) {
|
|
43
|
-
warnings.push('Spec seems very short (< 500 characters). Consider adding more detail.');
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
// Check for placeholder text that should be replaced
|
|
47
|
-
const placeholders = ['TODO', 'TBD', 'PLACEHOLDER', '{{'];
|
|
48
|
-
placeholders.forEach(placeholder => {
|
|
49
|
-
if (content.includes(placeholder)) {
|
|
50
|
-
warnings.push(`Spec contains "${placeholder}" - ensure all placeholders are replaced`);
|
|
51
|
-
}
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
// Check for anti-patterns mentioned in the plan
|
|
55
|
-
if (content.toLowerCase().includes('manual azure portal')) {
|
|
56
|
-
errors.push(
|
|
57
|
-
'Spec mentions manual Azure portal creation. ' +
|
|
58
|
-
'All infrastructure must be defined in Bicep (Infrastructure as Code).'
|
|
59
|
-
);
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
if (content.toLowerCase().includes('create resource manually')) {
|
|
63
|
-
errors.push(
|
|
64
|
-
'Spec mentions manual resource creation. ' +
|
|
65
|
-
'Use IaC (Bicep) for all infrastructure resources.'
|
|
66
|
-
);
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
return {
|
|
70
|
-
valid: missing.length === 0 && errors.length === 0,
|
|
71
|
-
missing,
|
|
72
|
-
errors,
|
|
73
|
-
warnings,
|
|
74
|
-
sections: {
|
|
75
|
-
required: requiredSections.length,
|
|
76
|
-
found: requiredSections.length - missing.length
|
|
77
|
-
}
|
|
78
|
-
};
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
/**
|
|
82
|
-
* Validate tasks.json structure
|
|
83
|
-
* @param {string} tasksPath - Path to tasks.json file
|
|
84
|
-
* @returns {Object} Validation result
|
|
85
|
-
*/
|
|
86
|
-
export function validateTasksContent(tasksPath) {
|
|
87
|
-
if (!existsSync(tasksPath)) {
|
|
88
|
-
return {
|
|
89
|
-
valid: false,
|
|
90
|
-
errors: ['Tasks file not found at: ' + tasksPath]
|
|
91
|
-
};
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
let tasks;
|
|
95
|
-
try {
|
|
96
|
-
const content = readFileSync(tasksPath, 'utf8');
|
|
97
|
-
tasks = JSON.parse(content);
|
|
98
|
-
} catch (error) {
|
|
99
|
-
return {
|
|
100
|
-
valid: false,
|
|
101
|
-
errors: ['Invalid JSON in tasks file: ' + error.message]
|
|
102
|
-
};
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
const errors = [];
|
|
106
|
-
const warnings = [];
|
|
107
|
-
|
|
108
|
-
// Check required top-level fields
|
|
109
|
-
if (!tasks.feature) {
|
|
110
|
-
errors.push('Missing "feature" field in tasks.json');
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (!tasks.tasks || !Array.isArray(tasks.tasks)) {
|
|
114
|
-
errors.push('Missing or invalid "tasks" array in tasks.json');
|
|
115
|
-
return { valid: false, errors, warnings };
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
if (tasks.tasks.length === 0) {
|
|
119
|
-
errors.push('Tasks array is empty - no tasks defined');
|
|
120
|
-
return { valid: false, errors, warnings };
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
// Validate individual tasks
|
|
124
|
-
tasks.tasks.forEach((task, index) => {
|
|
125
|
-
const taskId = task.id || `Task ${index}`;
|
|
126
|
-
|
|
127
|
-
// Check required fields
|
|
128
|
-
if (!task.id) {
|
|
129
|
-
errors.push(`${taskId}: Missing "id" field`);
|
|
130
|
-
} else if (!/^(T\d{3}|CHECKPOINT_\d{3})$/.test(task.id)) {
|
|
131
|
-
warnings.push(`${taskId}: ID should follow format T### or CHECKPOINT_###`);
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
if (!task.title) {
|
|
135
|
-
errors.push(`${taskId}: Missing "title" field`);
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
if (!task.description) {
|
|
139
|
-
errors.push(`${taskId}: Missing "description" field`);
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
if (!task.dependencies) {
|
|
143
|
-
errors.push(`${taskId}: Missing "dependencies" field (use empty array if no deps)`);
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
// For regular tasks (not checkpoints)
|
|
147
|
-
if (task.id && task.id.startsWith('T')) {
|
|
148
|
-
if (!task.category) {
|
|
149
|
-
warnings.push(`${taskId}: Missing "category" field`);
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
if (!task.estimatedMinutes) {
|
|
153
|
-
warnings.push(`${taskId}: Missing "estimatedMinutes" field`);
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
if (!task.files || task.files.length === 0) {
|
|
157
|
-
warnings.push(`${taskId}: No files specified - consider adding affected files`);
|
|
158
|
-
}
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
// For checkpoints
|
|
162
|
-
if (task.id && task.id.startsWith('CHECKPOINT')) {
|
|
163
|
-
if (!task.afterTasks || task.afterTasks.length === 0) {
|
|
164
|
-
warnings.push(`${taskId}: Checkpoint should specify "afterTasks"`);
|
|
165
|
-
}
|
|
166
|
-
|
|
167
|
-
if (!task.validations || task.validations.length === 0) {
|
|
168
|
-
warnings.push(`${taskId}: Checkpoint should specify "validations"`);
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
});
|
|
172
|
-
|
|
173
|
-
// Check for orphaned tasks (missing dependencies)
|
|
174
|
-
const taskIds = new Set(tasks.tasks.map(t => t.id));
|
|
175
|
-
tasks.tasks.forEach(task => {
|
|
176
|
-
if (task.dependencies && Array.isArray(task.dependencies)) {
|
|
177
|
-
task.dependencies.forEach(depId => {
|
|
178
|
-
if (depId && !taskIds.has(depId)) {
|
|
179
|
-
errors.push(`${task.id}: References non-existent dependency "${depId}"`);
|
|
180
|
-
}
|
|
181
|
-
});
|
|
182
|
-
}
|
|
183
|
-
});
|
|
184
|
-
|
|
185
|
-
// Check for circular dependencies (simple check)
|
|
186
|
-
const hasCycle = (taskId, visited = new Set()) => {
|
|
187
|
-
if (visited.has(taskId)) return true;
|
|
188
|
-
visited.add(taskId);
|
|
189
|
-
|
|
190
|
-
const task = tasks.tasks.find(t => t.id === taskId);
|
|
191
|
-
if (!task || !task.dependencies) return false;
|
|
192
|
-
|
|
193
|
-
return task.dependencies.some(depId => hasCycle(depId, new Set(visited)));
|
|
194
|
-
};
|
|
195
|
-
|
|
196
|
-
tasks.tasks.forEach(task => {
|
|
197
|
-
if (task.id && hasCycle(task.id)) {
|
|
198
|
-
errors.push(`Circular dependency detected involving task ${task.id}`);
|
|
199
|
-
}
|
|
200
|
-
});
|
|
201
|
-
|
|
202
|
-
return {
|
|
203
|
-
valid: errors.length === 0,
|
|
204
|
-
errors,
|
|
205
|
-
warnings,
|
|
206
|
-
stats: {
|
|
207
|
-
totalTasks: tasks.tasks.length,
|
|
208
|
-
regularTasks: tasks.tasks.filter(t => t.id?.startsWith('T')).length,
|
|
209
|
-
checkpoints: tasks.tasks.filter(t => t.id?.startsWith('CHECKPOINT')).length
|
|
210
|
-
}
|
|
211
|
-
};
|
|
212
|
-
}
|
|
213
|
-
|
|
214
|
-
/**
|
|
215
|
-
* Validate contracts file structure (C# contracts)
|
|
216
|
-
* @param {string} contractsPath - Path to contracts file
|
|
217
|
-
* @returns {Object} Validation result
|
|
218
|
-
*/
|
|
219
|
-
export function validateContractsContent(contractsPath) {
|
|
220
|
-
if (!existsSync(contractsPath)) {
|
|
221
|
-
return {
|
|
222
|
-
valid: false,
|
|
223
|
-
errors: ['Contracts file not found at: ' + contractsPath]
|
|
224
|
-
};
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
const content = readFileSync(contractsPath, 'utf8');
|
|
228
|
-
const errors = [];
|
|
229
|
-
const warnings = [];
|
|
230
|
-
|
|
231
|
-
// Check for basic C# structure
|
|
232
|
-
if (!content.includes('namespace')) {
|
|
233
|
-
errors.push('Contracts file missing namespace declaration');
|
|
234
|
-
}
|
|
235
|
-
|
|
236
|
-
// Check for at least one interface or record
|
|
237
|
-
const hasInterface = /interface\s+\w+/.test(content);
|
|
238
|
-
const hasRecord = /record\s+\w+/.test(content);
|
|
239
|
-
const hasClass = /class\s+\w+/.test(content);
|
|
240
|
-
|
|
241
|
-
if (!hasInterface && !hasRecord && !hasClass) {
|
|
242
|
-
errors.push('Contracts file should define at least one interface, record, or class');
|
|
243
|
-
}
|
|
244
|
-
|
|
245
|
-
// Check for common anti-patterns
|
|
246
|
-
if (content.includes('// TODO') || content.includes('// PLACEHOLDER')) {
|
|
247
|
-
warnings.push('Contracts contain TODO/PLACEHOLDER comments - ensure they are completed');
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
return {
|
|
251
|
-
valid: errors.length === 0,
|
|
252
|
-
errors,
|
|
253
|
-
warnings,
|
|
254
|
-
found: {
|
|
255
|
-
hasNamespace: content.includes('namespace'),
|
|
256
|
-
hasInterfaces: hasInterface,
|
|
257
|
-
hasRecords: hasRecord,
|
|
258
|
-
hasClasses: hasClass
|
|
259
|
-
}
|
|
260
|
-
};
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
/**
|
|
264
|
-
* Validate proposal.md structure
|
|
265
|
-
* @param {string} proposalPath - Path to proposal.md file
|
|
266
|
-
* @returns {Object} Validation result
|
|
267
|
-
*/
|
|
268
|
-
export function validateProposalContent(proposalPath) {
|
|
269
|
-
if (!existsSync(proposalPath)) {
|
|
270
|
-
return {
|
|
271
|
-
valid: false,
|
|
272
|
-
errors: ['Proposal file not found at: ' + proposalPath]
|
|
273
|
-
};
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
const content = readFileSync(proposalPath, 'utf8');
|
|
277
|
-
const errors = [];
|
|
278
|
-
const warnings = [];
|
|
279
|
-
|
|
280
|
-
// Required sections
|
|
281
|
-
const requiredSections = [
|
|
282
|
-
'## Problem Statement',
|
|
283
|
-
'## Proposed Solution'
|
|
284
|
-
];
|
|
285
|
-
|
|
286
|
-
const missing = requiredSections.filter(section => !content.includes(section));
|
|
287
|
-
missing.forEach(section => errors.push(`Missing required section: ${section}`));
|
|
288
|
-
|
|
289
|
-
// Check for minimum content
|
|
290
|
-
if (content.length < 300) {
|
|
291
|
-
warnings.push('Proposal seems very brief (< 300 characters)');
|
|
292
|
-
}
|
|
293
|
-
|
|
294
|
-
return {
|
|
295
|
-
valid: errors.length === 0,
|
|
296
|
-
errors,
|
|
297
|
-
warnings,
|
|
298
|
-
sections: {
|
|
299
|
-
required: requiredSections.length,
|
|
300
|
-
found: requiredSections.length - missing.length
|
|
301
|
-
}
|
|
302
|
-
};
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
/**
|
|
306
|
-
* Validate all outputs for a feature comprehensively
|
|
307
|
-
* @param {Object} featureState - Feature state object
|
|
308
|
-
* @param {string} targetPhase - Target phase for validation
|
|
309
|
-
* @returns {Object} Comprehensive validation result
|
|
310
|
-
*/
|
|
311
|
-
export function validateFeatureOutputs(featureState, targetPhase) {
|
|
312
|
-
const results = {
|
|
313
|
-
valid: true,
|
|
314
|
-
validations: []
|
|
315
|
-
};
|
|
316
|
-
|
|
317
|
-
// Validate based on target phase
|
|
318
|
-
if (targetPhase === 'clarify' || targetPhase === 'tasks' || targetPhase === 'implement') {
|
|
319
|
-
// Spec is required
|
|
320
|
-
if (featureState.outputs?.spec?.created) {
|
|
321
|
-
const specResult = validateSpecContent(featureState.outputs.spec.path);
|
|
322
|
-
results.validations.push({ type: 'spec', ...specResult });
|
|
323
|
-
if (!specResult.valid) results.valid = false;
|
|
324
|
-
} else {
|
|
325
|
-
results.valid = false;
|
|
326
|
-
results.validations.push({
|
|
327
|
-
type: 'spec',
|
|
328
|
-
valid: false,
|
|
329
|
-
errors: ['Spec file not created']
|
|
330
|
-
});
|
|
331
|
-
}
|
|
332
|
-
}
|
|
333
|
-
|
|
334
|
-
if (targetPhase === 'implement') {
|
|
335
|
-
// Tasks are required
|
|
336
|
-
if (featureState.outputs?.tasks?.created) {
|
|
337
|
-
const tasksResult = validateTasksContent(featureState.outputs.tasks.path);
|
|
338
|
-
results.validations.push({ type: 'tasks', ...tasksResult });
|
|
339
|
-
if (!tasksResult.valid) results.valid = false;
|
|
340
|
-
} else {
|
|
341
|
-
results.valid = false;
|
|
342
|
-
results.validations.push({
|
|
343
|
-
type: 'tasks',
|
|
344
|
-
valid: false,
|
|
345
|
-
errors: ['Tasks file not created']
|
|
346
|
-
});
|
|
347
|
-
}
|
|
348
|
-
}
|
|
349
|
-
|
|
350
|
-
return results;
|
|
351
|
-
}
|
|
1
|
+
import { readFileSync, existsSync } from 'fs';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Content Validator - Validates structure and content of output files
|
|
5
|
+
*
|
|
6
|
+
* Ensures spec.md, tasks.json, and other outputs have required sections
|
|
7
|
+
* and proper structure before allowing phase transitions.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Validate spec.md structure
|
|
12
|
+
* @param {string} specPath - Path to spec.md file
|
|
13
|
+
* @returns {Object} Validation result
|
|
14
|
+
*/
|
|
15
|
+
export function validateSpecContent(specPath) {
|
|
16
|
+
if (!existsSync(specPath)) {
|
|
17
|
+
return {
|
|
18
|
+
valid: false,
|
|
19
|
+
missing: ['File does not exist'],
|
|
20
|
+
errors: ['Spec file not found at: ' + specPath]
|
|
21
|
+
};
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
const content = readFileSync(specPath, 'utf8');
|
|
25
|
+
|
|
26
|
+
// Required sections for a complete spec
|
|
27
|
+
const requiredSections = [
|
|
28
|
+
'## Overview',
|
|
29
|
+
'## Requirements',
|
|
30
|
+
'## Technical Design',
|
|
31
|
+
'## Data Model',
|
|
32
|
+
'## API Contracts'
|
|
33
|
+
];
|
|
34
|
+
|
|
35
|
+
const missing = requiredSections.filter(section => !content.includes(section));
|
|
36
|
+
|
|
37
|
+
// Additional quality checks
|
|
38
|
+
const errors = [];
|
|
39
|
+
const warnings = [];
|
|
40
|
+
|
|
41
|
+
// Check minimum content length
|
|
42
|
+
if (content.length < 500) {
|
|
43
|
+
warnings.push('Spec seems very short (< 500 characters). Consider adding more detail.');
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
// Check for placeholder text that should be replaced
|
|
47
|
+
const placeholders = ['TODO', 'TBD', 'PLACEHOLDER', '{{'];
|
|
48
|
+
placeholders.forEach(placeholder => {
|
|
49
|
+
if (content.includes(placeholder)) {
|
|
50
|
+
warnings.push(`Spec contains "${placeholder}" - ensure all placeholders are replaced`);
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
// Check for anti-patterns mentioned in the plan
|
|
55
|
+
if (content.toLowerCase().includes('manual azure portal')) {
|
|
56
|
+
errors.push(
|
|
57
|
+
'Spec mentions manual Azure portal creation. ' +
|
|
58
|
+
'All infrastructure must be defined in Bicep (Infrastructure as Code).'
|
|
59
|
+
);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
if (content.toLowerCase().includes('create resource manually')) {
|
|
63
|
+
errors.push(
|
|
64
|
+
'Spec mentions manual resource creation. ' +
|
|
65
|
+
'Use IaC (Bicep) for all infrastructure resources.'
|
|
66
|
+
);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
return {
|
|
70
|
+
valid: missing.length === 0 && errors.length === 0,
|
|
71
|
+
missing,
|
|
72
|
+
errors,
|
|
73
|
+
warnings,
|
|
74
|
+
sections: {
|
|
75
|
+
required: requiredSections.length,
|
|
76
|
+
found: requiredSections.length - missing.length
|
|
77
|
+
}
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Validate tasks.json structure
|
|
83
|
+
* @param {string} tasksPath - Path to tasks.json file
|
|
84
|
+
* @returns {Object} Validation result
|
|
85
|
+
*/
|
|
86
|
+
export function validateTasksContent(tasksPath) {
|
|
87
|
+
if (!existsSync(tasksPath)) {
|
|
88
|
+
return {
|
|
89
|
+
valid: false,
|
|
90
|
+
errors: ['Tasks file not found at: ' + tasksPath]
|
|
91
|
+
};
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
let tasks;
|
|
95
|
+
try {
|
|
96
|
+
const content = readFileSync(tasksPath, 'utf8');
|
|
97
|
+
tasks = JSON.parse(content);
|
|
98
|
+
} catch (error) {
|
|
99
|
+
return {
|
|
100
|
+
valid: false,
|
|
101
|
+
errors: ['Invalid JSON in tasks file: ' + error.message]
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
const errors = [];
|
|
106
|
+
const warnings = [];
|
|
107
|
+
|
|
108
|
+
// Check required top-level fields
|
|
109
|
+
if (!tasks.feature) {
|
|
110
|
+
errors.push('Missing "feature" field in tasks.json');
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
if (!tasks.tasks || !Array.isArray(tasks.tasks)) {
|
|
114
|
+
errors.push('Missing or invalid "tasks" array in tasks.json');
|
|
115
|
+
return { valid: false, errors, warnings };
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
if (tasks.tasks.length === 0) {
|
|
119
|
+
errors.push('Tasks array is empty - no tasks defined');
|
|
120
|
+
return { valid: false, errors, warnings };
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
// Validate individual tasks
|
|
124
|
+
tasks.tasks.forEach((task, index) => {
|
|
125
|
+
const taskId = task.id || `Task ${index}`;
|
|
126
|
+
|
|
127
|
+
// Check required fields
|
|
128
|
+
if (!task.id) {
|
|
129
|
+
errors.push(`${taskId}: Missing "id" field`);
|
|
130
|
+
} else if (!/^(T\d{3}|CHECKPOINT_\d{3})$/.test(task.id)) {
|
|
131
|
+
warnings.push(`${taskId}: ID should follow format T### or CHECKPOINT_###`);
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (!task.title) {
|
|
135
|
+
errors.push(`${taskId}: Missing "title" field`);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
if (!task.description) {
|
|
139
|
+
errors.push(`${taskId}: Missing "description" field`);
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
if (!task.dependencies) {
|
|
143
|
+
errors.push(`${taskId}: Missing "dependencies" field (use empty array if no deps)`);
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
// For regular tasks (not checkpoints)
|
|
147
|
+
if (task.id && task.id.startsWith('T')) {
|
|
148
|
+
if (!task.category) {
|
|
149
|
+
warnings.push(`${taskId}: Missing "category" field`);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
if (!task.estimatedMinutes) {
|
|
153
|
+
warnings.push(`${taskId}: Missing "estimatedMinutes" field`);
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
if (!task.files || task.files.length === 0) {
|
|
157
|
+
warnings.push(`${taskId}: No files specified - consider adding affected files`);
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
// For checkpoints
|
|
162
|
+
if (task.id && task.id.startsWith('CHECKPOINT')) {
|
|
163
|
+
if (!task.afterTasks || task.afterTasks.length === 0) {
|
|
164
|
+
warnings.push(`${taskId}: Checkpoint should specify "afterTasks"`);
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
if (!task.validations || task.validations.length === 0) {
|
|
168
|
+
warnings.push(`${taskId}: Checkpoint should specify "validations"`);
|
|
169
|
+
}
|
|
170
|
+
}
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
// Check for orphaned tasks (missing dependencies)
|
|
174
|
+
const taskIds = new Set(tasks.tasks.map(t => t.id));
|
|
175
|
+
tasks.tasks.forEach(task => {
|
|
176
|
+
if (task.dependencies && Array.isArray(task.dependencies)) {
|
|
177
|
+
task.dependencies.forEach(depId => {
|
|
178
|
+
if (depId && !taskIds.has(depId)) {
|
|
179
|
+
errors.push(`${task.id}: References non-existent dependency "${depId}"`);
|
|
180
|
+
}
|
|
181
|
+
});
|
|
182
|
+
}
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
// Check for circular dependencies (simple check)
|
|
186
|
+
const hasCycle = (taskId, visited = new Set()) => {
|
|
187
|
+
if (visited.has(taskId)) return true;
|
|
188
|
+
visited.add(taskId);
|
|
189
|
+
|
|
190
|
+
const task = tasks.tasks.find(t => t.id === taskId);
|
|
191
|
+
if (!task || !task.dependencies) return false;
|
|
192
|
+
|
|
193
|
+
return task.dependencies.some(depId => hasCycle(depId, new Set(visited)));
|
|
194
|
+
};
|
|
195
|
+
|
|
196
|
+
tasks.tasks.forEach(task => {
|
|
197
|
+
if (task.id && hasCycle(task.id)) {
|
|
198
|
+
errors.push(`Circular dependency detected involving task ${task.id}`);
|
|
199
|
+
}
|
|
200
|
+
});
|
|
201
|
+
|
|
202
|
+
return {
|
|
203
|
+
valid: errors.length === 0,
|
|
204
|
+
errors,
|
|
205
|
+
warnings,
|
|
206
|
+
stats: {
|
|
207
|
+
totalTasks: tasks.tasks.length,
|
|
208
|
+
regularTasks: tasks.tasks.filter(t => t.id?.startsWith('T')).length,
|
|
209
|
+
checkpoints: tasks.tasks.filter(t => t.id?.startsWith('CHECKPOINT')).length
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
/**
|
|
215
|
+
* Validate contracts file structure (C# contracts)
|
|
216
|
+
* @param {string} contractsPath - Path to contracts file
|
|
217
|
+
* @returns {Object} Validation result
|
|
218
|
+
*/
|
|
219
|
+
export function validateContractsContent(contractsPath) {
|
|
220
|
+
if (!existsSync(contractsPath)) {
|
|
221
|
+
return {
|
|
222
|
+
valid: false,
|
|
223
|
+
errors: ['Contracts file not found at: ' + contractsPath]
|
|
224
|
+
};
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
const content = readFileSync(contractsPath, 'utf8');
|
|
228
|
+
const errors = [];
|
|
229
|
+
const warnings = [];
|
|
230
|
+
|
|
231
|
+
// Check for basic C# structure
|
|
232
|
+
if (!content.includes('namespace')) {
|
|
233
|
+
errors.push('Contracts file missing namespace declaration');
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
// Check for at least one interface or record
|
|
237
|
+
const hasInterface = /interface\s+\w+/.test(content);
|
|
238
|
+
const hasRecord = /record\s+\w+/.test(content);
|
|
239
|
+
const hasClass = /class\s+\w+/.test(content);
|
|
240
|
+
|
|
241
|
+
if (!hasInterface && !hasRecord && !hasClass) {
|
|
242
|
+
errors.push('Contracts file should define at least one interface, record, or class');
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// Check for common anti-patterns
|
|
246
|
+
if (content.includes('// TODO') || content.includes('// PLACEHOLDER')) {
|
|
247
|
+
warnings.push('Contracts contain TODO/PLACEHOLDER comments - ensure they are completed');
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
return {
|
|
251
|
+
valid: errors.length === 0,
|
|
252
|
+
errors,
|
|
253
|
+
warnings,
|
|
254
|
+
found: {
|
|
255
|
+
hasNamespace: content.includes('namespace'),
|
|
256
|
+
hasInterfaces: hasInterface,
|
|
257
|
+
hasRecords: hasRecord,
|
|
258
|
+
hasClasses: hasClass
|
|
259
|
+
}
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
/**
|
|
264
|
+
* Validate proposal.md structure
|
|
265
|
+
* @param {string} proposalPath - Path to proposal.md file
|
|
266
|
+
* @returns {Object} Validation result
|
|
267
|
+
*/
|
|
268
|
+
export function validateProposalContent(proposalPath) {
|
|
269
|
+
if (!existsSync(proposalPath)) {
|
|
270
|
+
return {
|
|
271
|
+
valid: false,
|
|
272
|
+
errors: ['Proposal file not found at: ' + proposalPath]
|
|
273
|
+
};
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
const content = readFileSync(proposalPath, 'utf8');
|
|
277
|
+
const errors = [];
|
|
278
|
+
const warnings = [];
|
|
279
|
+
|
|
280
|
+
// Required sections
|
|
281
|
+
const requiredSections = [
|
|
282
|
+
'## Problem Statement',
|
|
283
|
+
'## Proposed Solution'
|
|
284
|
+
];
|
|
285
|
+
|
|
286
|
+
const missing = requiredSections.filter(section => !content.includes(section));
|
|
287
|
+
missing.forEach(section => errors.push(`Missing required section: ${section}`));
|
|
288
|
+
|
|
289
|
+
// Check for minimum content
|
|
290
|
+
if (content.length < 300) {
|
|
291
|
+
warnings.push('Proposal seems very brief (< 300 characters)');
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
return {
|
|
295
|
+
valid: errors.length === 0,
|
|
296
|
+
errors,
|
|
297
|
+
warnings,
|
|
298
|
+
sections: {
|
|
299
|
+
required: requiredSections.length,
|
|
300
|
+
found: requiredSections.length - missing.length
|
|
301
|
+
}
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
/**
|
|
306
|
+
* Validate all outputs for a feature comprehensively
|
|
307
|
+
* @param {Object} featureState - Feature state object
|
|
308
|
+
* @param {string} targetPhase - Target phase for validation
|
|
309
|
+
* @returns {Object} Comprehensive validation result
|
|
310
|
+
*/
|
|
311
|
+
export function validateFeatureOutputs(featureState, targetPhase) {
|
|
312
|
+
const results = {
|
|
313
|
+
valid: true,
|
|
314
|
+
validations: []
|
|
315
|
+
};
|
|
316
|
+
|
|
317
|
+
// Validate based on target phase
|
|
318
|
+
if (targetPhase === 'clarify' || targetPhase === 'tasks' || targetPhase === 'implement') {
|
|
319
|
+
// Spec is required
|
|
320
|
+
if (featureState.outputs?.spec?.created) {
|
|
321
|
+
const specResult = validateSpecContent(featureState.outputs.spec.path);
|
|
322
|
+
results.validations.push({ type: 'spec', ...specResult });
|
|
323
|
+
if (!specResult.valid) results.valid = false;
|
|
324
|
+
} else {
|
|
325
|
+
results.valid = false;
|
|
326
|
+
results.validations.push({
|
|
327
|
+
type: 'spec',
|
|
328
|
+
valid: false,
|
|
329
|
+
errors: ['Spec file not created']
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
if (targetPhase === 'implement') {
|
|
335
|
+
// Tasks are required
|
|
336
|
+
if (featureState.outputs?.tasks?.created) {
|
|
337
|
+
const tasksResult = validateTasksContent(featureState.outputs.tasks.path);
|
|
338
|
+
results.validations.push({ type: 'tasks', ...tasksResult });
|
|
339
|
+
if (!tasksResult.valid) results.valid = false;
|
|
340
|
+
} else {
|
|
341
|
+
results.valid = false;
|
|
342
|
+
results.validations.push({
|
|
343
|
+
type: 'tasks',
|
|
344
|
+
valid: false,
|
|
345
|
+
errors: ['Tasks file not created']
|
|
346
|
+
});
|
|
347
|
+
}
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
return results;
|
|
351
|
+
}
|