@polymorphism-tech/morph-spec 4.7.0 → 4.7.2

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.
Files changed (232) hide show
  1. package/.morph/.morphversion +5 -0
  2. package/.morph/analytics/threads-log.jsonl +5 -0
  3. package/.morph/config/config.json +8 -0
  4. package/.morph/framework/agents.json +1815 -0
  5. package/.morph/framework/hooks/README.md +205 -0
  6. package/.morph/framework/hooks/claude-code/notification/approval-reminder.js +54 -0
  7. package/.morph/framework/hooks/claude-code/post-tool-use/dispatch.js +83 -0
  8. package/.morph/framework/hooks/claude-code/post-tool-use/handle-tool-failure.js +42 -0
  9. package/.morph/framework/hooks/claude-code/pre-compact/save-morph-context.js +61 -0
  10. package/.morph/framework/hooks/claude-code/pre-tool-use/enforce-phase-writes.js +71 -0
  11. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-readonly-files.js +58 -0
  12. package/.morph/framework/hooks/claude-code/pre-tool-use/protect-spec-files.js +64 -0
  13. package/.morph/framework/hooks/claude-code/session-start/inject-morph-context.js +94 -0
  14. package/.morph/framework/hooks/claude-code/statusline.py +538 -0
  15. package/.morph/framework/hooks/claude-code/statusline.sh +7 -0
  16. package/.morph/framework/hooks/claude-code/stop/validate-completion.js +88 -0
  17. package/.morph/framework/hooks/claude-code/user-prompt/enrich-prompt.js +91 -0
  18. package/.morph/framework/hooks/git/commit-msg/conventional-commits.sh +33 -0
  19. package/.morph/framework/hooks/git/pre-commit/agents.sh +25 -0
  20. package/.morph/framework/hooks/git/pre-commit/orchestrator.sh +64 -0
  21. package/.morph/framework/hooks/git/pre-commit/specs.sh +50 -0
  22. package/.morph/framework/hooks/git/pre-push/run-tests.sh +44 -0
  23. package/.morph/framework/hooks/shared/hook-response.js +45 -0
  24. package/.morph/framework/hooks/shared/phase-utils.js +129 -0
  25. package/.morph/framework/hooks/shared/state-reader.js +138 -0
  26. package/.morph/framework/hooks/shared/stdin-reader.js +26 -0
  27. package/.morph/framework/standards/STANDARDS.json +933 -0
  28. package/.morph/framework/standards/ai-agents/blazor-ui.md +364 -0
  29. package/.morph/framework/standards/ai-agents/production.md +415 -0
  30. package/.morph/framework/standards/ai-agents/setup.md +418 -0
  31. package/.morph/framework/standards/ai-agents/team-orchestration.md +479 -0
  32. package/.morph/framework/standards/ai-agents/workflows.md +354 -0
  33. package/.morph/framework/standards/architecture/ddd/aggregates.md +120 -0
  34. package/.morph/framework/standards/architecture/ddd/bounded-contexts.md +105 -0
  35. package/.morph/framework/standards/architecture/ddd/complexity-levels.md +108 -0
  36. package/.morph/framework/standards/architecture/ddd/entities.md +99 -0
  37. package/.morph/framework/standards/architecture/ddd/ubiquitous-language.md +58 -0
  38. package/.morph/framework/standards/architecture/ddd/value-objects.md +124 -0
  39. package/.morph/framework/standards/backend/api/minimal-api.md +494 -0
  40. package/.morph/framework/standards/backend/api/rest.md +492 -0
  41. package/.morph/framework/standards/backend/api/validation.md +88 -0
  42. package/.morph/framework/standards/backend/authentication/passkeys.md +428 -0
  43. package/.morph/framework/standards/backend/database/ef-core.md +199 -0
  44. package/.morph/framework/standards/backend/database/migrations.md +393 -0
  45. package/.morph/framework/standards/backend/database/postgresql/database.md +352 -0
  46. package/.morph/framework/standards/backend/database/repository-patterns.md +528 -0
  47. package/.morph/framework/standards/backend/database/vector-search-rag.md +541 -0
  48. package/.morph/framework/standards/backend/dotnet/async.md +366 -0
  49. package/.morph/framework/standards/backend/dotnet/core.md +117 -0
  50. package/.morph/framework/standards/backend/dotnet/di.md +439 -0
  51. package/.morph/framework/standards/backend/dotnet/program-cs-checklist.md +92 -0
  52. package/.morph/framework/standards/backend/integrations/asaas/asaas-api.md +216 -0
  53. package/.morph/framework/standards/backend/integrations/clerk/clerk-auth.md +290 -0
  54. package/.morph/framework/standards/backend/integrations/hangfire/hangfire-jobs.md +350 -0
  55. package/.morph/framework/standards/backend/integrations/resend/resend-email.md +385 -0
  56. package/.morph/framework/standards/context/analytics.md +96 -0
  57. package/.morph/framework/standards/context/bundles.md +110 -0
  58. package/.morph/framework/standards/context/priming.md +78 -0
  59. package/.morph/framework/standards/core/architecture.md +185 -0
  60. package/.morph/framework/standards/core/coding.md +214 -0
  61. package/.morph/framework/standards/core/git-branching-strategy.md +403 -0
  62. package/.morph/framework/standards/core/git.md +185 -0
  63. package/.morph/framework/standards/core/testing.md +295 -0
  64. package/.morph/framework/standards/data/nosql/blob-storage.md +102 -0
  65. package/.morph/framework/standards/data/nosql/cache/redis.md +97 -0
  66. package/.morph/framework/standards/data/nosql/cosmos-db.md +118 -0
  67. package/.morph/framework/standards/data/vector-search/azure-ai-search.md +121 -0
  68. package/.morph/framework/standards/data/vector-search/rag-chunking.md +104 -0
  69. package/.morph/framework/standards/frontend/blazor/design-checklist.md +222 -0
  70. package/.morph/framework/standards/frontend/blazor/fluent-ui-setup.md +595 -0
  71. package/.morph/framework/standards/frontend/blazor/fluent-ui.md +137 -0
  72. package/.morph/framework/standards/frontend/blazor/html-conversion.md +184 -0
  73. package/.morph/framework/standards/frontend/blazor/lifecycle.md +195 -0
  74. package/.morph/framework/standards/frontend/blazor/pitfalls.md +198 -0
  75. package/.morph/framework/standards/frontend/blazor/state.md +191 -0
  76. package/.morph/framework/standards/frontend/design-system/animations.md +151 -0
  77. package/.morph/framework/standards/frontend/design-system/naming.md +64 -0
  78. package/.morph/framework/standards/frontend/nextjs/app-router.md +123 -0
  79. package/.morph/framework/standards/frontend/nextjs/components.md +132 -0
  80. package/.morph/framework/standards/frontend/nextjs/data-fetching.md +126 -0
  81. package/.morph/framework/standards/frontend/nextjs/forms.md +128 -0
  82. package/.morph/framework/standards/frontend/nextjs/naming-conventions.md +67 -0
  83. package/.morph/framework/standards/frontend/nextjs/nextjs-patterns.md +215 -0
  84. package/.morph/framework/standards/frontend/nextjs/project-structure.md +102 -0
  85. package/.morph/framework/standards/frontend/nextjs/state-management.md +72 -0
  86. package/.morph/framework/standards/frontend/nextjs/testing.md +111 -0
  87. package/.morph/framework/standards/infrastructure/azure/azure.md +624 -0
  88. package/.morph/framework/standards/infrastructure/azure/bicep/bicep-patterns.md +422 -0
  89. package/.morph/framework/standards/infrastructure/azure/devops/azure-devops-setup.md +516 -0
  90. package/.morph/framework/standards/infrastructure/azure/devops/local-development.md +520 -0
  91. package/.morph/framework/standards/infrastructure/azure/services/functions.md +486 -0
  92. package/.morph/framework/standards/infrastructure/azure/services/service-bus.md +459 -0
  93. package/.morph/framework/standards/infrastructure/azure/services/storage.md +407 -0
  94. package/.morph/framework/standards/infrastructure/docker/easypanel-deploy.md +196 -0
  95. package/.morph/framework/standards/infrastructure/supabase/mcp-setup.md +252 -0
  96. package/.morph/framework/standards/infrastructure/supabase/supabase-auth.md +176 -0
  97. package/.morph/framework/standards/infrastructure/supabase/supabase-pgvector.md +169 -0
  98. package/.morph/framework/standards/infrastructure/supabase/supabase-rls.md +184 -0
  99. package/.morph/framework/standards/infrastructure/supabase/supabase-storage.md +153 -0
  100. package/.morph/framework/standards/integration/api/graphql.md +91 -0
  101. package/.morph/framework/standards/integration/api/grpc.md +114 -0
  102. package/.morph/framework/standards/integration/api/rest-design.md +95 -0
  103. package/.morph/framework/standards/integration/event-driven/cqrs.md +101 -0
  104. package/.morph/framework/standards/integration/event-driven/event-sourcing.md +124 -0
  105. package/.morph/framework/standards/integration/event-driven/service-bus.md +95 -0
  106. package/.morph/framework/standards/integration/mcp/mcp-tools.md +384 -0
  107. package/.morph/framework/standards/observability/logging.md +131 -0
  108. package/.morph/framework/standards/observability/metrics.md +121 -0
  109. package/.morph/framework/standards/observability/monitoring.md +114 -0
  110. package/.morph/framework/standards/observability/tracing.md +132 -0
  111. package/.morph/framework/standards/workflows/parallel-execution.md +112 -0
  112. package/.morph/framework/standards/workflows/thread-management.md +113 -0
  113. package/.morph/framework/templates/.idea/morph-templates.xml +92 -0
  114. package/.morph/framework/templates/.vscode/morph-templates.code-snippets +186 -0
  115. package/.morph/framework/templates/IDE-SNIPPETS.md +266 -0
  116. package/.morph/framework/templates/README.md +814 -0
  117. package/.morph/framework/templates/REGISTRY.json +1888 -0
  118. package/.morph/framework/templates/code/dotnet/backend/repository.cs +141 -0
  119. package/.morph/framework/templates/code/dotnet/backend/service.cs +139 -0
  120. package/.morph/framework/templates/code/dotnet/contracts/Commands.cs +74 -0
  121. package/.morph/framework/templates/code/dotnet/contracts/Entities.cs +25 -0
  122. package/.morph/framework/templates/code/dotnet/contracts/Queries.cs +74 -0
  123. package/.morph/framework/templates/code/dotnet/contracts/README.md +74 -0
  124. package/.morph/framework/templates/code/dotnet/contracts/api-contracts.cs +173 -0
  125. package/.morph/framework/templates/code/dotnet/contracts/contracts-level1.cs +69 -0
  126. package/.morph/framework/templates/code/dotnet/contracts/contracts-level2.cs +86 -0
  127. package/.morph/framework/templates/code/dotnet/contracts/contracts-level3.cs +41 -0
  128. package/.morph/framework/templates/code/dotnet/database/migration.cs +83 -0
  129. package/.morph/framework/templates/code/dotnet/frontend/component.razor +239 -0
  130. package/.morph/framework/templates/code/dotnet/jobs/agent.cs +163 -0
  131. package/.morph/framework/templates/code/dotnet/jobs/job.cs +171 -0
  132. package/.morph/framework/templates/code/dotnet/test.cs +239 -0
  133. package/.morph/framework/templates/code/sql/rls-policy.sql +57 -0
  134. package/.morph/framework/templates/code/sql/supabase-migration.sql +100 -0
  135. package/.morph/framework/templates/code/sql/supabase-migration.template.sql +113 -0
  136. package/.morph/framework/templates/code/typescript/contracts.ts +168 -0
  137. package/.morph/framework/templates/context/CONTEXT-FEATURE.md +276 -0
  138. package/.morph/framework/templates/context/CONTEXT.md +181 -0
  139. package/.morph/framework/templates/docs/clarifications.md +253 -0
  140. package/.morph/framework/templates/docs/onboarding.md +123 -0
  141. package/.morph/framework/templates/docs/proposal.md +182 -0
  142. package/.morph/framework/templates/docs/schema-analysis.md +119 -0
  143. package/.morph/framework/templates/docs/spec.md +198 -0
  144. package/.morph/framework/templates/docs/ui-components.md +124 -0
  145. package/.morph/framework/templates/docs/ui-design-system.md +76 -0
  146. package/.morph/framework/templates/docs/ui-flows.md +167 -0
  147. package/.morph/framework/templates/docs/ui-mockups.md +98 -0
  148. package/.morph/framework/templates/docs/user-stories.md +34 -0
  149. package/.morph/framework/templates/examples/design-system-examples.md +357 -0
  150. package/.morph/framework/templates/examples/spec-examples.md +90 -0
  151. package/.morph/framework/templates/feature/decisions.md +187 -0
  152. package/.morph/framework/templates/feature/recap.md +146 -0
  153. package/.morph/framework/templates/feature/tasks.md +199 -0
  154. package/.morph/framework/templates/frontend/nextjs/Dockerfile.nextjs.hbs +43 -0
  155. package/.morph/framework/templates/frontend/nextjs/client-component.tsx.hbs +26 -0
  156. package/.morph/framework/templates/frontend/nextjs/env.mjs.hbs +32 -0
  157. package/.morph/framework/templates/frontend/nextjs/feature-form.tsx.hbs +56 -0
  158. package/.morph/framework/templates/frontend/nextjs/page.tsx.hbs +22 -0
  159. package/.morph/framework/templates/frontend/nextjs/tsconfig.json.hbs +26 -0
  160. package/.morph/framework/templates/frontend/nextjs/use-feature.ts.hbs +54 -0
  161. package/.morph/framework/templates/infrastructure/azure/Dockerfile.example +82 -0
  162. package/.morph/framework/templates/infrastructure/azure/README.md +286 -0
  163. package/.morph/framework/templates/infrastructure/azure/app-insights.bicep +63 -0
  164. package/.morph/framework/templates/infrastructure/azure/app-service.bicep +164 -0
  165. package/.morph/framework/templates/infrastructure/azure/container-app-env.bicep +49 -0
  166. package/.morph/framework/templates/infrastructure/azure/container-app.bicep +156 -0
  167. package/.morph/framework/templates/infrastructure/azure/deploy-checklist.md +426 -0
  168. package/.morph/framework/templates/infrastructure/azure/deploy.ps1 +229 -0
  169. package/.morph/framework/templates/infrastructure/azure/deploy.sh +208 -0
  170. package/.morph/framework/templates/infrastructure/azure/key-vault.bicep +91 -0
  171. package/.morph/framework/templates/infrastructure/azure/main.bicep +189 -0
  172. package/.morph/framework/templates/infrastructure/azure/parameters.dev.json +29 -0
  173. package/.morph/framework/templates/infrastructure/azure/parameters.prod.json +29 -0
  174. package/.morph/framework/templates/infrastructure/azure/parameters.staging.json +29 -0
  175. package/.morph/framework/templates/infrastructure/azure/sql-database.bicep +103 -0
  176. package/.morph/framework/templates/infrastructure/azure/storage.bicep +106 -0
  177. package/.morph/framework/templates/infrastructure/docker/Dockerfile.template +58 -0
  178. package/.morph/framework/templates/infrastructure/docker/docker-compose.template.yml +67 -0
  179. package/.morph/framework/templates/infrastructure/docker/dockerfile-api.dockerfile +38 -0
  180. package/.morph/framework/templates/infrastructure/docker/dockerfile-web.dockerfile +48 -0
  181. package/.morph/framework/templates/infrastructure/docker/easypanel.template.json +54 -0
  182. package/.morph/framework/templates/infrastructure/github/README.md +593 -0
  183. package/.morph/framework/templates/infrastructure/github/actions/azure-auth/action.yml.hbs +22 -0
  184. package/.morph/framework/templates/infrastructure/github/actions/docker-build-push/action.yml.hbs +45 -0
  185. package/.morph/framework/templates/infrastructure/github/actions/health-check/action.yml.hbs +27 -0
  186. package/.morph/framework/templates/infrastructure/github/workflows/deploy-azure-app-service.yml.hbs +61 -0
  187. package/.morph/framework/templates/infrastructure/github/workflows/deploy-easypanel.yml.hbs +31 -0
  188. package/.morph/framework/templates/infrastructure/github/workflows/docker-build-push.yml.hbs +59 -0
  189. package/.morph/framework/templates/infrastructure/github/workflows/dotnet-build.yml.hbs +39 -0
  190. package/.morph/framework/templates/integrations/asaas-client.cs +387 -0
  191. package/.morph/framework/templates/integrations/asaas-webhook.cs +351 -0
  192. package/.morph/framework/templates/integrations/azure-identity-config.cs +288 -0
  193. package/.morph/framework/templates/integrations/clerk-config.cs +258 -0
  194. package/.morph/framework/templates/meta-prompts/fusion/fusion-agent.md +76 -0
  195. package/.morph/framework/templates/meta-prompts/fusion/fusion-aggregator.md +100 -0
  196. package/.morph/framework/templates/meta-prompts/hops/hop-retry.md +78 -0
  197. package/.morph/framework/templates/meta-prompts/hops/hop-validation.md +97 -0
  198. package/.morph/framework/templates/meta-prompts/hops/hop-wrapper.md +36 -0
  199. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-coordinator.md +113 -0
  200. package/.morph/framework/templates/meta-prompts/parallel-workers/parallel-worker.md +80 -0
  201. package/.morph/framework/templates/meta-prompts/squad-leaders/backend-squad.md +90 -0
  202. package/.morph/framework/templates/meta-prompts/squad-leaders/frontend-squad.md +126 -0
  203. package/.morph/framework/templates/meta-prompts/squad-leaders/squad-leader.md +43 -0
  204. package/.morph/framework/templates/meta-prompts/validators/checkpoint-validator.md +107 -0
  205. package/.morph/framework/templates/meta-prompts/validators/pre-commit-validator.md +95 -0
  206. package/.morph/framework/templates/project-structure/dotnet-ddd.md +70 -0
  207. package/.morph/framework/templates/saas/subscription.cs +347 -0
  208. package/.morph/framework/templates/saas/tenant.cs +338 -0
  209. package/.morph/framework/templates/state.template.json +17 -0
  210. package/.morph/framework/templates/ui/FluentDesignTheme.cs +149 -0
  211. package/.morph/framework/templates/ui/MudTheme.cs +281 -0
  212. package/.morph/framework/templates/ui/design-system.css +226 -0
  213. package/.morph/logs/tool-failures.log +17 -0
  214. package/.morph/memory/pre-compact-2026-02-24T17-43-30-049Z.json +16 -0
  215. package/.morph/plans/eager-watching-bunny.md +105 -0
  216. package/.morph/plans/temporal-seeking-nebula.md +45 -0
  217. package/.morph/state.json +48 -0
  218. package/CLAUDE.md +1 -1
  219. package/README.md +119 -99
  220. package/bin/morph-spec.js +0 -9
  221. package/framework/CLAUDE.md +1 -1
  222. package/framework/hooks/README.md +10 -6
  223. package/framework/hooks/claude-code/notification/approval-reminder.js +2 -0
  224. package/framework/hooks/claude-code/post-tool-use/dispatch.js +1 -1
  225. package/framework/hooks/claude-code/stop/validate-completion.js +1 -1
  226. package/framework/hooks/claude-code/user-prompt/enrich-prompt.js +1 -1
  227. package/package.json +1 -1
  228. package/src/commands/project/init.js +15 -42
  229. package/src/commands/project/update.js +22 -37
  230. package/src/lib/installers/mcp-installer.js +18 -3
  231. package/src/utils/hooks-installer.js +5 -15
  232. package/src/commands/project/detect.js +0 -114
