@polymorphism-tech/morph-spec 4.7.2 → 4.8.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.morph/analytics/threads-log.jsonl +54 -5
- package/.morph/state.json +152 -2
- package/LICENSE +1 -2
- package/README.md +379 -414
- package/bin/morph-spec.js +57 -394
- package/bin/validate.js +2 -26
- package/claude-plugin.json +2 -2
- package/docs/ARCHITECTURE.md +43 -46
- package/docs/CHEATSHEET.md +203 -221
- package/docs/COMMAND-FLOWS.md +319 -289
- package/docs/QUICKSTART.md +2 -8
- package/docs/plans/2026-02-22-claude-docs-morph-alignment-analysis.md +2 -0
- package/docs/plans/2026-02-22-claude-settings.md +2 -0
- package/docs/plans/2026-02-22-morph-cc-alignment-impl.md +2 -0
- package/docs/plans/2026-02-22-morph-spec-next.md +2 -0
- package/docs/plans/2026-02-22-native-alignment-design.md +2 -0
- package/docs/plans/2026-02-22-native-alignment-impl.md +2 -0
- package/docs/plans/2026-02-22-native-enrichment-design.md +2 -0
- package/docs/plans/2026-02-22-native-enrichment.md +2 -0
- package/docs/plans/2026-02-23-ddd-architecture-refactor.md +2 -0
- package/docs/plans/2026-02-23-ddd-nextsteps.md +2 -0
- package/docs/plans/2026-02-23-infra-architect-refactor.md +2 -0
- package/docs/plans/2026-02-23-nextjs-code-review-design.md +2 -1
- package/docs/plans/2026-02-23-nextjs-code-review-impl.md +2 -0
- package/docs/plans/2026-02-23-nextjs-standards-design.md +2 -1
- package/docs/plans/2026-02-23-nextjs-standards-impl.md +2 -0
- package/docs/plans/2026-02-24-cli-radical-simplification.md +592 -0
- package/docs/plans/2026-02-24-framework-failure-points.md +125 -0
- package/docs/plans/2026-02-24-morph-init-design.md +337 -0
- package/docs/plans/2026-02-24-morph-init-impl.md +1269 -0
- package/docs/plans/2026-02-24-tutorial-command-design.md +71 -0
- package/docs/plans/2026-02-24-tutorial-command.md +298 -0
- package/framework/CLAUDE.md +1 -1
- package/framework/commands/morph-proposal.md +3 -3
- package/framework/hooks/README.md +2 -5
- package/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +4 -55
- package/framework/hooks/claude-code/session-start/inject-morph-context.js +20 -5
- package/framework/hooks/claude-code/statusline.py +6 -1
- package/framework/hooks/dev/check-sync-health.js +117 -0
- package/framework/hooks/dev/guard-version-numbers.js +57 -0
- package/framework/hooks/dev/sync-standards-registry.js +60 -0
- package/framework/hooks/dev/sync-template-registry.js +60 -0
- package/framework/hooks/dev/validate-skill-format.js +70 -0
- package/framework/hooks/dev/validate-standard-format.js +73 -0
- package/framework/hooks/shared/payload-utils.js +39 -0
- package/framework/hooks/shared/state-reader.js +25 -1
- package/framework/rules/morph-workflow.md +1 -1
- package/framework/skills/level-0-meta/morph-init/SKILL.md +216 -0
- package/framework/skills/level-0-meta/morph-replicate/SKILL.md +4 -4
- package/framework/skills/level-0-meta/tool-usage-guide/SKILL.md +4 -4
- package/framework/skills/level-0-meta/verification-before-completion/SKILL.md +1 -1
- package/framework/skills/level-1-workflows/phase-clarify/SKILL.md +192 -191
- package/framework/skills/level-1-workflows/phase-codebase-analysis/SKILL.md +181 -180
- package/framework/skills/level-1-workflows/phase-design/SKILL.md +339 -338
- package/framework/skills/level-1-workflows/phase-implement/SKILL.md +254 -253
- package/framework/skills/level-1-workflows/phase-setup/SKILL.md +168 -170
- package/framework/skills/level-1-workflows/phase-tasks/SKILL.md +284 -283
- package/framework/skills/level-1-workflows/phase-uiux/SKILL.md +246 -245
- package/framework/templates/examples/design-system-examples.md +1 -1
- package/framework/templates/ui/FluentDesignTheme.cs +1 -1
- package/framework/templates/ui/MudTheme.cs +1 -1
- package/framework/templates/ui/design-system.css +1 -1
- package/package.json +4 -2
- package/scripts/bump-version.js +248 -0
- package/scripts/install-dev-hooks.js +138 -0
- package/src/commands/agents/index.js +1 -2
- package/src/commands/index.js +13 -16
- package/src/commands/project/doctor.js +100 -14
- package/src/commands/project/index.js +7 -10
- package/src/commands/project/init.js +398 -528
- package/src/commands/project/install-plugin-cmd.js +28 -0
- package/src/commands/project/setup-infra-cmd.js +12 -0
- package/src/commands/project/tutorial.js +115 -0
- package/src/commands/state/approve.js +213 -221
- package/src/commands/state/index.js +0 -1
- package/src/commands/state/state.js +337 -365
- package/src/commands/templates/index.js +0 -4
- package/src/commands/trust/trust.js +1 -93
- package/src/commands/utils/index.js +1 -5
- package/src/commands/validation/index.js +1 -5
- package/src/core/registry/command-registry.js +11 -285
- package/src/core/state/state-manager.js +5 -2
- package/src/lib/detectors/index.js +81 -87
- package/src/lib/detectors/structure-detector.js +275 -273
- package/src/lib/generators/recap-generator.js +232 -225
- package/src/scripts/global-install.js +34 -0
- package/src/scripts/install-plugin.js +126 -0
- package/src/scripts/setup-infra.js +203 -0
- package/src/utils/agents-installer.js +10 -1
- package/src/utils/hooks-installer.js +66 -3
- package/.morph/.morphversion +0 -5
- package/.morph/config/config.json +0 -8
- package/.morph/framework/agents.json +0 -1815
- package/.morph/framework/hooks/README.md +0 -205
- package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +0 -54
- package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +0 -83
- package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +0 -42
- package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +0 -61
- package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +0 -71
- package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +0 -58
- package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +0 -64
- package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +0 -94
- package/.morph/framework/hooks/claude-code/statusline.py +0 -538
- package/.morph/framework/hooks/claude-code/statusline.sh +0 -7
- package/.morph/framework/hooks/claude-code/stop/validate-completion.js +0 -88
- package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +0 -91
- package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +0 -33
- package/.morph/framework/hooks/git/pre-commit/agents.sh +0 -25
- package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +0 -64
- package/.morph/framework/hooks/git/pre-commit/specs.sh +0 -50
- package/.morph/framework/hooks/git/pre-push/run-tests.sh +0 -44
- package/.morph/framework/hooks/shared/hook-response.js +0 -45
- package/.morph/framework/hooks/shared/phase-utils.js +0 -129
- package/.morph/framework/hooks/shared/state-reader.js +0 -138
- package/.morph/framework/hooks/shared/stdin-reader.js +0 -26
- package/.morph/framework/standards/STANDARDS.json +0 -933
- package/.morph/framework/standards/ai-agents/blazor-ui.md +0 -364
- package/.morph/framework/standards/ai-agents/production.md +0 -415
- package/.morph/framework/standards/ai-agents/setup.md +0 -418
- package/.morph/framework/standards/ai-agents/team-orchestration.md +0 -479
- package/.morph/framework/standards/ai-agents/workflows.md +0 -354
- package/.morph/framework/standards/architecture/ddd/aggregates.md +0 -120
- package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +0 -105
- package/.morph/framework/standards/architecture/ddd/complexity-levels.md +0 -108
- package/.morph/framework/standards/architecture/ddd/entities.md +0 -99
- package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +0 -58
- package/.morph/framework/standards/architecture/ddd/value-objects.md +0 -124
- package/.morph/framework/standards/backend/api/minimal-api.md +0 -494
- package/.morph/framework/standards/backend/api/rest.md +0 -492
- package/.morph/framework/standards/backend/api/validation.md +0 -88
- package/.morph/framework/standards/backend/authentication/passkeys.md +0 -428
- package/.morph/framework/standards/backend/database/ef-core.md +0 -199
- package/.morph/framework/standards/backend/database/migrations.md +0 -393
- package/.morph/framework/standards/backend/database/postgresql/database.md +0 -352
- package/.morph/framework/standards/backend/database/repository-patterns.md +0 -528
- package/.morph/framework/standards/backend/database/vector-search-rag.md +0 -541
- package/.morph/framework/standards/backend/dotnet/async.md +0 -366
- package/.morph/framework/standards/backend/dotnet/core.md +0 -117
- package/.morph/framework/standards/backend/dotnet/di.md +0 -439
- package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +0 -92
- package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +0 -216
- package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +0 -290
- package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +0 -350
- package/.morph/framework/standards/backend/integrations/resend/resend-email.md +0 -385
- package/.morph/framework/standards/context/analytics.md +0 -96
- package/.morph/framework/standards/context/bundles.md +0 -110
- package/.morph/framework/standards/context/priming.md +0 -78
- package/.morph/framework/standards/core/architecture.md +0 -185
- package/.morph/framework/standards/core/coding.md +0 -214
- package/.morph/framework/standards/core/git-branching-strategy.md +0 -403
- package/.morph/framework/standards/core/git.md +0 -185
- package/.morph/framework/standards/core/testing.md +0 -295
- package/.morph/framework/standards/data/nosql/blob-storage.md +0 -102
- package/.morph/framework/standards/data/nosql/cache/redis.md +0 -97
- package/.morph/framework/standards/data/nosql/cosmos-db.md +0 -118
- package/.morph/framework/standards/data/vector-search/azure-ai-search.md +0 -121
- package/.morph/framework/standards/data/vector-search/rag-chunking.md +0 -104
- package/.morph/framework/standards/frontend/blazor/design-checklist.md +0 -222
- package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +0 -595
- package/.morph/framework/standards/frontend/blazor/fluent-ui.md +0 -137
- package/.morph/framework/standards/frontend/blazor/html-conversion.md +0 -184
- package/.morph/framework/standards/frontend/blazor/lifecycle.md +0 -195
- package/.morph/framework/standards/frontend/blazor/pitfalls.md +0 -198
- package/.morph/framework/standards/frontend/blazor/state.md +0 -191
- package/.morph/framework/standards/frontend/design-system/animations.md +0 -151
- package/.morph/framework/standards/frontend/design-system/naming.md +0 -64
- package/.morph/framework/standards/frontend/nextjs/app-router.md +0 -123
- package/.morph/framework/standards/frontend/nextjs/components.md +0 -132
- package/.morph/framework/standards/frontend/nextjs/data-fetching.md +0 -126
- package/.morph/framework/standards/frontend/nextjs/forms.md +0 -128
- package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +0 -67
- package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +0 -215
- package/.morph/framework/standards/frontend/nextjs/project-structure.md +0 -102
- package/.morph/framework/standards/frontend/nextjs/state-management.md +0 -72
- package/.morph/framework/standards/frontend/nextjs/testing.md +0 -111
- package/.morph/framework/standards/infrastructure/azure/azure.md +0 -624
- package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +0 -422
- package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +0 -516
- package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +0 -520
- package/.morph/framework/standards/infrastructure/azure/services/functions.md +0 -486
- package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +0 -459
- package/.morph/framework/standards/infrastructure/azure/services/storage.md +0 -407
- package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +0 -196
- package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +0 -252
- package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +0 -176
- package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +0 -169
- package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +0 -184
- package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +0 -153
- package/.morph/framework/standards/integration/api/graphql.md +0 -91
- package/.morph/framework/standards/integration/api/grpc.md +0 -114
- package/.morph/framework/standards/integration/api/rest-design.md +0 -95
- package/.morph/framework/standards/integration/event-driven/cqrs.md +0 -101
- package/.morph/framework/standards/integration/event-driven/event-sourcing.md +0 -124
- package/.morph/framework/standards/integration/event-driven/service-bus.md +0 -95
- package/.morph/framework/standards/integration/mcp/mcp-tools.md +0 -384
- package/.morph/framework/standards/observability/logging.md +0 -131
- package/.morph/framework/standards/observability/metrics.md +0 -121
- package/.morph/framework/standards/observability/monitoring.md +0 -114
- package/.morph/framework/standards/observability/tracing.md +0 -132
- package/.morph/framework/standards/workflows/parallel-execution.md +0 -112
- package/.morph/framework/standards/workflows/thread-management.md +0 -113
- package/.morph/framework/templates/.idea/morph-templates.xml +0 -92
- package/.morph/framework/templates/.vscode/morph-templates.code-snippets +0 -186
- package/.morph/framework/templates/IDE-SNIPPETS.md +0 -266
- package/.morph/framework/templates/README.md +0 -814
- package/.morph/framework/templates/REGISTRY.json +0 -1888
- package/.morph/framework/templates/code/dotnet/backend/repository.cs +0 -141
- package/.morph/framework/templates/code/dotnet/backend/service.cs +0 -139
- package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +0 -74
- package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +0 -25
- package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +0 -74
- package/.morph/framework/templates/code/dotnet/contracts/README.md +0 -74
- package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +0 -173
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +0 -69
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +0 -86
- package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +0 -41
- package/.morph/framework/templates/code/dotnet/database/migration.cs +0 -83
- package/.morph/framework/templates/code/dotnet/frontend/component.razor +0 -239
- package/.morph/framework/templates/code/dotnet/jobs/agent.cs +0 -163
- package/.morph/framework/templates/code/dotnet/jobs/job.cs +0 -171
- package/.morph/framework/templates/code/dotnet/test.cs +0 -239
- package/.morph/framework/templates/code/sql/rls-policy.sql +0 -57
- package/.morph/framework/templates/code/sql/supabase-migration.sql +0 -100
- package/.morph/framework/templates/code/sql/supabase-migration.template.sql +0 -113
- package/.morph/framework/templates/code/typescript/contracts.ts +0 -168
- package/.morph/framework/templates/context/CONTEXT-FEATURE.md +0 -276
- package/.morph/framework/templates/context/CONTEXT.md +0 -181
- package/.morph/framework/templates/docs/clarifications.md +0 -253
- package/.morph/framework/templates/docs/onboarding.md +0 -123
- package/.morph/framework/templates/docs/proposal.md +0 -182
- package/.morph/framework/templates/docs/schema-analysis.md +0 -119
- package/.morph/framework/templates/docs/spec.md +0 -198
- package/.morph/framework/templates/docs/ui-components.md +0 -124
- package/.morph/framework/templates/docs/ui-design-system.md +0 -76
- package/.morph/framework/templates/docs/ui-flows.md +0 -167
- package/.morph/framework/templates/docs/ui-mockups.md +0 -98
- package/.morph/framework/templates/docs/user-stories.md +0 -34
- package/.morph/framework/templates/examples/design-system-examples.md +0 -357
- package/.morph/framework/templates/examples/spec-examples.md +0 -90
- package/.morph/framework/templates/feature/decisions.md +0 -187
- package/.morph/framework/templates/feature/recap.md +0 -146
- package/.morph/framework/templates/feature/tasks.md +0 -199
- package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +0 -43
- package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +0 -26
- package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +0 -32
- package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +0 -56
- package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +0 -22
- package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +0 -26
- package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +0 -54
- package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +0 -82
- package/.morph/framework/templates/infrastructure/azure/README.md +0 -286
- package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +0 -63
- package/.morph/framework/templates/infrastructure/azure/app-service.bicep +0 -164
- package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +0 -49
- package/.morph/framework/templates/infrastructure/azure/container-app.bicep +0 -156
- package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +0 -426
- package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +0 -229
- package/.morph/framework/templates/infrastructure/azure/deploy.sh +0 -208
- package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +0 -91
- package/.morph/framework/templates/infrastructure/azure/main.bicep +0 -189
- package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +0 -29
- package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +0 -29
- package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +0 -29
- package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +0 -103
- package/.morph/framework/templates/infrastructure/azure/storage.bicep +0 -106
- package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +0 -58
- package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +0 -67
- package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +0 -38
- package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +0 -48
- package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +0 -54
- package/.morph/framework/templates/infrastructure/github/README.md +0 -593
- package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +0 -22
- package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +0 -45
- package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +0 -27
- package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +0 -61
- package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +0 -31
- package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +0 -59
- package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +0 -39
- package/.morph/framework/templates/integrations/asaas-client.cs +0 -387
- package/.morph/framework/templates/integrations/asaas-webhook.cs +0 -351
- package/.morph/framework/templates/integrations/azure-identity-config.cs +0 -288
- package/.morph/framework/templates/integrations/clerk-config.cs +0 -258
- package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +0 -76
- package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +0 -100
- package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +0 -78
- package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +0 -97
- package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +0 -36
- package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +0 -113
- package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +0 -80
- package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +0 -90
- package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +0 -126
- package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +0 -43
- package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +0 -107
- package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +0 -95
- package/.morph/framework/templates/project-structure/dotnet-ddd.md +0 -70
- package/.morph/framework/templates/saas/subscription.cs +0 -347
- package/.morph/framework/templates/saas/tenant.cs +0 -338
- package/.morph/framework/templates/state.template.json +0 -17
- package/.morph/framework/templates/ui/FluentDesignTheme.cs +0 -149
- package/.morph/framework/templates/ui/MudTheme.cs +0 -281
- package/.morph/framework/templates/ui/design-system.css +0 -226
- package/.morph/logs/tool-failures.log +0 -17
- package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +0 -16
- package/.morph/plans/eager-watching-bunny.md +0 -105
- package/.morph/plans/temporal-seeking-nebula.md +0 -45
- package/CLAUDE.md +0 -77
- package/docs/claude-alignment-report.md +0 -137
- package/docs/examples/order-management/contracts.cs +0 -84
- package/docs/examples/order-management/proposal.md +0 -24
- package/docs/examples/order-management/spec.md +0 -162
- package/src/commands/feature/create-story.js +0 -362
- package/src/commands/feature/index.js +0 -6
- package/src/commands/feature/shard-spec.js +0 -225
- package/src/commands/feature/sprint-status.js +0 -250
- package/src/commands/generation/generate-onboarding.js +0 -169
- package/src/commands/generation/generate.js +0 -276
- package/src/commands/generation/index.js +0 -5
- package/src/commands/learning/capture-pattern.js +0 -121
- package/src/commands/learning/index.js +0 -5
- package/src/commands/learning/search-patterns.js +0 -126
- package/src/commands/mcp/mcp.js +0 -102
- package/src/commands/project/changes.js +0 -66
- package/src/commands/project/cost.js +0 -179
- package/src/commands/project/diff.js +0 -278
- package/src/commands/project/revert.js +0 -173
- package/src/commands/project/standards.js +0 -80
- package/src/commands/project/sync.js +0 -167
- package/src/commands/project/update-agents.js +0 -23
- package/src/commands/state/rollback-phase.js +0 -185
- package/src/commands/templates/template-customize.js +0 -87
- package/src/commands/templates/template-list.js +0 -114
- package/src/commands/templates/template-show.js +0 -129
- package/src/commands/templates/template-validate.js +0 -91
- package/src/commands/utils/troubleshoot.js +0 -222
- package/src/commands/validation/analyze-blazor-concurrency.js +0 -193
- package/src/commands/validation/lint-fluent.js +0 -352
- package/src/commands/validation/validate-blazor-state.js +0 -210
- package/src/commands/validation/validate-blazor.js +0 -156
- package/src/commands/validation/validate-css.js +0 -84
- package/src/lib/detectors/conversation-analyzer.js +0 -163
- package/src/lib/learning/index.js +0 -7
- package/src/lib/learning/learning-system.js +0 -520
- package/src/lib/troubleshooting/index.js +0 -8
- package/src/lib/troubleshooting/troubleshoot-grep.js +0 -198
- package/src/lib/troubleshooting/troubleshoot-index.js +0 -144
- package/src/llm/environment-detector.js +0 -43
|
@@ -1,163 +0,0 @@
|
|
|
1
|
-
// ============================================================
|
|
2
|
-
// MICROSOFT AGENT FRAMEWORK TEMPLATE
|
|
3
|
-
// Generated by MORPH Framework
|
|
4
|
-
// ============================================================
|
|
5
|
-
|
|
6
|
-
using System.ComponentModel;
|
|
7
|
-
using System.Text.Json;
|
|
8
|
-
using Microsoft.Agents.AI;
|
|
9
|
-
using Microsoft.Extensions.AI;
|
|
10
|
-
using Microsoft.Extensions.Logging;
|
|
11
|
-
|
|
12
|
-
namespace {{NAMESPACE}}.Agents.{{pascalCase FEATURE_NAME}};
|
|
13
|
-
|
|
14
|
-
/// <summary>
|
|
15
|
-
/// AI Agent for analyzing {{pascalCase FEATURE_NAME}} data.
|
|
16
|
-
/// Uses Microsoft Agent Framework with IChatClient (gpt-4o-mini by default).
|
|
17
|
-
/// </summary>
|
|
18
|
-
public class {{pascalCase FEATURE_NAME}}AnalyzerAgent(
|
|
19
|
-
IChatClient chatClient,
|
|
20
|
-
ILogger<{{pascalCase FEATURE_NAME}}AnalyzerAgent> logger) : I{{pascalCase FEATURE_NAME}}AnalyzerAgent
|
|
21
|
-
{
|
|
22
|
-
/// <inheritdoc />
|
|
23
|
-
public async Task<{{pascalCase FEATURE_NAME}}AnalysisResult> AnalyzeAsync(
|
|
24
|
-
{{pascalCase FEATURE_NAME}}Data data,
|
|
25
|
-
CancellationToken cancellationToken = default)
|
|
26
|
-
{
|
|
27
|
-
logger.LogInformation("Starting {{pascalCase FEATURE_NAME}} analysis for: {Content}",
|
|
28
|
-
data.Content.Length > 100 ? data.Content[..100] + "..." : data.Content);
|
|
29
|
-
|
|
30
|
-
var agent = new ChatClientAgent(
|
|
31
|
-
chatClient,
|
|
32
|
-
new ChatClientAgentOptions
|
|
33
|
-
{
|
|
34
|
-
Name = "{{pascalCase FEATURE_NAME}}Analyzer",
|
|
35
|
-
Instructions = """
|
|
36
|
-
You are an expert analyst for {{pascalCase FEATURE_NAME}} data.
|
|
37
|
-
|
|
38
|
-
Analyze the provided data and return a JSON object with:
|
|
39
|
-
- summary: Brief summary (1-2 sentences)
|
|
40
|
-
- insights: Array of at least 2 insights
|
|
41
|
-
- recommendations: Array of at least 1 recommendation
|
|
42
|
-
- confidenceScore: Number between 0 and 1
|
|
43
|
-
|
|
44
|
-
Respond ONLY with valid JSON, no additional text.
|
|
45
|
-
""",
|
|
46
|
-
ChatOptions = new ChatOptions
|
|
47
|
-
{
|
|
48
|
-
Tools =
|
|
49
|
-
[
|
|
50
|
-
AIFunctionFactory.Create(GetContextAsync),
|
|
51
|
-
AIFunctionFactory.Create(GetMetricsAsync)
|
|
52
|
-
]
|
|
53
|
-
}
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
var response = await agent.RunAsync(
|
|
57
|
-
$"Analyze this data:\n{data.Content}",
|
|
58
|
-
cancellationToken: cancellationToken);
|
|
59
|
-
|
|
60
|
-
var result = ParseResponse(response.Text);
|
|
61
|
-
|
|
62
|
-
logger.LogInformation(
|
|
63
|
-
"Analysis completed. Confidence: {Confidence:P0}",
|
|
64
|
-
result.ConfidenceScore);
|
|
65
|
-
|
|
66
|
-
return result;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
#region Tools
|
|
70
|
-
|
|
71
|
-
[Description("Gets additional context for the analysis")]
|
|
72
|
-
private async Task<string> GetContextAsync(
|
|
73
|
-
[Description("What context is needed")] string query,
|
|
74
|
-
CancellationToken ct = default)
|
|
75
|
-
{
|
|
76
|
-
// TODO: Implement context retrieval (e.g., from database or search)
|
|
77
|
-
await Task.CompletedTask;
|
|
78
|
-
return $"Context for: {query}";
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
[Description("Gets relevant metrics for the analysis")]
|
|
82
|
-
private async Task<string> GetMetricsAsync(
|
|
83
|
-
[Description("Metric type to retrieve")] string metricType,
|
|
84
|
-
CancellationToken ct = default)
|
|
85
|
-
{
|
|
86
|
-
// TODO: Implement metrics retrieval
|
|
87
|
-
await Task.CompletedTask;
|
|
88
|
-
return $"Metrics for: {metricType}";
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
#endregion
|
|
92
|
-
|
|
93
|
-
#region Response Parsing
|
|
94
|
-
|
|
95
|
-
private {{pascalCase FEATURE_NAME}}AnalysisResult ParseResponse(string response)
|
|
96
|
-
{
|
|
97
|
-
try
|
|
98
|
-
{
|
|
99
|
-
var json = response
|
|
100
|
-
.Replace("```json", "")
|
|
101
|
-
.Replace("```", "")
|
|
102
|
-
.Trim();
|
|
103
|
-
|
|
104
|
-
var parsed = JsonSerializer.Deserialize<AnalysisResponse>(json,
|
|
105
|
-
new JsonSerializerOptions { PropertyNameCaseInsensitive = true });
|
|
106
|
-
|
|
107
|
-
if (parsed is null)
|
|
108
|
-
throw new {{pascalCase FEATURE_NAME}}ProcessingException("Failed to parse analysis response");
|
|
109
|
-
|
|
110
|
-
return new {{pascalCase FEATURE_NAME}}AnalysisResult(
|
|
111
|
-
parsed.Summary ?? "Analysis completed",
|
|
112
|
-
parsed.Insights ?? [],
|
|
113
|
-
parsed.Recommendations ?? [],
|
|
114
|
-
parsed.ConfidenceScore);
|
|
115
|
-
}
|
|
116
|
-
catch (JsonException ex)
|
|
117
|
-
{
|
|
118
|
-
logger.LogError(ex, "Failed to parse AI response: {Response}", response);
|
|
119
|
-
throw new {{pascalCase FEATURE_NAME}}ProcessingException("Failed to parse analysis response", ex);
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
private class AnalysisResponse
|
|
124
|
-
{
|
|
125
|
-
public string? Summary { get; set; }
|
|
126
|
-
public List<string>? Insights { get; set; }
|
|
127
|
-
public List<string>? Recommendations { get; set; }
|
|
128
|
-
public double ConfidenceScore { get; set; }
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
#endregion
|
|
132
|
-
}
|
|
133
|
-
|
|
134
|
-
// ============================================================
|
|
135
|
-
// DEPENDENCY INJECTION SETUP
|
|
136
|
-
// ============================================================
|
|
137
|
-
//
|
|
138
|
-
// In Program.cs:
|
|
139
|
-
//
|
|
140
|
-
// // Register IChatClient (Azure OpenAI with Managed Identity)
|
|
141
|
-
// builder.Services.AddSingleton<IChatClient>(sp =>
|
|
142
|
-
// {
|
|
143
|
-
// var cfg = sp.GetRequiredService<IConfiguration>();
|
|
144
|
-
// return new AzureOpenAIClient(
|
|
145
|
-
// new Uri(cfg["AzureOpenAI:Endpoint"]!),
|
|
146
|
-
// new DefaultAzureCredential())
|
|
147
|
-
// .GetChatClient(cfg["AzureOpenAI:DeploymentName"] ?? "gpt-4o-mini")
|
|
148
|
-
// .AsIChatClient();
|
|
149
|
-
// });
|
|
150
|
-
//
|
|
151
|
-
// // Option A: Register as scoped service
|
|
152
|
-
// builder.Services.AddScoped<I{{pascalCase FEATURE_NAME}}AnalyzerAgent, {{pascalCase FEATURE_NAME}}AnalyzerAgent>();
|
|
153
|
-
//
|
|
154
|
-
// // Option B: Register as keyed agent (recommended for multiple agents)
|
|
155
|
-
// builder.AddAIAgent("{{pascalCase FEATURE_NAME}}Analyzer", (sp, key) =>
|
|
156
|
-
// {
|
|
157
|
-
// var chatClient = sp.GetRequiredService<IChatClient>();
|
|
158
|
-
// return new ChatClientAgent(chatClient,
|
|
159
|
-
// name: key,
|
|
160
|
-
// instructions: "You are an expert {{pascalCase FEATURE_NAME}} analyst.");
|
|
161
|
-
// });
|
|
162
|
-
//
|
|
163
|
-
// ============================================================
|
|
@@ -1,171 +0,0 @@
|
|
|
1
|
-
// ============================================================
|
|
2
|
-
// HANGFIRE JOB TEMPLATE
|
|
3
|
-
// Generated by MORPH Framework
|
|
4
|
-
// ============================================================
|
|
5
|
-
|
|
6
|
-
using Hangfire;
|
|
7
|
-
using Microsoft.Extensions.Logging;
|
|
8
|
-
|
|
9
|
-
namespace {{NAMESPACE}}.Application.Features.{{pascalCase FEATURE_NAME}}.Jobs;
|
|
10
|
-
|
|
11
|
-
/// <summary>
|
|
12
|
-
/// Background job for processing {{pascalCase FEATURE_NAME}}.
|
|
13
|
-
/// Uses Hangfire for scheduling and retry handling.
|
|
14
|
-
/// </summary>
|
|
15
|
-
public class {{pascalCase FEATURE_NAME}}ProcessorJob(
|
|
16
|
-
I{{pascalCase FEATURE_NAME}}Service service,
|
|
17
|
-
I{{pascalCase FEATURE_NAME}}AnalyzerAgent analyzer,
|
|
18
|
-
ILogger<{{pascalCase FEATURE_NAME}}ProcessorJob> logger) : I{{pascalCase FEATURE_NAME}}ProcessorJob
|
|
19
|
-
{
|
|
20
|
-
/// <summary>
|
|
21
|
-
/// Executes the {{pascalCase FEATURE_NAME}} processing job.
|
|
22
|
-
/// </summary>
|
|
23
|
-
/// <param name="id">The {{pascalCase FEATURE_NAME}} ID to process</param>
|
|
24
|
-
/// <param name="cancellationToken">Cancellation token</param>
|
|
25
|
-
[AutomaticRetry(Attempts = 3, DelaysInSeconds = new[] { 60, 300, 900 })]
|
|
26
|
-
[Queue("default")]
|
|
27
|
-
[JobDisplayName("{{pascalCase FEATURE_NAME}} Processing - ID: {0}")]
|
|
28
|
-
public async Task ExecuteAsync(int id, CancellationToken cancellationToken)
|
|
29
|
-
{
|
|
30
|
-
logger.LogInformation("Starting {{pascalCase FEATURE_NAME}} processing for ID {Id}", id);
|
|
31
|
-
|
|
32
|
-
try
|
|
33
|
-
{
|
|
34
|
-
// Get the entity
|
|
35
|
-
var item = await service.GetByIdAsync(id, cancellationToken);
|
|
36
|
-
if (item is null)
|
|
37
|
-
{
|
|
38
|
-
logger.LogWarning("{{pascalCase FEATURE_NAME}} with ID {Id} not found, skipping", id);
|
|
39
|
-
return;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
// Check if already processed
|
|
43
|
-
if (item.Status == {{pascalCase FEATURE_NAME}}Status.Completed)
|
|
44
|
-
{
|
|
45
|
-
logger.LogInformation("{{pascalCase FEATURE_NAME}} {Id} already completed, skipping", id);
|
|
46
|
-
return;
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
// Perform analysis (if applicable)
|
|
50
|
-
var analysisData = new {{pascalCase FEATURE_NAME}}Data(item.Name);
|
|
51
|
-
var analysis = await analyzer.AnalyzeAsync(analysisData, cancellationToken);
|
|
52
|
-
|
|
53
|
-
logger.LogInformation(
|
|
54
|
-
"{{pascalCase FEATURE_NAME}} {Id} analyzed. Confidence: {Confidence:P0}",
|
|
55
|
-
id, analysis.ConfidenceScore);
|
|
56
|
-
|
|
57
|
-
// Update status
|
|
58
|
-
// Note: You might need to add a method to update with analysis results
|
|
59
|
-
// await service.CompleteWithAnalysisAsync(id, analysis, cancellationToken);
|
|
60
|
-
|
|
61
|
-
logger.LogInformation("Completed {{pascalCase FEATURE_NAME}} processing for ID {Id}", id);
|
|
62
|
-
}
|
|
63
|
-
catch (Exception ex)
|
|
64
|
-
{
|
|
65
|
-
logger.LogError(ex, "Failed to process {{pascalCase FEATURE_NAME}} {Id}", id);
|
|
66
|
-
throw; // Re-throw to trigger Hangfire retry
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
// ============================================================
|
|
72
|
-
// RECURRING JOB CONFIGURATION
|
|
73
|
-
// ============================================================
|
|
74
|
-
//
|
|
75
|
-
// For scheduled/recurring jobs, configure in Program.cs:
|
|
76
|
-
//
|
|
77
|
-
// // Run every hour
|
|
78
|
-
// RecurringJob.AddOrUpdate<I{{pascalCase FEATURE_NAME}}ProcessorJob>(
|
|
79
|
-
// "{{kebabCase FEATURE_NAME}}-processor",
|
|
80
|
-
// job => job.ExecuteAsync(0, CancellationToken.None),
|
|
81
|
-
// Cron.Hourly);
|
|
82
|
-
//
|
|
83
|
-
// // Run daily at midnight
|
|
84
|
-
// RecurringJob.AddOrUpdate<I{{pascalCase FEATURE_NAME}}ProcessorJob>(
|
|
85
|
-
// "{{kebabCase FEATURE_NAME}}-daily-processor",
|
|
86
|
-
// job => job.ExecuteAsync(0, CancellationToken.None),
|
|
87
|
-
// Cron.Daily);
|
|
88
|
-
//
|
|
89
|
-
// // Custom cron expression (every 15 minutes)
|
|
90
|
-
// RecurringJob.AddOrUpdate<I{{pascalCase FEATURE_NAME}}ProcessorJob>(
|
|
91
|
-
// "{{kebabCase FEATURE_NAME}}-frequent-processor",
|
|
92
|
-
// job => job.ExecuteAsync(0, CancellationToken.None),
|
|
93
|
-
// "*/15 * * * *");
|
|
94
|
-
//
|
|
95
|
-
// ============================================================
|
|
96
|
-
|
|
97
|
-
// ============================================================
|
|
98
|
-
// BATCH JOB TEMPLATE
|
|
99
|
-
// ============================================================
|
|
100
|
-
|
|
101
|
-
/// <summary>
|
|
102
|
-
/// Batch job for processing multiple {{pascalCase FEATURE_NAME}}s.
|
|
103
|
-
/// </summary>
|
|
104
|
-
public class {{pascalCase FEATURE_NAME}}BatchProcessorJob(
|
|
105
|
-
I{{pascalCase FEATURE_NAME}}Service service,
|
|
106
|
-
I{{pascalCase FEATURE_NAME}}ProcessorJob itemProcessor,
|
|
107
|
-
ILogger<{{pascalCase FEATURE_NAME}}BatchProcessorJob> logger)
|
|
108
|
-
{
|
|
109
|
-
/// <summary>
|
|
110
|
-
/// Processes all pending {{pascalCase FEATURE_NAME}}s.
|
|
111
|
-
/// </summary>
|
|
112
|
-
[AutomaticRetry(Attempts = 1)]
|
|
113
|
-
[Queue("batch")]
|
|
114
|
-
[JobDisplayName("{{pascalCase FEATURE_NAME}} Batch Processing")]
|
|
115
|
-
public async Task ProcessAllPendingAsync(CancellationToken cancellationToken)
|
|
116
|
-
{
|
|
117
|
-
logger.LogInformation("Starting batch processing for pending {{pascalCase FEATURE_NAME}}s");
|
|
118
|
-
|
|
119
|
-
var items = await service.GetAllAsync(cancellationToken);
|
|
120
|
-
var pending = items.Where(x => x.Status == {{pascalCase FEATURE_NAME}}Status.Pending).ToList();
|
|
121
|
-
|
|
122
|
-
logger.LogInformation("Found {Count} pending {{pascalCase FEATURE_NAME}}s to process", pending.Count);
|
|
123
|
-
|
|
124
|
-
foreach (var item in pending)
|
|
125
|
-
{
|
|
126
|
-
// Enqueue individual processing jobs
|
|
127
|
-
BackgroundJob.Enqueue<I{{pascalCase FEATURE_NAME}}ProcessorJob>(
|
|
128
|
-
job => job.ExecuteAsync(item.Id, CancellationToken.None));
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
logger.LogInformation("Enqueued {Count} processing jobs", pending.Count);
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
// ============================================================
|
|
136
|
-
// HANGFIRE CONFIGURATION
|
|
137
|
-
// ============================================================
|
|
138
|
-
//
|
|
139
|
-
// In Program.cs:
|
|
140
|
-
//
|
|
141
|
-
// // Add Hangfire services
|
|
142
|
-
// builder.Services.AddHangfire(config => config
|
|
143
|
-
// .SetDataCompatibilityLevel(CompatibilityLevel.Version_180)
|
|
144
|
-
// .UseSimpleAssemblyNameTypeSerializer()
|
|
145
|
-
// .UseRecommendedSerializerSettings()
|
|
146
|
-
// .UseSqlServerStorage(connectionString, new SqlServerStorageOptions
|
|
147
|
-
// {
|
|
148
|
-
// CommandBatchMaxTimeout = TimeSpan.FromMinutes(5),
|
|
149
|
-
// SlidingInvisibilityTimeout = TimeSpan.FromMinutes(5),
|
|
150
|
-
// QueuePollInterval = TimeSpan.Zero,
|
|
151
|
-
// UseRecommendedIsolationLevel = true,
|
|
152
|
-
// DisableGlobalLocks = true
|
|
153
|
-
// }));
|
|
154
|
-
//
|
|
155
|
-
// builder.Services.AddHangfireServer(options =>
|
|
156
|
-
// {
|
|
157
|
-
// options.Queues = new[] { "default", "batch" };
|
|
158
|
-
// options.WorkerCount = Environment.ProcessorCount * 2;
|
|
159
|
-
// });
|
|
160
|
-
//
|
|
161
|
-
// // Register jobs
|
|
162
|
-
// builder.Services.AddScoped<I{{pascalCase FEATURE_NAME}}ProcessorJob, {{pascalCase FEATURE_NAME}}ProcessorJob>();
|
|
163
|
-
// builder.Services.AddScoped<{{pascalCase FEATURE_NAME}}BatchProcessorJob>();
|
|
164
|
-
//
|
|
165
|
-
// // In app pipeline
|
|
166
|
-
// app.UseHangfireDashboard("/hangfire", new DashboardOptions
|
|
167
|
-
// {
|
|
168
|
-
// Authorization = new[] { new HangfireAuthorizationFilter() }
|
|
169
|
-
// });
|
|
170
|
-
//
|
|
171
|
-
// ============================================================
|
|
@@ -1,239 +0,0 @@
|
|
|
1
|
-
// ============================================================
|
|
2
|
-
// TEST TEMPLATE
|
|
3
|
-
// Generated by MORPH Framework
|
|
4
|
-
// ============================================================
|
|
5
|
-
|
|
6
|
-
using FluentAssertions;
|
|
7
|
-
using Microsoft.Extensions.Logging;
|
|
8
|
-
using NSubstitute;
|
|
9
|
-
using Xunit;
|
|
10
|
-
|
|
11
|
-
namespace {{NAMESPACE}}.Tests.Unit.Features.{{pascalCase FEATURE_NAME}};
|
|
12
|
-
|
|
13
|
-
/// <summary>
|
|
14
|
-
/// Unit tests for {{pascalCase FEATURE_NAME}}Service.
|
|
15
|
-
/// </summary>
|
|
16
|
-
public class {{pascalCase FEATURE_NAME}}ServiceTests
|
|
17
|
-
{
|
|
18
|
-
private readonly I{{pascalCase FEATURE_NAME}}Repository _repository;
|
|
19
|
-
private readonly ILogger<{{pascalCase FEATURE_NAME}}Service> _logger;
|
|
20
|
-
private readonly {{pascalCase FEATURE_NAME}}Service _sut;
|
|
21
|
-
|
|
22
|
-
public {{pascalCase FEATURE_NAME}}ServiceTests()
|
|
23
|
-
{
|
|
24
|
-
_repository = Substitute.For<I{{pascalCase FEATURE_NAME}}Repository>();
|
|
25
|
-
_logger = Substitute.For<ILogger<{{pascalCase FEATURE_NAME}}Service>>();
|
|
26
|
-
_sut = new {{pascalCase FEATURE_NAME}}Service(_repository, _logger);
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
#region GetByIdAsync Tests
|
|
30
|
-
|
|
31
|
-
[Fact]
|
|
32
|
-
public async Task GetByIdAsync_WithValidId_ReturnsDto()
|
|
33
|
-
{
|
|
34
|
-
// Arrange
|
|
35
|
-
var id = 1;
|
|
36
|
-
var entity = CreateTestEntity(id, "Test {{pascalCase FEATURE_NAME}}");
|
|
37
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
38
|
-
.Returns(entity);
|
|
39
|
-
|
|
40
|
-
// Act
|
|
41
|
-
var result = await _sut.GetByIdAsync(id);
|
|
42
|
-
|
|
43
|
-
// Assert
|
|
44
|
-
result.Should().NotBeNull();
|
|
45
|
-
result!.Id.Should().Be(id);
|
|
46
|
-
result.Name.Should().Be("Test {{pascalCase FEATURE_NAME}}");
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
[Fact]
|
|
50
|
-
public async Task GetByIdAsync_WithInvalidId_ReturnsNull()
|
|
51
|
-
{
|
|
52
|
-
// Arrange
|
|
53
|
-
var id = 999;
|
|
54
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
55
|
-
.Returns((Domain.Entities.{{pascalCase FEATURE_NAME}}?)null);
|
|
56
|
-
|
|
57
|
-
// Act
|
|
58
|
-
var result = await _sut.GetByIdAsync(id);
|
|
59
|
-
|
|
60
|
-
// Assert
|
|
61
|
-
result.Should().BeNull();
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
#endregion
|
|
65
|
-
|
|
66
|
-
#region GetAllAsync Tests
|
|
67
|
-
|
|
68
|
-
[Fact]
|
|
69
|
-
public async Task GetAllAsync_ReturnsAllItems()
|
|
70
|
-
{
|
|
71
|
-
// Arrange
|
|
72
|
-
var entities = new List<Domain.Entities.{{pascalCase FEATURE_NAME}}>
|
|
73
|
-
{
|
|
74
|
-
CreateTestEntity(1, "First"),
|
|
75
|
-
CreateTestEntity(2, "Second"),
|
|
76
|
-
CreateTestEntity(3, "Third")
|
|
77
|
-
};
|
|
78
|
-
_repository.GetAllAsync(Arg.Any<CancellationToken>())
|
|
79
|
-
.Returns(entities);
|
|
80
|
-
|
|
81
|
-
// Act
|
|
82
|
-
var result = await _sut.GetAllAsync();
|
|
83
|
-
|
|
84
|
-
// Assert
|
|
85
|
-
result.Should().HaveCount(3);
|
|
86
|
-
result[0].Name.Should().Be("First");
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
[Fact]
|
|
90
|
-
public async Task GetAllAsync_WithNoItems_ReturnsEmptyList()
|
|
91
|
-
{
|
|
92
|
-
// Arrange
|
|
93
|
-
_repository.GetAllAsync(Arg.Any<CancellationToken>())
|
|
94
|
-
.Returns(new List<Domain.Entities.{{pascalCase FEATURE_NAME}}>());
|
|
95
|
-
|
|
96
|
-
// Act
|
|
97
|
-
var result = await _sut.GetAllAsync();
|
|
98
|
-
|
|
99
|
-
// Assert
|
|
100
|
-
result.Should().BeEmpty();
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
#endregion
|
|
104
|
-
|
|
105
|
-
#region CreateAsync Tests
|
|
106
|
-
|
|
107
|
-
[Fact]
|
|
108
|
-
public async Task CreateAsync_WithValidRequest_CreatesAndReturnsDto()
|
|
109
|
-
{
|
|
110
|
-
// Arrange
|
|
111
|
-
var request = new Create{{pascalCase FEATURE_NAME}}Request("New {{pascalCase FEATURE_NAME}}");
|
|
112
|
-
|
|
113
|
-
_repository
|
|
114
|
-
.When(x => x.AddAsync(Arg.Any<Domain.Entities.{{pascalCase FEATURE_NAME}}>(), Arg.Any<CancellationToken>()))
|
|
115
|
-
.Do(x =>
|
|
116
|
-
{
|
|
117
|
-
// Simulate ID assignment
|
|
118
|
-
var entity = x.Arg<Domain.Entities.{{pascalCase FEATURE_NAME}}>();
|
|
119
|
-
// entity.Id would be set by EF Core
|
|
120
|
-
});
|
|
121
|
-
|
|
122
|
-
// Act
|
|
123
|
-
var result = await _sut.CreateAsync(request);
|
|
124
|
-
|
|
125
|
-
// Assert
|
|
126
|
-
result.Should().NotBeNull();
|
|
127
|
-
result.Name.Should().Be("New {{pascalCase FEATURE_NAME}}");
|
|
128
|
-
|
|
129
|
-
await _repository.Received(1).AddAsync(Arg.Any<Domain.Entities.{{pascalCase FEATURE_NAME}}>(), Arg.Any<CancellationToken>());
|
|
130
|
-
await _repository.Received(1).SaveChangesAsync(Arg.Any<CancellationToken>());
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
[Theory]
|
|
134
|
-
[InlineData("")]
|
|
135
|
-
[InlineData(" ")]
|
|
136
|
-
[InlineData(null)]
|
|
137
|
-
public async Task CreateAsync_WithInvalidName_ThrowsValidationException(string? name)
|
|
138
|
-
{
|
|
139
|
-
// Arrange
|
|
140
|
-
var request = new Create{{pascalCase FEATURE_NAME}}Request(name!);
|
|
141
|
-
|
|
142
|
-
// Act
|
|
143
|
-
var act = async () => await _sut.CreateAsync(request);
|
|
144
|
-
|
|
145
|
-
// Assert
|
|
146
|
-
await act.Should().ThrowAsync<ValidationException>();
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
#endregion
|
|
150
|
-
|
|
151
|
-
#region UpdateAsync Tests
|
|
152
|
-
|
|
153
|
-
[Fact]
|
|
154
|
-
public async Task UpdateAsync_WithValidRequest_UpdatesEntity()
|
|
155
|
-
{
|
|
156
|
-
// Arrange
|
|
157
|
-
var id = 1;
|
|
158
|
-
var entity = CreateTestEntity(id, "Original");
|
|
159
|
-
var request = new Update{{pascalCase FEATURE_NAME}}Request("Updated");
|
|
160
|
-
|
|
161
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
162
|
-
.Returns(entity);
|
|
163
|
-
|
|
164
|
-
// Act
|
|
165
|
-
await _sut.UpdateAsync(id, request);
|
|
166
|
-
|
|
167
|
-
// Assert
|
|
168
|
-
_repository.Received(1).Update(Arg.Any<Domain.Entities.{{pascalCase FEATURE_NAME}}>());
|
|
169
|
-
await _repository.Received(1).SaveChangesAsync(Arg.Any<CancellationToken>());
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
[Fact]
|
|
173
|
-
public async Task UpdateAsync_WithNonExistentId_ThrowsNotFoundException()
|
|
174
|
-
{
|
|
175
|
-
// Arrange
|
|
176
|
-
var id = 999;
|
|
177
|
-
var request = new Update{{pascalCase FEATURE_NAME}}Request("Updated");
|
|
178
|
-
|
|
179
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
180
|
-
.Returns((Domain.Entities.{{pascalCase FEATURE_NAME}}?)null);
|
|
181
|
-
|
|
182
|
-
// Act
|
|
183
|
-
var act = async () => await _sut.UpdateAsync(id, request);
|
|
184
|
-
|
|
185
|
-
// Assert
|
|
186
|
-
await act.Should().ThrowAsync<{{pascalCase FEATURE_NAME}}NotFoundException>();
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
#endregion
|
|
190
|
-
|
|
191
|
-
#region DeleteAsync Tests
|
|
192
|
-
|
|
193
|
-
[Fact]
|
|
194
|
-
public async Task DeleteAsync_WithValidId_DeletesEntity()
|
|
195
|
-
{
|
|
196
|
-
// Arrange
|
|
197
|
-
var id = 1;
|
|
198
|
-
var entity = CreateTestEntity(id, "To Delete");
|
|
199
|
-
|
|
200
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
201
|
-
.Returns(entity);
|
|
202
|
-
|
|
203
|
-
// Act
|
|
204
|
-
await _sut.DeleteAsync(id);
|
|
205
|
-
|
|
206
|
-
// Assert
|
|
207
|
-
_repository.Received(1).Remove(entity);
|
|
208
|
-
await _repository.Received(1).SaveChangesAsync(Arg.Any<CancellationToken>());
|
|
209
|
-
}
|
|
210
|
-
|
|
211
|
-
[Fact]
|
|
212
|
-
public async Task DeleteAsync_WithNonExistentId_ThrowsNotFoundException()
|
|
213
|
-
{
|
|
214
|
-
// Arrange
|
|
215
|
-
var id = 999;
|
|
216
|
-
|
|
217
|
-
_repository.GetByIdAsync(id, Arg.Any<CancellationToken>())
|
|
218
|
-
.Returns((Domain.Entities.{{pascalCase FEATURE_NAME}}?)null);
|
|
219
|
-
|
|
220
|
-
// Act
|
|
221
|
-
var act = async () => await _sut.DeleteAsync(id);
|
|
222
|
-
|
|
223
|
-
// Assert
|
|
224
|
-
await act.Should().ThrowAsync<{{pascalCase FEATURE_NAME}}NotFoundException>();
|
|
225
|
-
}
|
|
226
|
-
|
|
227
|
-
#endregion
|
|
228
|
-
|
|
229
|
-
#region Helper Methods
|
|
230
|
-
|
|
231
|
-
private static Domain.Entities.{{pascalCase FEATURE_NAME}} CreateTestEntity(int id, string name)
|
|
232
|
-
{
|
|
233
|
-
return Domain.Entities.{{pascalCase FEATURE_NAME}}.Create(name);
|
|
234
|
-
// Note: In real tests, you might need reflection to set the Id
|
|
235
|
-
// or use a factory method that accepts an id for testing
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
#endregion
|
|
239
|
-
}
|
|
@@ -1,57 +0,0 @@
|
|
|
1
|
-
-- ============================================================
|
|
2
|
-
-- RLS Policy: {{titleCase FEATURE_NAME}}
|
|
3
|
-
-- Stack: Next.js + Supabase
|
|
4
|
-
-- Generated by MORPH Framework
|
|
5
|
-
-- Date: {{DATE}}
|
|
6
|
-
-- ============================================================
|
|
7
|
-
|
|
8
|
-
-- Enable RLS on table
|
|
9
|
-
alter table public.{{TABLE_NAME}} enable row level security;
|
|
10
|
-
|
|
11
|
-
-- Force RLS for table owner (prevents bypassing)
|
|
12
|
-
alter table public.{{TABLE_NAME}} force row level security;
|
|
13
|
-
|
|
14
|
-
-- Policy: {{POLICY_NAME}}
|
|
15
|
-
-- Operation: {{OPERATION}} (SELECT | INSERT | UPDATE | DELETE | ALL)
|
|
16
|
-
|
|
17
|
-
-- SELECT policy: users can read their own rows
|
|
18
|
-
create policy "{{POLICY_NAME}}_select"
|
|
19
|
-
on public.{{TABLE_NAME}}
|
|
20
|
-
for select
|
|
21
|
-
using ({{USING_EXPRESSION}});
|
|
22
|
-
|
|
23
|
-
-- INSERT policy: users can insert rows for themselves
|
|
24
|
-
create policy "{{POLICY_NAME}}_insert"
|
|
25
|
-
on public.{{TABLE_NAME}}
|
|
26
|
-
for insert
|
|
27
|
-
with check ({{WITH_CHECK_EXPRESSION}});
|
|
28
|
-
|
|
29
|
-
-- UPDATE policy: users can update their own rows
|
|
30
|
-
create policy "{{POLICY_NAME}}_update"
|
|
31
|
-
on public.{{TABLE_NAME}}
|
|
32
|
-
for update
|
|
33
|
-
using ({{USING_EXPRESSION}})
|
|
34
|
-
with check ({{WITH_CHECK_EXPRESSION}});
|
|
35
|
-
|
|
36
|
-
-- DELETE policy: users can delete their own rows
|
|
37
|
-
create policy "{{POLICY_NAME}}_delete"
|
|
38
|
-
on public.{{TABLE_NAME}}
|
|
39
|
-
for delete
|
|
40
|
-
using ({{USING_EXPRESSION}});
|
|
41
|
-
|
|
42
|
-
-- ============================================================
|
|
43
|
-
-- Common USING / WITH CHECK expressions:
|
|
44
|
-
--
|
|
45
|
-
-- Owner-based:
|
|
46
|
-
-- auth.uid() = user_id
|
|
47
|
-
--
|
|
48
|
-
-- Role-based:
|
|
49
|
-
-- auth.jwt() ->> 'role' = 'admin'
|
|
50
|
-
--
|
|
51
|
-
-- Organization-based:
|
|
52
|
-
-- org_id in (select org_id from public.org_members where user_id = auth.uid())
|
|
53
|
-
--
|
|
54
|
-
-- Public read, owner write:
|
|
55
|
-
-- SELECT using (true)
|
|
56
|
-
-- INSERT/UPDATE/DELETE using (auth.uid() = user_id)
|
|
57
|
-
-- ============================================================
|