@polymorphism-tech/morph-spec 3.1.0 → 4.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CLAUDE.md +882 -3
- package/README.md +79 -18
- package/bin/detect-agents.js +1 -1
- package/bin/morph-spec.js +163 -26
- package/bin/task-manager.cjs +101 -7
- package/bin/validate.js +1 -1
- package/docs/cli-auto-detection.md +219 -0
- package/docs/getting-started.md +0 -5
- package/docs/llm-interaction-config.md +735 -0
- package/docs/troubleshooting.md +269 -0
- package/docs/v3.0/AGENTS.md +521 -0
- package/docs/v3.0/ANALYSIS.md +555 -0
- package/docs/v3.0/ARCHITECTURE.md +436 -0
- package/docs/v3.0/EXECUTION-FLOW.md +1304 -0
- package/docs/v3.0/FEATURES.md +688 -0
- package/docs/v3.0/README.md +231 -0
- package/docs/v3.0/ROADMAP.md +801 -0
- package/docs/validation-checklist.md +0 -1
- package/package.json +5 -1
- package/src/commands/agents/index.js +4 -0
- package/src/commands/agents/spawn-team.js +172 -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} +130 -3
- package/src/commands/generation/index.js +5 -0
- package/src/commands/index.js +16 -0
- package/src/commands/learning/capture-pattern.js +121 -0
- package/src/commands/learning/index.js +5 -0
- package/src/commands/learning/search-patterns.js +126 -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} +356 -356
- package/src/commands/project/index.js +10 -0
- package/src/commands/{init.js → project/init.js} +305 -258
- package/src/commands/{sync.js → project/sync.js} +167 -167
- package/src/commands/{update.js → project/update.js} +240 -204
- package/src/commands/{advance-phase.js → state/advance-phase.js} +416 -266
- package/src/commands/state/approve.js +221 -0
- 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 +95 -0
- package/src/commands/templates/template-show.js +131 -0
- package/src/commands/templates/template-validate.js +91 -0
- package/src/commands/utils/index.js +7 -0
- package/src/commands/utils/migrate-state.js +158 -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/utils/upgrade.js +346 -0
- 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/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/core/state/phase-state-machine.js +214 -0
- package/src/{lib → core/state}/state-manager.js +572 -414
- package/src/core/templates/index.js +9 -0
- package/src/core/templates/template-data-sources.js +325 -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 +354 -0
- package/src/generator/config-generator.js +206 -0
- package/src/generator/templates/config.json.template +40 -0
- package/src/generator/templates/project.md.template +67 -0
- package/src/lib/{complexity-analyzer.js → analysis/complexity-analyzer.js} +441 -441
- package/src/lib/analysis/index.js +7 -0
- package/src/lib/checkpoints/checkpoint-hooks.js +258 -0
- package/src/lib/checkpoints/index.js +7 -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/{context-generator.js → generators/context-generator.js} +526 -516
- package/src/lib/generators/index.js +10 -0
- package/src/lib/generators/metadata-extractor.js +387 -0
- package/src/lib/{recap-generator.js → generators/recap-generator.js} +205 -205
- 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/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/validators/architecture/architecture-validator.js +387 -0
- package/src/lib/validators/architecture/index.js +7 -0
- package/src/lib/validators/architecture-validator.js +40 -367
- 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/content-validator.js +351 -0
- package/src/lib/validators/content/index.js +7 -0
- package/src/lib/validators/content-validator.js +164 -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/package-validator.js +41 -340
- package/src/lib/validators/packages/index.js +7 -0
- package/src/lib/validators/packages/package-validator.js +360 -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/validators/ui/ui-contrast-validator.js +422 -0
- package/src/lib/validators/ui-contrast-validator.js +31 -409
- package/src/lib/{validation-runner.js → validators/validation-runner.js} +286 -284
- package/src/llm/analyzer.js +215 -0
- package/src/llm/environment-detector.js +43 -0
- package/src/llm/few-shot-examples.js +216 -0
- package/src/llm/project-config-schema.json +188 -0
- package/src/llm/prompt-builder.js +96 -0
- package/src/orchestrator.js +206 -0
- package/src/sanitizer/context-sanitizer.js +221 -0
- package/src/sanitizer/patterns.js +163 -0
- package/src/scanner/project-scanner.js +242 -0
- package/src/ui/diff-display.js +91 -0
- package/src/ui/interactive-wizard.js +96 -0
- package/src/ui/user-review.js +211 -0
- package/src/ui/wizard-questions.js +188 -0
- 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/src/writer/file-writer.js +86 -0
- package/stacks/blazor-azure/.claude/skills/level-2-domains/ai-agents/ai-system-architect.md +3 -3
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/api-designer.md +59 -0
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/dotnet-senior.md +45 -255
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ef-modeler.md +33 -88
- package/stacks/blazor-azure/.claude/skills/level-2-domains/backend/ms-agent-expert.md +25 -89
- package/stacks/blazor-azure/.claude/skills/level-2-domains/integrations/hangfire-orchestrator.md +64 -0
- package/stacks/blazor-azure/.morph/config/agents.json +879 -764
- package/stacks/blazor-azure/.morph/hooks/{pre-commit-tests.sh → pre-commit/tests-csharp.sh} +3 -2
- package/stacks/blazor-azure/.morph/templates/.gitkeep +0 -0
- 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/.claude/commands/morph-apply.md +221 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-archive.md +79 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-deploy.md +529 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-infra.md +209 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-preflight.md +227 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-proposal.md +122 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-status.md +86 -0
- package/stacks/nextjs-supabase/.claude/commands/morph-troubleshoot.md +122 -0
- package/stacks/nextjs-supabase/.claude/settings.local.json +6 -0
- package/stacks/nextjs-supabase/.claude/skills/level-2-domains/integrations/supabase-expert.md +30 -150
- 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/.gitkeep +0 -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/bin/render-template.js +0 -303
- 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/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/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/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/settings.local.json +0 -15
- 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/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/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/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/lib/{design-system-generator.js → generators/design-system-generator.js} +0 -0
- /package/src/lib/{learning-system.js → learning/learning-system.js} +0 -0
|
@@ -0,0 +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
|
+
}
|
|
@@ -0,0 +1,164 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Content Validator
|
|
3
|
+
*
|
|
4
|
+
* Validates the content and structure of spec.md, tasks.json, and other feature outputs
|
|
5
|
+
*
|
|
6
|
+
* MORPH-SPEC 3.0 - Phase Validation
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import { readFileSync, existsSync } from 'fs';
|
|
10
|
+
|
|
11
|
+
/**
|
|
12
|
+
* Validate spec.md content and structure
|
|
13
|
+
*
|
|
14
|
+
* @param {string} specPath - Path to spec.md file
|
|
15
|
+
* @returns {Object} { valid, errors }
|
|
16
|
+
*/
|
|
17
|
+
export function validateSpecContent(specPath) {
|
|
18
|
+
if (!existsSync(specPath)) {
|
|
19
|
+
return {
|
|
20
|
+
valid: false,
|
|
21
|
+
errors: ['Spec file does not exist']
|
|
22
|
+
};
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const content = readFileSync(specPath, 'utf8');
|
|
26
|
+
const errors = [];
|
|
27
|
+
|
|
28
|
+
// Required sections for spec.md
|
|
29
|
+
const requiredSections = [
|
|
30
|
+
'## Overview',
|
|
31
|
+
'## Functional Requirements',
|
|
32
|
+
'## Technical Architecture',
|
|
33
|
+
'## Data Model'
|
|
34
|
+
];
|
|
35
|
+
|
|
36
|
+
for (const section of requiredSections) {
|
|
37
|
+
if (!content.includes(section) && !content.match(new RegExp(`##\\s+\\d+\\.\\s+${section.replace('## ', '')}`, 'i'))) {
|
|
38
|
+
errors.push(`Missing required section: ${section}`);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Check minimum length
|
|
43
|
+
if (content.length < 500) {
|
|
44
|
+
errors.push('Spec content is too short (minimum 500 characters)');
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return {
|
|
48
|
+
valid: errors.length === 0,
|
|
49
|
+
errors
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
/**
|
|
54
|
+
* Validate tasks.json content and structure
|
|
55
|
+
*
|
|
56
|
+
* @param {string} tasksPath - Path to tasks.json file
|
|
57
|
+
* @returns {Object} { valid, errors }
|
|
58
|
+
*/
|
|
59
|
+
export function validateTasksContent(tasksPath) {
|
|
60
|
+
if (!existsSync(tasksPath)) {
|
|
61
|
+
return {
|
|
62
|
+
valid: false,
|
|
63
|
+
errors: ['Tasks file does not exist']
|
|
64
|
+
};
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
try {
|
|
68
|
+
const content = readFileSync(tasksPath, 'utf8');
|
|
69
|
+
const tasks = JSON.parse(content);
|
|
70
|
+
const errors = [];
|
|
71
|
+
|
|
72
|
+
// Check if it's an array
|
|
73
|
+
if (!Array.isArray(tasks)) {
|
|
74
|
+
errors.push('tasks.json must contain an array of tasks');
|
|
75
|
+
return { valid: false, errors };
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// Validate each task
|
|
79
|
+
for (const task of tasks) {
|
|
80
|
+
if (!task.id) {
|
|
81
|
+
errors.push(`Task missing required field: id`);
|
|
82
|
+
}
|
|
83
|
+
if (!task.description) {
|
|
84
|
+
errors.push(`Task ${task.id || '(unknown)'} missing required field: description`);
|
|
85
|
+
}
|
|
86
|
+
if (!task.status) {
|
|
87
|
+
errors.push(`Task ${task.id || '(unknown)'} missing required field: status`);
|
|
88
|
+
}
|
|
89
|
+
if (task.status && !['pending', 'in_progress', 'completed'].includes(task.status)) {
|
|
90
|
+
errors.push(`Task ${task.id} has invalid status: ${task.status}`);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// Check for duplicate IDs
|
|
95
|
+
const ids = tasks.map(t => t.id);
|
|
96
|
+
const duplicates = ids.filter((id, index) => ids.indexOf(id) !== index);
|
|
97
|
+
if (duplicates.length > 0) {
|
|
98
|
+
errors.push(`Duplicate task IDs found: ${duplicates.join(', ')}`);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Check for circular dependencies
|
|
102
|
+
for (const task of tasks) {
|
|
103
|
+
if (task.dependencies && Array.isArray(task.dependencies)) {
|
|
104
|
+
for (const depId of task.dependencies) {
|
|
105
|
+
if (!ids.includes(depId)) {
|
|
106
|
+
errors.push(`Task ${task.id} depends on non-existent task: ${depId}`);
|
|
107
|
+
}
|
|
108
|
+
if (depId === task.id) {
|
|
109
|
+
errors.push(`Task ${task.id} cannot depend on itself`);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
return {
|
|
116
|
+
valid: errors.length === 0,
|
|
117
|
+
errors
|
|
118
|
+
};
|
|
119
|
+
} catch (err) {
|
|
120
|
+
return {
|
|
121
|
+
valid: false,
|
|
122
|
+
errors: [`Failed to parse tasks.json: ${err.message}`]
|
|
123
|
+
};
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/**
|
|
128
|
+
* Validate feature outputs (proposal, spec, tasks, etc.)
|
|
129
|
+
*
|
|
130
|
+
* @param {string} projectPath - Project root path
|
|
131
|
+
* @param {string} featureName - Feature name
|
|
132
|
+
* @param {string} phase - Current phase
|
|
133
|
+
* @returns {Object} { valid, errors }
|
|
134
|
+
*/
|
|
135
|
+
export function validateFeatureOutputs(projectPath, featureName, phase) {
|
|
136
|
+
const errors = [];
|
|
137
|
+
|
|
138
|
+
// Phase-specific output requirements
|
|
139
|
+
const phaseOutputs = {
|
|
140
|
+
proposal: ['proposal.md'],
|
|
141
|
+
setup: [], // No specific outputs required
|
|
142
|
+
uiux: ['ui-design-system.md'],
|
|
143
|
+
design: ['spec.md'],
|
|
144
|
+
clarify: [], // Updates to spec.md
|
|
145
|
+
tasks: ['tasks.json'],
|
|
146
|
+
implement: [], // Code files
|
|
147
|
+
sync: ['decisions.md', 'recap.md']
|
|
148
|
+
};
|
|
149
|
+
|
|
150
|
+
const requiredOutputs = phaseOutputs[phase] || [];
|
|
151
|
+
|
|
152
|
+
// Check if required outputs exist
|
|
153
|
+
for (const output of requiredOutputs) {
|
|
154
|
+
const outputPath = `${projectPath}/.morph/project/outputs/${featureName}/${output}`;
|
|
155
|
+
if (!existsSync(outputPath)) {
|
|
156
|
+
errors.push(`Missing required output for phase ${phase}: ${output}`);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
return {
|
|
161
|
+
valid: errors.length === 0,
|
|
162
|
+
errors
|
|
163
|
+
};
|
|
164
|
+
}
|