@@ -0,0 +1,101 @@
1
+ # Integration Standard: CQRS with MediatR
2
+
3
+ ## Overview
4
+ Commands mutate state. Queries read state. Never mix them.
5
+
6
+ ## Setup
7
+ ```xml
8
+ <PackageReference Include="MediatR" Version="12.*" />
9
+ <PackageReference Include="MediatR.Extensions.Microsoft.DependencyInjection" Version="11.*" />
10
+ ```
11
+
12
+ ```csharp
13
+ // Program.cs
14
+ builder.Services.AddMediatR(cfg => cfg.RegisterServicesFromAssembly(typeof(Program).Assembly));
15
+ ```
16
+
17
+ ## Command Pattern
18
+ ```csharp
19
+ // Command (mutates state, returns result)
20
+ public record CreateOrderCommand(Guid UserId, List<OrderItem> Items)
21
+ : IRequest<CreateOrderResult>;
22
+
23
+ public record CreateOrderResult(Guid OrderId, decimal Total);
24
+
25
+ // Handler
26
+ public class CreateOrderHandler : IRequestHandler<CreateOrderCommand, CreateOrderResult>
27
+ {
28
+ public async Task<CreateOrderResult> Handle(CreateOrderCommand cmd, CancellationToken ct)
29
+ {
30
+ var order = Order.Create(cmd.UserId, cmd.Items);
31
+ await _repository.AddAsync(order, ct);
32
+ await _unitOfWork.SaveChangesAsync(ct);
33
+
34
+ // Publish domain event
35
+ await _publisher.Publish(new OrderCreatedEvent(order.Id), ct);
36
+
37
+ return new CreateOrderResult(order.Id, order.Total);
38
+ }
39
+ }
40
+ ```
41
+
42
+ ## Query Pattern
43
+ ```csharp
44
+ // Query (reads only, no side effects)
45
+ public record GetOrderQuery(Guid OrderId) : IRequest<OrderDto?>;
46
+
47
+ // Handler — use read model, not domain entity
48
+ public class GetOrderHandler : IRequestHandler<GetOrderQuery, OrderDto?>
49
+ {
50
+ public async Task<OrderDto?> Handle(GetOrderQuery query, CancellationToken ct)
51
+ {
52
+ return await _context.Orders
53
+ .Where(o => o.Id == query.OrderId)
54
+ .Select(o => new OrderDto(o.Id, o.Status, o.Total))
55
+ .FirstOrDefaultAsync(ct);
56
+ }
57
+ }
58
+ ```
59
+
60
+ ## Pipeline Behaviors (Cross-Cutting)
61
+ ```csharp
62
+ // Validation behavior
63
+ public class ValidationBehavior<TRequest, TResponse>
64
+ : IPipelineBehavior<TRequest, TResponse>
65
+ where TRequest : IRequest<TResponse>
66
+ {
67
+ public async Task<TResponse> Handle(TRequest request,
68
+ RequestHandlerDelegate<TResponse> next, CancellationToken ct)
69
+ {
70
+ var failures = _validators
71
+ .SelectMany(v => v.Validate(request).Errors)
72
+ .Where(f => f != null)
73
+ .ToList();
74
+
75
+ if (failures.Any())
76
+ throw new ValidationException(failures);
77
+
78
+ return await next();
79
+ }
80
+ }
81
+ ```
82
+
83
+ ## DI Registration Order
84
+ ```csharp
85
+ // 1. MediatR (includes handler scanning)
86
+ builder.Services.AddMediatR(cfg =>
87
+ {
88
+ cfg.RegisterServicesFromAssembly(Assembly.GetExecutingAssembly());
89
+ cfg.AddBehavior(typeof(IPipelineBehavior<,>), typeof(ValidationBehavior<,>));
90
+ cfg.AddBehavior(typeof(IPipelineBehavior<,>), typeof(PerformanceBehavior<,>));
91
+ cfg.AddBehavior(typeof(IPipelineBehavior<,>), typeof(LoggingBehavior<,>));
92
+ });
93
+ // 2. FluentValidation
94
+ builder.Services.AddValidatorsFromAssembly(Assembly.GetExecutingAssembly());
95
+ ```
96
+
97
+ ## Rules
98
+ - Commands return results, not void (enables error handling)
99
+ - Queries use DTOs, never expose domain entities
100
+ - Never call command handlers from query handlers
101
+ - One handler per Command/Query (no inheritance hierarchies)
@@ -0,0 +1,124 @@
1
+ # Integration Standard: Event Sourcing
2
+
3
+ ## Overview
4
+ State is derived from a sequence of immutable events. Use only when audit trail or temporal queries are required.
5
+
6
+ ## When to Use
7
+ ✅ Financial transactions (audit required by law)
8
+ ✅ Order lifecycle (multiple state transitions, replay needed)
9
+ ✅ Inventory changes (why did stock go negative?)
10
+ ❌ User profiles (too much churn, GDPR deletion complexity)
11
+ ❌ Session/cache state (ephemeral, no audit need)
12
+
13
+ ## Event Design
14
+ ```csharp
15
+ // Base event
16
+ public abstract record DomainEvent
17
+ {
18
+ public Guid EventId { get; init; } = Guid.NewGuid();
19
+ public DateTime OccurredAt { get; init; } = DateTime.UtcNow;
20
+ public int Version { get; init; }
21
+ public string EventType => GetType().Name;
22
+ }
23
+
24
+ // Specific events — past tense, immutable
25
+ public record OrderPlaced(Guid OrderId, Guid UserId, decimal Total, List<OrderItem> Items)
26
+ : DomainEvent;
27
+
28
+ public record OrderShipped(Guid OrderId, string TrackingNumber, DateTime ShippedAt)
29
+ : DomainEvent;
30
+
31
+ public record OrderCancelled(Guid OrderId, string Reason)
32
+ : DomainEvent;
33
+ ```
34
+
35
+ ## Aggregate with Event Sourcing
36
+ ```csharp
37
+ public class Order : AggregateRoot
38
+ {
39
+ public OrderStatus Status { get; private set; }
40
+ public decimal Total { get; private set; }
41
+
42
+ // Reconstruct from events
43
+ public static Order Replay(IEnumerable<DomainEvent> events)
44
+ {
45
+ var order = new Order();
46
+ foreach (var e in events) order.Apply(e);
47
+ return order;
48
+ }
49
+
50
+ // Command → validate → raise event
51
+ public void Ship(string trackingNumber)
52
+ {
53
+ if (Status != OrderStatus.Confirmed)
54
+ throw new InvalidOperationException($"Cannot ship order in {Status} status");
55
+
56
+ RaiseEvent(new OrderShipped(Id, trackingNumber, DateTime.UtcNow));
57
+ }
58
+
59
+ // State transition from event (idempotent)
60
+ protected override void Apply(DomainEvent @event)
61
+ {
62
+ switch (@event)
63
+ {
64
+ case OrderPlaced e:
65
+ Id = e.OrderId;
66
+ Total = e.Total;
67
+ Status = OrderStatus.Pending;
68
+ break;
69
+
70
+ case OrderShipped e:
71
+ Status = OrderStatus.Shipped;
72
+ break;
73
+
74
+ case OrderCancelled:
75
+ Status = OrderStatus.Cancelled;
76
+ break;
77
+ }
78
+ }
79
+ }
80
+ ```
81
+
82
+ ## Event Store (Azure Cosmos DB pattern)
83
+ ```csharp
84
+ public class CosmosEventStore : IEventStore
85
+ {
86
+ public async Task AppendAsync(Guid streamId, IEnumerable<DomainEvent> events, int expectedVersion)
87
+ {
88
+ foreach (var (evt, i) in events.Select((e, i) => (e, i)))
89
+ {
90
+ var document = new EventDocument
91
+ {
92
+ Id = $"{streamId}:{expectedVersion + i + 1}",
93
+ StreamId = streamId.ToString(),
94
+ Version = expectedVersion + i + 1,
95
+ EventType = evt.EventType,
96
+ Data = JsonSerializer.Serialize(evt, evt.GetType()),
97
+ OccurredAt = evt.OccurredAt
98
+ };
99
+ await _container.CreateItemAsync(document, new PartitionKey(streamId.ToString()));
100
+ }
101
+ }
102
+ }
103
+ ```
104
+
105
+ ## Projections (Read Models)
106
+ ```csharp
107
+ // Project events to read-optimized views
108
+ public class OrderSummaryProjection
109
+ {
110
+ public void Handle(OrderPlaced e, AppDbContext ctx)
111
+ => ctx.OrderSummaries.Add(new OrderSummary { Id = e.OrderId, Total = e.Total, Status = "Pending" });
112
+
113
+ public void Handle(OrderShipped e, AppDbContext ctx)
114
+ => ctx.OrderSummaries.Where(s => s.Id == e.OrderId).ExecuteUpdate(s => s.SetProperty(p => p.Status, "Shipped"));
115
+ }
116
+ ```
117
+
118
+ ## Trade-offs
119
+ | Pro | Con |
120
+ |-----|-----|
121
+ | Complete audit trail | Schema evolution complexity |
122
+ | Temporal queries (state at time T) | Higher storage cost |
123
+ | Event replay for debugging | Eventual consistency for read models |
124
+ | Natural integration events | GDPR deletion requires event masking |
@@ -0,0 +1,95 @@
1
+ # Integration Standard: Azure Service Bus
2
+
3
+ ## Patterns
4
+
5
+ ### Queue vs Topic
6
+ - **Queue**: Point-to-point, one consumer per message. Use for jobs, commands.
7
+ - **Topic + Subscription**: Fan-out, multiple consumers. Use for domain events.
8
+
9
+ ### Sender Pattern
10
+ ```csharp
11
+ public class ServiceBusSender<T>
12
+ {
13
+ private readonly ServiceBusSender _sender;
14
+
15
+ public async Task SendAsync(T message, CancellationToken ct = default)
16
+ {
17
+ var json = JsonSerializer.Serialize(message);
18
+ var serviceBusMessage = new ServiceBusMessage(json)
19
+ {
20
+ ContentType = "application/json",
21
+ MessageId = Guid.NewGuid().ToString(),
22
+ Subject = typeof(T).Name // For filtering
23
+ };
24
+ await _sender.SendMessageAsync(serviceBusMessage, ct);
25
+ }
26
+ }
27
+ ```
28
+
29
+ ### Receiver Pattern (with IHostedService)
30
+ ```csharp
31
+ public class OrderProcessor : IHostedService
32
+ {
33
+ private ServiceBusProcessor _processor;
34
+
35
+ public async Task StartAsync(CancellationToken ct)
36
+ {
37
+ _processor.ProcessMessageAsync += HandleMessageAsync;
38
+ _processor.ProcessErrorAsync += HandleErrorAsync;
39
+ await _processor.StartProcessingAsync(ct);
40
+ }
41
+
42
+ private async Task HandleMessageAsync(ProcessMessageEventArgs args)
43
+ {
44
+ var order = JsonSerializer.Deserialize<OrderCreatedEvent>(args.Message.Body);
45
+ await _orderService.ProcessAsync(order!);
46
+ await args.CompleteMessageAsync(args.Message);
47
+ }
48
+
49
+ private Task HandleErrorAsync(ProcessErrorEventArgs args)
50
+ {
51
+ _logger.LogError(args.Exception, "Service Bus error on {EntityPath}", args.EntityPath);
52
+ return Task.CompletedTask; // Abandon → retry → dead-letter
53
+ }
54
+ }
55
+ ```
56
+
57
+ ### Dead-Letter Processing
58
+ ```csharp
59
+ // Check dead-letter queue periodically
60
+ var client = new ServiceBusClient(connectionString);
61
+ var receiver = client.CreateReceiver(queueName, new ServiceBusReceiverOptions
62
+ {
63
+ SubQueue = SubQueue.DeadLetter
64
+ });
65
+
66
+ await foreach (var msg in receiver.ReceiveMessagesAsync())
67
+ {
68
+ _logger.LogError("Dead-lettered: {Reason} | Body: {Body}",
69
+ msg.DeadLetterReason, msg.Body.ToString());
70
+ await receiver.CompleteMessageAsync(msg);
71
+ }
72
+ ```
73
+
74
+ ## Retry Policy
75
+ - MaxDeliveryCount: 5 (queue setting)
76
+ - Lock duration: 5 minutes (for long-processing messages)
77
+ - Dead-letter after MaxDeliveryCount exceeded
78
+
79
+ ## Required Settings
80
+ ```json
81
+ {
82
+ "ServiceBus": {
83
+ "ConnectionString": "from-key-vault",
84
+ "QueueName": "feature-name-queue",
85
+ "MaxConcurrentCalls": 4,
86
+ "PrefetchCount": 10
87
+ }
88
+ }
89
+ ```
90
+
91
+ ## Anti-Patterns
92
+ - Never use connection string in code — always Key Vault
93
+ - Never process in `HandleErrorAsync` — only log
94
+ - Never use topics for commands (use queues)
95
+ - Never use synchronous sends in Blazor components
@@ -0,0 +1,384 @@
1
+ # Integration Standard: MCP Tools in MORPH Workflows
2
+
3
+ > Reference for using Model Context Protocol (MCP) tools across morph-spec phases. Each phase benefits from different MCPs depending on what data is needed.
4
+
5
+ ## MCP Availability Detection
6
+
7
+ Before using any MCP tool, check if it's available in the current session:
8
+
9
+ ```javascript
10
+ // Pattern: Attempt the call — if the tool doesn't exist, Claude Code will report it
11
+ // There is no "list MCPs" command; just try the most common operation
12
+
13
+ // Supabase MCP
14
+ await mcp__supabase__list_tables();
15
+
16
+ // GitHub MCP
17
+ await mcp__github__get_repo();
18
+
19
+ // Context7 (library docs)
20
+ await mcp__context7__resolve_library_id({ libraryName: "react" });
21
+
22
+ // Playwright (browser automation)
23
+ await mcp__playwright__browser_navigate({ url: "https://example.com" });
24
+ ```
25
+
26
+ **Fallback rule:** If an MCP is not available, every phase has a manual alternative using Claude Code native tools (Read, Grep, Glob, Bash).
27
+
28
+ ---
29
+
30
+ ## MCP Providers by Phase
31
+
32
+ ### Phase 1 (Setup) — Project Discovery
33
+
34
+ | MCP | Use Case | Example |
35
+ |-----|----------|---------|
36
+ | **GitHub** | Repo info, recent PRs, issues | `mcp__github__get_repo()` |
37
+ | **Filesystem** | Scan project structure | Native Glob/Read preferred |
38
+
39
+ ```javascript
40
+ // Detect project type from repo metadata
41
+ const repo = await mcp__github__get_repo();
42
+ // → language, default_branch, topics
43
+
44
+ // Fallback: use Glob to detect stack
45
+ // Glob: "**/{package.json,*.csproj,*.sln,go.mod,Cargo.toml}"
46
+ ```
47
+
48
+ ### Phase 1.5 (UI/UX) — Design References
49
+
50
+ | MCP | Use Case | Example |
51
+ |-----|----------|---------|
52
+ | **Figma** | Extract design tokens, components | `mcp__figma__get_file({ fileKey })` |
53
+ | **Playwright** | Navigate, screenshot, inspect live pages | `mcp__playwright__browser_navigate({ url })` |
54
+ | **Context7** | Component library documentation | `mcp__context7__query_docs({ libraryId, query })` |
55
+
56
+ ```javascript
57
+ // Get design tokens from Figma
58
+ const file = await mcp__figma__get_file({ fileKey: "abc123" });
59
+ // → colors, typography, spacing from design file
60
+
61
+ // Screenshot existing app for design reference
62
+ await mcp__playwright__browser_navigate({ url: "https://app.example.com/dashboard" });
63
+ const screenshot = await mcp__playwright__browser_take_screenshot();
64
+ // → Visual reference for UI/UX design
65
+
66
+ // Inspect page structure via accessibility tree
67
+ const snapshot = await mcp__playwright__browser_snapshot();
68
+ // → Structured element tree (headings, buttons, inputs, etc.)
69
+
70
+ // Test responsive layout
71
+ await mcp__playwright__browser_resize({ width: 375, height: 812 });
72
+ const mobileScreenshot = await mcp__playwright__browser_take_screenshot();
73
+
74
+ // Get MudBlazor component docs for UI specs
75
+ const libId = await mcp__context7__resolve_library_id({
76
+ libraryName: "mudblazor",
77
+ query: "data grid with sorting and filtering"
78
+ });
79
+ const docs = await mcp__context7__query_docs({
80
+ libraryId: libId,
81
+ query: "DataGrid component props and events"
82
+ });
83
+
84
+ // Fallback: WebSearch for component documentation
85
+ // WebFetch for specific component API pages
86
+ ```
87
+
88
+ ### Phase 2 (Design) — Schema & Architecture
89
+
90
+ | MCP | Use Case | Example |
91
+ |-----|----------|---------|
92
+ | **Supabase** | Database schema, tables, relationships, RLS | `mcp__supabase__get_table_schema({ table })` |
93
+ | **Database MCPs** | PostgreSQL, MySQL, SQL Server schema | Provider-specific tools |
94
+ | **Context7** | Library docs for architecture decisions | `mcp__context7__query_docs()` |
95
+ | **GitHub** | Check existing code patterns, PRs | `mcp__github__search_code()` |
96
+
97
+ ```javascript
98
+ // === SCHEMA ANALYSIS (Critical for contracts.cs) ===
99
+
100
+ // 1. List all tables
101
+ const tables = await mcp__supabase__list_tables();
102
+
103
+ // 2. Get schema for each relevant table
104
+ const schema = await mcp__supabase__get_table_schema({ table: 'leads' });
105
+ // → column_name, data_type, is_nullable, column_default
106
+
107
+ // 3. Get foreign key relationships
108
+ const rels = await mcp__supabase__get_relationships({ table: 'leads' });
109
+ // → foreign_table, foreign_column, constraint_type
110
+
111
+ // 4. Run custom query for complex schema info
112
+ const result = await mcp__supabase__query({
113
+ query: `SELECT column_name, data_type, is_nullable
114
+ FROM information_schema.columns
115
+ WHERE table_name = 'leads'
116
+ ORDER BY ordinal_position`
117
+ });
118
+
119
+ // 5. Check RLS policies (security-critical)
120
+ const policies = await mcp__supabase__query({
121
+ query: `SELECT tablename, policyname, cmd, qual
122
+ FROM pg_policies
123
+ WHERE tablename = 'leads'`
124
+ });
125
+ ```
126
+
127
+ **Fallback (no Supabase MCP):**
128
+ ```
129
+ 1. Grep: "\.from\(|\.select\(|SELECT |supabase\.|DbSet<" → find query files
130
+ 2. Read each query file → extract table/column names
131
+ 3. Glob: "src/**/types/**/*.ts" or "**/Entities/**/*.cs" → find type definitions
132
+ 4. Read type files → map properties to database columns
133
+ ```
134
+
135
+ ### Phase 3 (Clarify) — Validation & Research
136
+
137
+ | MCP | Use Case | Example |
138
+ |-----|----------|---------|
139
+ | **Context7** | Verify library capabilities, API limits | `mcp__context7__query_docs()` |
140
+ | **GitHub** | Check issue discussions, known limitations | `mcp__github__search_issues()` |
141
+
142
+ ```javascript
143
+ // Verify if library supports a required feature
144
+ const docs = await mcp__context7__query_docs({
145
+ libraryId: "/mudblazor/mudblazor",
146
+ query: "DataGrid server-side pagination with virtual scrolling"
147
+ });
148
+ // → Confirms capability or identifies limitation for spec update
149
+
150
+ // Fallback: WebSearch for library capabilities
151
+ ```
152
+
153
+ ### Phase 4 (Tasks) — Planning & Organization
154
+
155
+ | MCP | Use Case | Example |
156
+ |-----|----------|---------|
157
+ | **GitHub** | Create issues from tasks, link to milestone | `mcp__github__create_issue()` |
158
+ | **Context7** | Estimate complexity based on library docs | `mcp__context7__query_docs()` |
159
+
160
+ ```javascript
161
+ // Create GitHub issues from tasks.json (if team uses GitHub Projects)
162
+ for (const task of tasks) {
163
+ await mcp__github__create_issue({
164
+ title: task.title,
165
+ body: task.description,
166
+ labels: [task.category],
167
+ milestone: featureMilestone
168
+ });
169
+ }
170
+
171
+ // Fallback: Bash with gh CLI
172
+ // gh issue create --title "T001: Create Entity Lead" --label "domain"
173
+ ```
174
+
175
+ ### Phase 5 (Implement) — Build & Deploy
176
+
177
+ | MCP | Use Case | Example |
178
+ |-----|----------|---------|
179
+ | **Supabase** | Run migrations, create RLS policies | `mcp__supabase__query()` |
180
+ | **GitHub** | Create PR, push branches | `mcp__github__create_pull_request()` |
181
+ | **Context7** | Look up API usage during coding | `mcp__context7__query_docs()` |
182
+ | **Playwright** | Smoke test deployed features, verify UI | `mcp__playwright__browser_navigate()` |
183
+ | **Azure** | Provision resources, check deployment status | Provider-specific |
184
+ | **Docker** | Build images, manage containers | Provider-specific |
185
+
186
+ ```javascript
187
+ // Run migration after implementing data model
188
+ await mcp__supabase__query({
189
+ query: `ALTER TABLE leads ADD COLUMN status varchar(20) DEFAULT 'new'`
190
+ });
191
+
192
+ // Create RLS policy for new feature
193
+ await mcp__supabase__query({
194
+ query: `CREATE POLICY "Users can view own leads"
195
+ ON leads FOR SELECT
196
+ USING (auth.uid() = user_id)`
197
+ });
198
+
199
+ // Lookup API during implementation
200
+ const docs = await mcp__context7__query_docs({
201
+ libraryId: "/dotnet/efcore",
202
+ query: "owned entity types configuration"
203
+ });
204
+
205
+ // Smoke test deployed feature via browser
206
+ await mcp__playwright__browser_navigate({ url: "https://localhost:5001/leads" });
207
+ const snapshot = await mcp__playwright__browser_snapshot();
208
+ // → Verify page renders correctly, check for errors
209
+
210
+ // Check for console errors after deploy
211
+ const logs = await mcp__playwright__browser_console_messages();
212
+ // → Catch JavaScript errors, failed API calls
213
+
214
+ // Screenshot for recap.md documentation
215
+ const screenshot = await mcp__playwright__browser_take_screenshot();
216
+
217
+ // Fallback: Bash for migrations, gh CLI for PRs
218
+ ```
219
+
220
+ ---
221
+
222
+ ## MCP vs Native Tools Decision
223
+
224
+ | Need | MCP Tool | Native Alternative |
225
+ |------|----------|--------------------|
226
+ | Database schema | Supabase/DB MCP | Grep queries + Read types |
227
+ | Repo metadata | GitHub MCP | Bash `gh` CLI |
228
+ | Design tokens | Figma MCP | Read CSS/SCSS variables |
229
+ | Library docs | Context7 | WebSearch + WebFetch |
230
+ | Live page preview | Playwright MCP | WebFetch URL |
231
+ | Page interaction (click, type, navigate) | Playwright MCP | Manual testing |
232
+ | Responsive layout testing | Playwright MCP (`browser_resize`) | Manual testing |
233
+ | Console error checking | Playwright MCP (`browser_console_messages`) | Browser DevTools |
234
+ | Container ops | Docker MCP | Bash `docker` CLI |
235
+ | Cloud resources | Azure MCP | Bash `az` CLI |
236
+
237
+ **Rule:** MCP tools provide structured data (JSON responses). Native tools require manual parsing. **Always prefer MCP when available** — fall back to native when not.
238
+
239
+ ---
240
+
241
+ ## Common MCP Patterns
242
+
243
+ ### Browser (Playwright): Page Automation & Analysis
244
+
245
+ ```javascript
246
+ // === SETUP ===
247
+ // Playwright MCP: npx @playwright/mcp@latest
248
+ // Config in .claude/settings.json or claude_desktop_config.json:
249
+ // { "mcpServers": { "playwright": { "command": "npx", "args": ["@playwright/mcp@latest"] } } }
250
+
251
+ // === NAVIGATION & SNAPSHOT ===
252
+
253
+ // 1. Navigate to a page
254
+ await mcp__playwright__browser_navigate({ url: "https://app.example.com/dashboard" });
255
+
256
+ // 2. Take accessibility snapshot (preferred — structured, LLM-friendly)
257
+ const snapshot = await mcp__playwright__browser_snapshot();
258
+ // → Returns accessibility tree with element refs for interaction
259
+
260
+ // 3. Take visual screenshot (requires --caps vision)
261
+ const screenshot = await mcp__playwright__browser_take_screenshot();
262
+ // → Returns PNG image of current page
263
+
264
+ // === INTERACTION ===
265
+
266
+ // 4. Click an element (use ref from snapshot)
267
+ await mcp__playwright__browser_click({ element: "Submit button", ref: "s1e15" });
268
+
269
+ // 5. Type into an input field
270
+ await mcp__playwright__browser_type({ element: "Search input", ref: "s1e8", text: "query" });
271
+
272
+ // 6. Fill a form field (replaces existing value)
273
+ await mcp__playwright__browser_fill_form({ ref: "s1e8", value: "new value" });
274
+
275
+ // 7. Select dropdown option
276
+ await mcp__playwright__browser_select_option({ element: "Status", ref: "s1e12", values: ["active"] });
277
+
278
+ // 8. Press keyboard key
279
+ await mcp__playwright__browser_press_key({ key: "Enter" });
280
+
281
+ // === TABS & NAVIGATION ===
282
+
283
+ // 9. List open tabs
284
+ const tabs = await mcp__playwright__browser_tabs();
285
+
286
+ // 10. Navigate back
287
+ await mcp__playwright__browser_navigate_back();
288
+
289
+ // 11. Close current tab
290
+ await mcp__playwright__browser_close();
291
+
292
+ // === ADVANCED ===
293
+
294
+ // 12. Evaluate JavaScript in page context
295
+ const result = await mcp__playwright__browser_evaluate({
296
+ expression: "document.querySelectorAll('.error').length"
297
+ });
298
+
299
+ // 13. Get console messages (debug)
300
+ const logs = await mcp__playwright__browser_console_messages();
301
+
302
+ // 14. Get network requests
303
+ const requests = await mcp__playwright__browser_network_requests();
304
+
305
+ // 15. Handle dialog (alert, confirm, prompt)
306
+ await mcp__playwright__browser_handle_dialog({ accept: true });
307
+
308
+ // 16. Resize viewport
309
+ await mcp__playwright__browser_resize({ width: 375, height: 812 }); // iPhone viewport
310
+
311
+ // 17. Save page as PDF (requires --caps pdf)
312
+ await mcp__playwright__browser_pdf_save();
313
+
314
+ // 18. Upload file
315
+ await mcp__playwright__browser_file_upload({ ref: "s1e20", paths: ["/path/to/file.png"] });
316
+ ```
317
+
318
+ **Use cases by phase:**
319
+
320
+ | Phase | Use Case | Tool |
321
+ |-------|----------|------|
322
+ | UI/UX (1.5) | Screenshot reference pages for design | `browser_navigate` + `browser_take_screenshot` |
323
+ | UI/UX (1.5) | Inspect existing app structure | `browser_navigate` + `browser_snapshot` |
324
+ | UI/UX (1.5) | Test responsive layouts | `browser_resize` + `browser_take_screenshot` |
325
+ | Clarify (3) | Verify existing UI behavior | `browser_navigate` + `browser_snapshot` |
326
+ | Implement (5) | Smoke test deployed features | `browser_navigate` + `browser_click` + `browser_snapshot` |
327
+ | Implement (5) | Verify form flows end-to-end | `browser_fill_form` + `browser_click` + `browser_snapshot` |
328
+ | Implement (5) | Check console errors after deploy | `browser_navigate` + `browser_console_messages` |
329
+ | Implement (5) | Screenshot for recap.md | `browser_take_screenshot` |
330
+
331
+ ---
332
+
333
+ ### Supabase: Full Schema Discovery
334
+
335
+ ```javascript
336
+ // Complete workflow for Phase 2 schema analysis
337
+ const tables = await mcp__supabase__list_tables();
338
+
339
+ for (const table of tables.filter(t => isRelevant(t))) {
340
+ const schema = await mcp__supabase__get_table_schema({ table: table.name });
341
+ const rels = await mcp__supabase__get_relationships({ table: table.name });
342
+ // → Write findings to schema-analysis.md
343
+ }
344
+ ```
345
+
346
+ ### Context7: Library Research
347
+
348
+ ```javascript
349
+ // Workflow for any phase needing library documentation
350
+ // Step 1: Resolve library ID
351
+ const lib = await mcp__context7__resolve_library_id({
352
+ libraryName: "fluent-ui-blazor",
353
+ query: "dialog component with form validation"
354
+ });
355
+
356
+ // Step 2: Query specific documentation
357
+ const docs = await mcp__context7__query_docs({
358
+ libraryId: lib.id,
359
+ query: "FluentDialog component usage with EditForm validation"
360
+ });
361
+ ```
362
+
363
+ ### GitHub: Code Search Across Repo
364
+
365
+ ```javascript
366
+ // Find patterns in large codebase during Phase 2
367
+ const results = await mcp__github__search_code({
368
+ query: "supabase.from leads repo:myorg/myrepo",
369
+ });
370
+ // → Find all places that query the leads table
371
+ ```
372
+
373
+ ---
374
+
375
+ ## Security Considerations
376
+
377
+ - **Never pass secrets** in MCP tool parameters (API keys, tokens, passwords)
378
+ - **Supabase MCP** uses project-level auth — queries run with service role by default
379
+ - **GitHub MCP** inherits the configured token's permissions
380
+ - **Always validate MCP responses** before using in contracts or code generation
381
+
382
+ ---
383
+
384
+ *MORPH-SPEC by Polymorphism Tech*