@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,252 +0,0 @@
|
|
|
1
|
-
# Supabase MCP Server Setup Guide
|
|
2
|
-
|
|
3
|
-
> **Scope:** nextjs-supabase
|
|
4
|
-
> **Layer:** 2 (on keyword)
|
|
5
|
-
> **Keywords:** supabase, mcp, setup, model context protocol
|
|
6
|
-
> **Load When:** supabase mcp setup keywords detected
|
|
7
|
-
|
|
8
|
-
Direct database management from Claude Code via Model Context Protocol
|
|
9
|
-
|
|
10
|
-
## What It Provides
|
|
11
|
-
|
|
12
|
-
The Supabase MCP Server lets Claude Code interact directly with your Supabase project's database. This enables:
|
|
13
|
-
|
|
14
|
-
- Running SQL queries and viewing results
|
|
15
|
-
- Creating and managing database migrations
|
|
16
|
-
- Inspecting table schemas, indexes, and RLS policies
|
|
17
|
-
- Managing Supabase Auth, Storage, and Edge Functions
|
|
18
|
-
- Schema diffing and migration generation
|
|
19
|
-
|
|
20
|
-
## CRITICAL WARNING
|
|
21
|
-
|
|
22
|
-
```
|
|
23
|
-
==========================================================
|
|
24
|
-
NEVER connect to a PRODUCTION Supabase project via MCP.
|
|
25
|
-
|
|
26
|
-
MCP provides direct database access. A single bad query
|
|
27
|
-
can DROP tables, DELETE data, or DISABLE RLS.
|
|
28
|
-
|
|
29
|
-
ONLY connect to:
|
|
30
|
-
- Local Supabase (supabase start)
|
|
31
|
-
- Development/staging projects
|
|
32
|
-
- Disposable test projects
|
|
33
|
-
==========================================================
|
|
34
|
-
```
|
|
35
|
-
|
|
36
|
-
## Setup
|
|
37
|
-
|
|
38
|
-
### 1. Get Your Supabase Connection Details
|
|
39
|
-
|
|
40
|
-
From your Supabase Dashboard > Project Settings > API:
|
|
41
|
-
- **Project URL**: `https://xxxxxxxxxxxx.supabase.co`
|
|
42
|
-
- **Service Role Key**: `eyJhbGc...` (Settings > API > service_role key)
|
|
43
|
-
|
|
44
|
-
For local Supabase (`supabase start`):
|
|
45
|
-
- **Project URL**: `http://127.0.0.1:54321`
|
|
46
|
-
- **Service Role Key**: Output from `supabase status`
|
|
47
|
-
|
|
48
|
-
### 2. Configure Claude Desktop / Claude Code
|
|
49
|
-
|
|
50
|
-
Add the Supabase MCP server to your configuration.
|
|
51
|
-
|
|
52
|
-
**Claude Desktop** (`claude_desktop_config.json`):
|
|
53
|
-
|
|
54
|
-
```json
|
|
55
|
-
{
|
|
56
|
-
"mcpServers": {
|
|
57
|
-
"supabase": {
|
|
58
|
-
"command": "npx",
|
|
59
|
-
"args": [
|
|
60
|
-
"-y",
|
|
61
|
-
"@supabase/mcp-server-supabase@latest",
|
|
62
|
-
"--supabase-url",
|
|
63
|
-
"https://xxxxxxxxxxxx.supabase.co",
|
|
64
|
-
"--supabase-service-role-key",
|
|
65
|
-
"eyJhbGc...",
|
|
66
|
-
"--read-only"
|
|
67
|
-
]
|
|
68
|
-
}
|
|
69
|
-
}
|
|
70
|
-
}
|
|
71
|
-
```
|
|
72
|
-
|
|
73
|
-
**Claude Code** (`.claude/settings.local.json`):
|
|
74
|
-
|
|
75
|
-
```json
|
|
76
|
-
{
|
|
77
|
-
"mcpServers": {
|
|
78
|
-
"supabase": {
|
|
79
|
-
"command": "npx",
|
|
80
|
-
"args": [
|
|
81
|
-
"-y",
|
|
82
|
-
"@supabase/mcp-server-supabase@latest",
|
|
83
|
-
"--supabase-url",
|
|
84
|
-
"https://xxxxxxxxxxxx.supabase.co",
|
|
85
|
-
"--supabase-service-role-key",
|
|
86
|
-
"eyJhbGc...",
|
|
87
|
-
"--read-only"
|
|
88
|
-
]
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
```
|
|
93
|
-
|
|
94
|
-
### 3. Default: Read-Only Mode
|
|
95
|
-
|
|
96
|
-
The `--read-only` flag is included by default. In this mode:
|
|
97
|
-
|
|
98
|
-
| Operation | Allowed |
|
|
99
|
-
|-----------|---------|
|
|
100
|
-
| SELECT queries | Yes |
|
|
101
|
-
| DESCRIBE / schema inspection | Yes |
|
|
102
|
-
| LIST tables, policies, indexes | Yes |
|
|
103
|
-
| INSERT / UPDATE / DELETE | No |
|
|
104
|
-
| CREATE / ALTER / DROP | No |
|
|
105
|
-
| Migration creation | No |
|
|
106
|
-
|
|
107
|
-
To enable write access (development only), remove `--read-only`:
|
|
108
|
-
|
|
109
|
-
```json
|
|
110
|
-
{
|
|
111
|
-
"mcpServers": {
|
|
112
|
-
"supabase": {
|
|
113
|
-
"command": "npx",
|
|
114
|
-
"args": [
|
|
115
|
-
"-y",
|
|
116
|
-
"@supabase/mcp-server-supabase@latest",
|
|
117
|
-
"--supabase-url",
|
|
118
|
-
"http://127.0.0.1:54321",
|
|
119
|
-
"--supabase-service-role-key",
|
|
120
|
-
"your-local-service-role-key"
|
|
121
|
-
]
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
}
|
|
125
|
-
```
|
|
126
|
-
|
|
127
|
-
Write mode should ONLY be used with local Supabase or disposable dev projects.
|
|
128
|
-
|
|
129
|
-
## Available Tool Groups
|
|
130
|
-
|
|
131
|
-
The Supabase MCP Server exposes these tool groups:
|
|
132
|
-
|
|
133
|
-
### Database Tools
|
|
134
|
-
|
|
135
|
-
| Tool | Description |
|
|
136
|
-
|------|-------------|
|
|
137
|
-
| `list_tables` | List all tables in the project |
|
|
138
|
-
| `list_extensions` | List installed PostgreSQL extensions |
|
|
139
|
-
| `list_migrations` | List applied migrations |
|
|
140
|
-
| `apply_migration` | Apply a new SQL migration |
|
|
141
|
-
| `execute_sql` | Run arbitrary SQL (read-only by default) |
|
|
142
|
-
| `get_migration_status` | Check pending vs applied migrations |
|
|
143
|
-
|
|
144
|
-
### Auth Tools
|
|
145
|
-
|
|
146
|
-
| Tool | Description |
|
|
147
|
-
|------|-------------|
|
|
148
|
-
| `get_auth_config` | View auth provider configuration |
|
|
149
|
-
| `list_auth_users` | List authenticated users |
|
|
150
|
-
|
|
151
|
-
### Storage Tools
|
|
152
|
-
|
|
153
|
-
| Tool | Description |
|
|
154
|
-
|------|-------------|
|
|
155
|
-
| `list_storage_buckets` | List storage buckets |
|
|
156
|
-
| `get_storage_config` | View storage configuration |
|
|
157
|
-
|
|
158
|
-
### Project Tools
|
|
159
|
-
|
|
160
|
-
| Tool | Description |
|
|
161
|
-
|------|-------------|
|
|
162
|
-
| `get_project_config` | View project settings |
|
|
163
|
-
| `get_project_api_keys` | View API keys |
|
|
164
|
-
|
|
165
|
-
## Usage for Migrations and Schema Management
|
|
166
|
-
|
|
167
|
-
### Inspecting Current Schema
|
|
168
|
-
|
|
169
|
-
Ask Claude Code to use the MCP tools:
|
|
170
|
-
|
|
171
|
-
```
|
|
172
|
-
"Show me all tables and their RLS policies"
|
|
173
|
-
"What indexes exist on the documents table?"
|
|
174
|
-
"List all migrations applied so far"
|
|
175
|
-
```
|
|
176
|
-
|
|
177
|
-
### Creating Migrations
|
|
178
|
-
|
|
179
|
-
With write mode enabled on a local/dev project:
|
|
180
|
-
|
|
181
|
-
```
|
|
182
|
-
"Create a migration to add a 'tags' column to the documents table"
|
|
183
|
-
"Add an RLS policy so only tenant admins can delete documents"
|
|
184
|
-
"Enable the pgvector extension and create the embedding column"
|
|
185
|
-
```
|
|
186
|
-
|
|
187
|
-
The MCP server will generate and apply SQL migrations through Supabase's migration system.
|
|
188
|
-
|
|
189
|
-
### Schema Diffing
|
|
190
|
-
|
|
191
|
-
```
|
|
192
|
-
"Compare the current schema against the migration files"
|
|
193
|
-
"Show me what would change if I apply the pending migration"
|
|
194
|
-
```
|
|
195
|
-
|
|
196
|
-
## Usage with MORPH-SPEC Workflow
|
|
197
|
-
|
|
198
|
-
During MORPH-SPEC phases, the MCP server is useful for:
|
|
199
|
-
|
|
200
|
-
| Phase | MCP Usage |
|
|
201
|
-
|-------|-----------|
|
|
202
|
-
| **Phase 2 (Design)** | Inspect existing schema to inform spec |
|
|
203
|
-
| **Phase 4 (Tasks)** | Verify migration dependencies |
|
|
204
|
-
| **Phase 5 (Implement)** | Apply migrations, verify RLS policies, test queries |
|
|
205
|
-
|
|
206
|
-
### Example Workflow
|
|
207
|
-
|
|
208
|
-
```
|
|
209
|
-
1. Design phase: "List current tables and extensions"
|
|
210
|
-
2. Write migration SQL in supabase/migrations/
|
|
211
|
-
3. Apply via MCP: "Apply migration 003_add_search_logs.sql"
|
|
212
|
-
4. Verify: "Show RLS policies on search_logs table"
|
|
213
|
-
5. Test: "Run SELECT * FROM search_logs LIMIT 5"
|
|
214
|
-
```
|
|
215
|
-
|
|
216
|
-
## Security Best Practices
|
|
217
|
-
|
|
218
|
-
1. **Always use `--read-only` unless actively running migrations**
|
|
219
|
-
2. **Never store service_role keys in version control** -- use environment variables or a secrets manager
|
|
220
|
-
3. **Use local Supabase for development** (`supabase start`) instead of cloud projects
|
|
221
|
-
4. **Rotate keys if accidentally exposed** -- Supabase Dashboard > Settings > API > Regenerate
|
|
222
|
-
5. **Restrict MCP to development machines only** -- never configure on CI/CD or shared servers
|
|
223
|
-
6. **Review SQL before applying** -- the MCP server executes SQL with service_role privileges, bypassing all RLS
|
|
224
|
-
|
|
225
|
-
### Environment Variable Alternative
|
|
226
|
-
|
|
227
|
-
Instead of hardcoding keys in the config file:
|
|
228
|
-
|
|
229
|
-
```json
|
|
230
|
-
{
|
|
231
|
-
"mcpServers": {
|
|
232
|
-
"supabase": {
|
|
233
|
-
"command": "npx",
|
|
234
|
-
"args": [
|
|
235
|
-
"-y",
|
|
236
|
-
"@supabase/mcp-server-supabase@latest",
|
|
237
|
-
"--read-only"
|
|
238
|
-
],
|
|
239
|
-
"env": {
|
|
240
|
-
"SUPABASE_URL": "${SUPABASE_URL}",
|
|
241
|
-
"SUPABASE_SERVICE_ROLE_KEY": "${SUPABASE_SERVICE_ROLE_KEY}"
|
|
242
|
-
}
|
|
243
|
-
}
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
```
|
|
247
|
-
|
|
248
|
-
Set the environment variables in your shell profile or `.env` file (never committed).
|
|
249
|
-
|
|
250
|
-
---
|
|
251
|
-
|
|
252
|
-
*MORPH-SPEC by Polymorphism Tech*
|
|
@@ -1,176 +0,0 @@
|
|
|
1
|
-
# Supabase Authentication Standard
|
|
2
|
-
|
|
3
|
-
> **Scope:** nextjs-supabase
|
|
4
|
-
> **Layer:** 2 (on keyword)
|
|
5
|
-
> **Keywords:** supabase, auth, rls, user, authentication
|
|
6
|
-
> **Load When:** supabase auth keywords detected
|
|
7
|
-
|
|
8
|
-
Stack: Next.js 15 + Supabase + .NET Backend
|
|
9
|
-
|
|
10
|
-
## Core Rules
|
|
11
|
-
|
|
12
|
-
- NEVER use `supabase.auth.getSession()` on server -- reads from cookies without validation
|
|
13
|
-
- ALWAYS use `supabase.auth.getUser()` on server -- validates JWT with Supabase
|
|
14
|
-
- NEVER expose `service_role` key on frontend -- bypasses RLS
|
|
15
|
-
- ALWAYS use `@supabase/ssr` for Next.js -- not `@supabase/auth-helpers-nextjs` (deprecated)
|
|
16
|
-
- ALWAYS use PKCE flow for SSR auth
|
|
17
|
-
|
|
18
|
-
## Client Setup
|
|
19
|
-
|
|
20
|
-
### Browser Client
|
|
21
|
-
|
|
22
|
-
```ts
|
|
23
|
-
// lib/supabase/client.ts
|
|
24
|
-
import { createBrowserClient } from "@supabase/ssr";
|
|
25
|
-
|
|
26
|
-
export function createClient() {
|
|
27
|
-
return createBrowserClient(
|
|
28
|
-
process.env.NEXT_PUBLIC_SUPABASE_URL!,
|
|
29
|
-
process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!
|
|
30
|
-
);
|
|
31
|
-
}
|
|
32
|
-
```
|
|
33
|
-
|
|
34
|
-
### Server Client
|
|
35
|
-
|
|
36
|
-
```ts
|
|
37
|
-
// lib/supabase/server.ts
|
|
38
|
-
import { createServerClient } from "@supabase/ssr";
|
|
39
|
-
import { cookies } from "next/headers";
|
|
40
|
-
|
|
41
|
-
export async function createClient() {
|
|
42
|
-
const cookieStore = await cookies();
|
|
43
|
-
return createServerClient(
|
|
44
|
-
process.env.NEXT_PUBLIC_SUPABASE_URL!,
|
|
45
|
-
process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!,
|
|
46
|
-
{
|
|
47
|
-
cookies: {
|
|
48
|
-
getAll() { return cookieStore.getAll(); },
|
|
49
|
-
setAll(cookiesToSet) {
|
|
50
|
-
cookiesToSet.forEach(({ name, value, options }) =>
|
|
51
|
-
cookieStore.set(name, value, options));
|
|
52
|
-
},
|
|
53
|
-
},
|
|
54
|
-
}
|
|
55
|
-
);
|
|
56
|
-
}
|
|
57
|
-
```
|
|
58
|
-
|
|
59
|
-
## Auth Flows
|
|
60
|
-
|
|
61
|
-
```ts
|
|
62
|
-
// Email/Password sign up
|
|
63
|
-
await supabase.auth.signUp({ email, password,
|
|
64
|
-
options: { emailRedirectTo: `${origin}/auth/callback` } });
|
|
65
|
-
|
|
66
|
-
// Email/Password sign in
|
|
67
|
-
await supabase.auth.signInWithPassword({ email, password });
|
|
68
|
-
|
|
69
|
-
// OAuth (Google / GitHub)
|
|
70
|
-
await supabase.auth.signInWithOAuth({
|
|
71
|
-
provider: "google", // or "github"
|
|
72
|
-
options: { redirectTo: `${origin}/auth/callback`,
|
|
73
|
-
queryParams: { access_type: "offline", prompt: "consent" } } // Google-specific
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
// Magic Link
|
|
77
|
-
await supabase.auth.signInWithOtp({ email,
|
|
78
|
-
options: { emailRedirectTo: `${origin}/auth/callback` } });
|
|
79
|
-
```
|
|
80
|
-
|
|
81
|
-
## Auth Callback Route (PKCE)
|
|
82
|
-
|
|
83
|
-
```ts
|
|
84
|
-
// app/auth/callback/route.ts
|
|
85
|
-
import { createClient } from "@/lib/supabase/server";
|
|
86
|
-
import { NextResponse } from "next/server";
|
|
87
|
-
|
|
88
|
-
export async function GET(request: Request) {
|
|
89
|
-
const { searchParams, origin } = new URL(request.url);
|
|
90
|
-
const code = searchParams.get("code");
|
|
91
|
-
const next = searchParams.get("next") ?? "/dashboard";
|
|
92
|
-
if (code) {
|
|
93
|
-
const supabase = await createClient();
|
|
94
|
-
const { error } = await supabase.auth.exchangeCodeForSession(code);
|
|
95
|
-
if (!error) return NextResponse.redirect(`${origin}${next}`);
|
|
96
|
-
}
|
|
97
|
-
return NextResponse.redirect(`${origin}/auth/error`);
|
|
98
|
-
}
|
|
99
|
-
```
|
|
100
|
-
|
|
101
|
-
## Middleware Pattern
|
|
102
|
-
|
|
103
|
-
```ts
|
|
104
|
-
// middleware.ts
|
|
105
|
-
import { createServerClient } from "@supabase/ssr";
|
|
106
|
-
import { NextResponse, type NextRequest } from "next/server";
|
|
107
|
-
|
|
108
|
-
export async function middleware(request: NextRequest) {
|
|
109
|
-
let supabaseResponse = NextResponse.next({ request });
|
|
110
|
-
const supabase = createServerClient(
|
|
111
|
-
process.env.NEXT_PUBLIC_SUPABASE_URL!,
|
|
112
|
-
process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!,
|
|
113
|
-
{
|
|
114
|
-
cookies: {
|
|
115
|
-
getAll() { return request.cookies.getAll(); },
|
|
116
|
-
setAll(cookiesToSet) {
|
|
117
|
-
cookiesToSet.forEach(({ name, value, options }) => {
|
|
118
|
-
request.cookies.set(name, value);
|
|
119
|
-
supabaseResponse.cookies.set(name, value, options);
|
|
120
|
-
});
|
|
121
|
-
},
|
|
122
|
-
},
|
|
123
|
-
}
|
|
124
|
-
);
|
|
125
|
-
const { data: { user } } = await supabase.auth.getUser();
|
|
126
|
-
if (!user && request.nextUrl.pathname.startsWith("/dashboard"))
|
|
127
|
-
return NextResponse.redirect(new URL("/login", request.url));
|
|
128
|
-
return supabaseResponse;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
export const config = {
|
|
132
|
-
matcher: ["/((?!_next/static|_next/image|favicon.ico|api/webhooks).*)"],
|
|
133
|
-
};
|
|
134
|
-
```
|
|
135
|
-
|
|
136
|
-
## .NET JWT Validation
|
|
137
|
-
|
|
138
|
-
```csharp
|
|
139
|
-
builder.Services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
|
|
140
|
-
.AddJwtBearer(options => {
|
|
141
|
-
options.TokenValidationParameters = new TokenValidationParameters {
|
|
142
|
-
ValidateIssuer = true,
|
|
143
|
-
ValidIssuer = $"https://{supabaseProjectRef}.supabase.co/auth/v1",
|
|
144
|
-
ValidateAudience = true,
|
|
145
|
-
ValidAudience = "authenticated",
|
|
146
|
-
ValidateIssuerSigningKey = true,
|
|
147
|
-
IssuerSigningKey = new SymmetricSecurityKey(
|
|
148
|
-
Encoding.UTF8.GetBytes(supabaseJwtSecret)),
|
|
149
|
-
ValidateLifetime = true,
|
|
150
|
-
ClockSkew = TimeSpan.FromSeconds(30)
|
|
151
|
-
};
|
|
152
|
-
});
|
|
153
|
-
|
|
154
|
-
// Extract user ID: maps to auth.uid()
|
|
155
|
-
var userId = User.FindFirstValue(ClaimTypes.NameIdentifier);
|
|
156
|
-
```
|
|
157
|
-
|
|
158
|
-
## Environment Variables
|
|
159
|
-
|
|
160
|
-
| Variable | Where | Purpose |
|
|
161
|
-
|----------|-------|---------|
|
|
162
|
-
| `NEXT_PUBLIC_SUPABASE_URL` | Frontend | Supabase project URL |
|
|
163
|
-
| `NEXT_PUBLIC_SUPABASE_ANON_KEY` | Frontend | Public anon key (respects RLS) |
|
|
164
|
-
| `SUPABASE_SERVICE_ROLE_KEY` | Backend ONLY | Bypasses RLS -- NEVER on frontend |
|
|
165
|
-
| `SUPABASE_JWT_SECRET` | Backend ONLY | JWT validation secret |
|
|
166
|
-
|
|
167
|
-
## Common Mistakes
|
|
168
|
-
|
|
169
|
-
| Wrong | Right | Why |
|
|
170
|
-
|-------|-------|-----|
|
|
171
|
-
| `getSession()` on server | `getUser()` on server | getSession reads unvalidated cookie data |
|
|
172
|
-
| `@supabase/auth-helpers-nextjs` | `@supabase/ssr` | auth-helpers is deprecated |
|
|
173
|
-
| `service_role` in `NEXT_PUBLIC_*` | `anon` key in `NEXT_PUBLIC_*` | service_role bypasses all RLS |
|
|
174
|
-
| Implicit flow for SSR | PKCE flow with code exchange | Implicit exposes tokens in URL fragments |
|
|
175
|
-
| Auth only in page components | Auth check in middleware.ts | Middleware prevents flash of content |
|
|
176
|
-
| Missing `setAll` in cookie config | Both `getAll` and `setAll` | Session refresh silently fails without setAll |
|
|
@@ -1,169 +0,0 @@
|
|
|
1
|
-
# Supabase pgvector Standard
|
|
2
|
-
|
|
3
|
-
> **Scope:** nextjs-supabase
|
|
4
|
-
> **Layer:** 2 (on keyword)
|
|
5
|
-
> **Keywords:** supabase, pgvector, embedding, similarity, vector search
|
|
6
|
-
> **Load When:** supabase pgvector keywords detected
|
|
7
|
-
|
|
8
|
-
Stack: Next.js 15 + Supabase + .NET Backend
|
|
9
|
-
|
|
10
|
-
## Core Rules
|
|
11
|
-
|
|
12
|
-
- ALWAYS use HNSW indexes for production (faster queries, no training required)
|
|
13
|
-
- ALWAYS match dimensions to embedding model (e.g., 1536 for text-embedding-3-small)
|
|
14
|
-
- NEVER store embeddings without an index -- full table scan at query time
|
|
15
|
-
- Use `halfvec` for large datasets to halve storage (16-bit vs 32-bit per dimension)
|
|
16
|
-
- ALWAYS use RLS on tables containing embeddings
|
|
17
|
-
|
|
18
|
-
## Setup and Table Design
|
|
19
|
-
|
|
20
|
-
```sql
|
|
21
|
-
CREATE EXTENSION IF NOT EXISTS vector;
|
|
22
|
-
|
|
23
|
-
CREATE TABLE documents (
|
|
24
|
-
id UUID PRIMARY KEY DEFAULT gen_random_uuid(),
|
|
25
|
-
user_id UUID NOT NULL REFERENCES auth.users(id),
|
|
26
|
-
title TEXT NOT NULL,
|
|
27
|
-
content TEXT NOT NULL,
|
|
28
|
-
metadata JSONB DEFAULT '{}',
|
|
29
|
-
embedding vector(1536),
|
|
30
|
-
created_at TIMESTAMPTZ DEFAULT now()
|
|
31
|
-
);
|
|
32
|
-
|
|
33
|
-
ALTER TABLE documents ENABLE ROW LEVEL SECURITY;
|
|
34
|
-
CREATE POLICY "owner_access" ON documents FOR ALL
|
|
35
|
-
USING (user_id = auth.uid()) WITH CHECK (user_id = auth.uid());
|
|
36
|
-
CREATE INDEX idx_documents_user_id ON documents (user_id);
|
|
37
|
-
```
|
|
38
|
-
|
|
39
|
-
### halfvec Optimization
|
|
40
|
-
|
|
41
|
-
| Type | Storage/dim | 1536-dim | Best for |
|
|
42
|
-
|------|------------|----------|----------|
|
|
43
|
-
| `vector` | 4 bytes | 6 KB | High precision, small datasets |
|
|
44
|
-
| `halfvec` | 2 bytes | 3 KB | Large datasets, cost optimization |
|
|
45
|
-
|
|
46
|
-
## Index Types
|
|
47
|
-
|
|
48
|
-
```sql
|
|
49
|
-
-- HNSW (recommended)
|
|
50
|
-
CREATE INDEX idx_docs_embedding ON documents
|
|
51
|
-
USING hnsw (embedding vector_cosine_ops) WITH (m = 16, ef_construction = 64);
|
|
52
|
-
|
|
53
|
-
-- IVFFlat (legacy, requires existing data)
|
|
54
|
-
CREATE INDEX idx_docs_ivf ON documents
|
|
55
|
-
USING ivfflat (embedding vector_cosine_ops) WITH (lists = 100);
|
|
56
|
-
```
|
|
57
|
-
|
|
58
|
-
| Feature | HNSW | IVFFlat |
|
|
59
|
-
|---------|------|---------|
|
|
60
|
-
| Query speed | Faster | Slower |
|
|
61
|
-
| Requires training | No | Yes |
|
|
62
|
-
| Recall quality | Higher | Lower |
|
|
63
|
-
| Recommended | Yes | Only for very large datasets |
|
|
64
|
-
|
|
65
|
-
## HNSW Parameters
|
|
66
|
-
|
|
67
|
-
| Parameter | Default | Tuning |
|
|
68
|
-
|-----------|---------|--------|
|
|
69
|
-
| `m` | 16 | Higher = better recall, more memory |
|
|
70
|
-
| `ef_construction` | 64 | Higher = better index, slower build |
|
|
71
|
-
| `ef_search` | 40 | `SET hnsw.ef_search = 100;` per session |
|
|
72
|
-
|
|
73
|
-
## Distance Functions
|
|
74
|
-
|
|
75
|
-
| Operator | Function | Index Ops | Use Case |
|
|
76
|
-
|----------|----------|-----------|----------|
|
|
77
|
-
| `<=>` | Cosine distance | `vector_cosine_ops` | Normalized embeddings (most common) |
|
|
78
|
-
| `<->` | L2 (Euclidean) | `vector_l2_ops` | Spatial/positional data |
|
|
79
|
-
| `<#>` | Inner product (neg) | `vector_ip_ops` | Pre-normalized, max similarity |
|
|
80
|
-
|
|
81
|
-
## Similarity Search
|
|
82
|
-
|
|
83
|
-
```sql
|
|
84
|
-
CREATE OR REPLACE FUNCTION match_documents(
|
|
85
|
-
query_embedding vector(1536),
|
|
86
|
-
match_threshold float DEFAULT 0.78,
|
|
87
|
-
match_count int DEFAULT 10,
|
|
88
|
-
p_user_id uuid DEFAULT auth.uid()
|
|
89
|
-
) RETURNS TABLE (id uuid, title text, content text, similarity float)
|
|
90
|
-
LANGUAGE sql STABLE AS $$
|
|
91
|
-
SELECT d.id, d.title, d.content,
|
|
92
|
-
1 - (d.embedding <=> query_embedding) AS similarity
|
|
93
|
-
FROM documents d
|
|
94
|
-
WHERE d.user_id = p_user_id
|
|
95
|
-
AND 1 - (d.embedding <=> query_embedding) > match_threshold
|
|
96
|
-
ORDER BY d.embedding <=> query_embedding
|
|
97
|
-
LIMIT match_count;
|
|
98
|
-
$$;
|
|
99
|
-
```
|
|
100
|
-
|
|
101
|
-
## Hybrid Search (Vector + Full-Text)
|
|
102
|
-
|
|
103
|
-
```sql
|
|
104
|
-
CREATE OR REPLACE FUNCTION hybrid_search(
|
|
105
|
-
query_text text, query_embedding vector(1536),
|
|
106
|
-
match_count int DEFAULT 10,
|
|
107
|
-
text_weight float DEFAULT 0.3, vector_weight float DEFAULT 0.7
|
|
108
|
-
) RETURNS TABLE (id uuid, title text, content text, score float)
|
|
109
|
-
LANGUAGE sql STABLE AS $$
|
|
110
|
-
WITH vector_results AS (
|
|
111
|
-
SELECT id, title, content,
|
|
112
|
-
1 - (embedding <=> query_embedding) AS vector_score
|
|
113
|
-
FROM documents WHERE user_id = auth.uid()
|
|
114
|
-
ORDER BY embedding <=> query_embedding LIMIT match_count * 2
|
|
115
|
-
),
|
|
116
|
-
text_results AS (
|
|
117
|
-
SELECT id, title, content,
|
|
118
|
-
ts_rank(to_tsvector('english', content), plainto_tsquery('english', query_text)) AS text_score
|
|
119
|
-
FROM documents WHERE user_id = auth.uid()
|
|
120
|
-
AND to_tsvector('english', content) @@ plainto_tsquery('english', query_text)
|
|
121
|
-
LIMIT match_count * 2
|
|
122
|
-
)
|
|
123
|
-
SELECT COALESCE(v.id, t.id), COALESCE(v.title, t.title), COALESCE(v.content, t.content),
|
|
124
|
-
(COALESCE(v.vector_score, 0) * vector_weight + COALESCE(t.text_score, 0) * text_weight)
|
|
125
|
-
FROM vector_results v FULL OUTER JOIN text_results t ON v.id = t.id
|
|
126
|
-
ORDER BY score DESC LIMIT match_count;
|
|
127
|
-
$$;
|
|
128
|
-
```
|
|
129
|
-
|
|
130
|
-
## .NET Integration (Npgsql)
|
|
131
|
-
|
|
132
|
-
```csharp
|
|
133
|
-
public sealed class DocumentRepository(AppDbContext db)
|
|
134
|
-
{
|
|
135
|
-
public async Task StoreEmbeddingAsync(
|
|
136
|
-
Guid documentId, float[] embedding, CancellationToken ct = default)
|
|
137
|
-
{
|
|
138
|
-
await db.Database.ExecuteSqlInterpolatedAsync(
|
|
139
|
-
$"UPDATE documents SET embedding = {new Vector(embedding)} WHERE id = {documentId}", ct);
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
public async Task<List<DocumentMatch>> SearchSimilarAsync(
|
|
143
|
-
float[] queryEmbedding, int limit = 10, float threshold = 0.78f,
|
|
144
|
-
CancellationToken ct = default)
|
|
145
|
-
{
|
|
146
|
-
return await db.Database.SqlQuery<DocumentMatch>($"""
|
|
147
|
-
SELECT id, title, content,
|
|
148
|
-
1 - (embedding <=> {new Vector(queryEmbedding)}::vector) AS similarity
|
|
149
|
-
FROM documents
|
|
150
|
-
WHERE 1 - (embedding <=> {new Vector(queryEmbedding)}::vector) > {threshold}
|
|
151
|
-
ORDER BY embedding <=> {new Vector(queryEmbedding)}::vector LIMIT {limit}
|
|
152
|
-
""").ToListAsync(ct);
|
|
153
|
-
}
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
// EF Core registration
|
|
157
|
-
builder.Services.AddDbContext<AppDbContext>(o =>
|
|
158
|
-
o.UseNpgsql(connectionString, npg => npg.UseVector()));
|
|
159
|
-
```
|
|
160
|
-
|
|
161
|
-
## Common Mistakes
|
|
162
|
-
|
|
163
|
-
| Wrong | Right | Why |
|
|
164
|
-
|-------|-------|-----|
|
|
165
|
-
| No index on embedding column | HNSW index | Full table scan, extremely slow |
|
|
166
|
-
| `ORDER BY similarity DESC` | `ORDER BY embedding <=> query ASC` | Operator returns distance, not similarity |
|
|
167
|
-
| Mixing embedding dimensions | Consistent dimensions per column | Dimension mismatch causes runtime errors |
|
|
168
|
-
| Full-precision for millions of rows | `halfvec` for large datasets | 2x storage savings, minimal quality loss |
|
|
169
|
-
| Missing RLS on embedding tables | RLS with user/tenant policies | Embeddings contain sensitive content context |
|