@polymorphism-tech/morph-spec 3.0.0 → 3.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (240) hide show
  1. package/CLAUDE.md +68 -400
  2. package/README.md +198 -76
  3. package/bin/detect-agents.js +227 -225
  4. package/bin/morph-spec.js +10 -0
  5. package/bin/render-template.js +303 -302
  6. package/bin/semantic-detect-agents.js +247 -246
  7. package/bin/{task-manager.js → task-manager.cjs} +12 -1
  8. package/bin/validate-agents-skills.js +257 -251
  9. package/bin/validate-agents.js +70 -69
  10. package/bin/validate-phase.js +263 -263
  11. package/docs/getting-started.md +3 -3
  12. package/package.json +3 -4
  13. package/scripts/reorganize-skills.cjs +175 -0
  14. package/scripts/validate-agents-structure.cjs +52 -0
  15. package/scripts/validate-skills.cjs +180 -0
  16. package/src/commands/create-story.js +354 -351
  17. package/src/commands/detect-agents.js +13 -2
  18. package/src/commands/detect.js +104 -104
  19. package/src/commands/state.js +334 -333
  20. package/src/commands/sync.js +167 -167
  21. package/src/commands/task.js +1 -1
  22. package/src/commands/update.js +13 -1
  23. package/src/lib/context-generator.js +7 -4
  24. package/{detectors → src/lib/detectors}/config-detector.js +223 -223
  25. package/{detectors → src/lib/detectors}/conversation-analyzer.js +163 -163
  26. package/{detectors → src/lib/detectors}/index.js +84 -84
  27. package/{detectors → src/lib/detectors}/standards-generator.js +275 -275
  28. package/src/lib/hook-executor.js +2 -1
  29. package/src/lib/stack-resolver.js +148 -0
  30. package/src/lib/standards-context-injector.js +4 -3
  31. package/src/lib/state-manager.js +21 -4
  32. package/src/lib/team-orchestrator.js +2 -1
  33. package/src/lib/troubleshoot-grep.js +13 -3
  34. package/src/lib/validation-runner.js +2 -1
  35. package/src/utils/file-copier.js +3 -1
  36. package/{content → stacks/blazor-azure}/.azure/README.md +293 -293
  37. package/{content → stacks/blazor-azure}/.azure/docs/azure-devops-setup.md +454 -454
  38. package/{content → stacks/blazor-azure}/.azure/docs/branch-strategy.md +398 -398
  39. package/{content → stacks/blazor-azure}/.azure/docs/local-development.md +515 -515
  40. package/{content → stacks/blazor-azure}/.azure/pipelines/pipeline-variables.yml +34 -34
  41. package/{content → stacks/blazor-azure}/.azure/pipelines/prod-pipeline.yml +319 -319
  42. package/{content → stacks/blazor-azure}/.azure/pipelines/staging-pipeline.yml +234 -234
  43. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/build-dotnet.yml +75 -75
  44. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/deploy-app-service.yml +94 -94
  45. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/deploy-container-app.yml +120 -120
  46. package/{content → stacks/blazor-azure}/.azure/pipelines/templates/infra-deploy.yml +90 -90
  47. package/{content → stacks/blazor-azure}/.claude/commands/morph-archive.md +79 -79
  48. package/{content → stacks/blazor-azure}/.claude/commands/morph-deploy.md +529 -529
  49. package/{content → stacks/blazor-azure}/.claude/commands/morph-infra.md +209 -209
  50. package/{content → stacks/blazor-azure}/.claude/commands/morph-troubleshoot.md +1 -1
  51. package/{content → stacks/blazor-azure}/.claude/settings.local.json +15 -15
  52. package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-setup.md +1 -1
  53. package/{content/.claude/skills/specialists → stacks/blazor-azure/.claude/skills/level-2-domains/architecture}/prompt-engineer.md +189 -189
  54. package/{content/.claude/skills/specialists → stacks/blazor-azure/.claude/skills/level-2-domains/architecture}/seo-growth-hacker.md +320 -320
  55. package/{content/.claude/skills/infra → stacks/blazor-azure/.claude/skills/level-2-domains/infrastructure}/azure-deploy-specialist.md +699 -699
  56. package/{content → stacks/blazor-azure}/.morph/.morphversion +5 -5
  57. package/{content → stacks/blazor-azure}/.morph/archive/.gitkeep +25 -25
  58. package/{content → stacks/blazor-azure}/.morph/config/agents.json +7 -5
  59. package/{content → stacks/blazor-azure}/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
  60. package/{content → stacks/blazor-azure}/.morph/docs/workflows/enforcement-pipeline.md +3 -3
  61. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/README.md +241 -241
  62. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/contracts.ts +307 -307
  63. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/spec.md +399 -399
  64. package/{content → stacks/blazor-azure}/.morph/examples/api-nextjs/tasks.md +168 -168
  65. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/README.md +125 -125
  66. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/contracts.cs +358 -358
  67. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/decisions.md +246 -246
  68. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/spec.md +236 -236
  69. package/{content → stacks/blazor-azure}/.morph/examples/micro-saas/tasks.md +150 -150
  70. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/README.md +309 -309
  71. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/contracts.cs +433 -433
  72. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/spec.md +479 -479
  73. package/{content → stacks/blazor-azure}/.morph/examples/multi-agent/tasks.md +185 -185
  74. package/{content → stacks/blazor-azure}/.morph/examples/state-v3.json +188 -188
  75. package/{content → stacks/blazor-azure}/.morph/features/.gitkeep +25 -25
  76. package/{content → stacks/blazor-azure}/.morph/hooks/README.md +12 -12
  77. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-all.sh +48 -48
  78. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-specs.sh +49 -49
  79. package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-tests.sh +60 -60
  80. package/{content → stacks/blazor-azure}/.morph/project.md +160 -160
  81. package/{content → stacks/blazor-azure}/.morph/schemas/agent.schema.json +296 -296
  82. package/{content → stacks/blazor-azure}/.morph/specs/.gitkeep +20 -20
  83. package/{content → stacks/blazor-azure}/.morph/standards/agent-teams-workflow.md +2 -2
  84. package/{content → stacks/blazor-azure}/.morph/standards/coding.md +377 -377
  85. package/{content → stacks/blazor-azure}/.morph/standards/fluent-ui-setup.md +590 -590
  86. package/{content → stacks/blazor-azure}/.morph/standards/migration-guide.md +514 -514
  87. package/{content → stacks/blazor-azure}/.morph/standards/passkeys-auth.md +423 -423
  88. package/{content → stacks/blazor-azure}/.morph/standards/vector-search-rag.md +536 -536
  89. package/{content → stacks/blazor-azure}/.morph/state.json +17 -17
  90. package/{content → stacks/blazor-azure}/.morph/templates/FluentDesignTheme.cs +149 -149
  91. package/{content → stacks/blazor-azure}/.morph/templates/MudTheme.cs +281 -281
  92. package/{content → stacks/blazor-azure}/.morph/templates/component.razor +239 -239
  93. package/{content → stacks/blazor-azure}/.morph/templates/contracts.cs +217 -217
  94. package/{content → stacks/blazor-azure}/.morph/templates/design-system.css +226 -226
  95. package/{content → stacks/blazor-azure}/.morph/templates/infra/.dockerignore.example +89 -89
  96. package/{content → stacks/blazor-azure}/.morph/templates/infra/Dockerfile.example +82 -82
  97. package/{content → stacks/blazor-azure}/.morph/templates/infra/README.md +286 -286
  98. package/{content → stacks/blazor-azure}/.morph/templates/infra/app-insights.bicep +63 -63
  99. package/{content → stacks/blazor-azure}/.morph/templates/infra/app-service.bicep +164 -164
  100. package/{content → stacks/blazor-azure}/.morph/templates/infra/azure-pipelines-deploy.yml +480 -480
  101. package/{content → stacks/blazor-azure}/.morph/templates/infra/container-app-env.bicep +49 -49
  102. package/{content → stacks/blazor-azure}/.morph/templates/infra/container-app.bicep +156 -156
  103. package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy.ps1 +229 -229
  104. package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy.sh +208 -208
  105. package/{content → stacks/blazor-azure}/.morph/templates/infra/key-vault.bicep +91 -91
  106. package/{content → stacks/blazor-azure}/.morph/templates/infra/main.bicep +189 -189
  107. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.dev.json +29 -29
  108. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.prod.json +29 -29
  109. package/{content → stacks/blazor-azure}/.morph/templates/infra/parameters.staging.json +29 -29
  110. package/{content → stacks/blazor-azure}/.morph/templates/infra/sql-database.bicep +103 -103
  111. package/{content → stacks/blazor-azure}/.morph/templates/infra/storage.bicep +106 -106
  112. package/{content → stacks/blazor-azure}/.morph/templates/integrations/asaas-client.cs +387 -387
  113. package/{content → stacks/blazor-azure}/.morph/templates/integrations/asaas-webhook.cs +351 -351
  114. package/{content → stacks/blazor-azure}/.morph/templates/integrations/azure-identity-config.cs +288 -288
  115. package/{content → stacks/blazor-azure}/.morph/templates/integrations/clerk-config.cs +258 -258
  116. package/{content → stacks/blazor-azure}/.morph/templates/job.cs +171 -171
  117. package/{content → stacks/blazor-azure}/.morph/templates/migration.cs +83 -83
  118. package/{content → stacks/blazor-azure}/.morph/templates/repository.cs +141 -141
  119. package/{content → stacks/blazor-azure}/.morph/templates/saas/subscription.cs +347 -347
  120. package/{content → stacks/blazor-azure}/.morph/templates/saas/tenant.cs +338 -338
  121. package/{content → stacks/blazor-azure}/.morph/templates/service.cs +139 -139
  122. package/{content → stacks/blazor-azure}/.morph/templates/sprint-status.yaml +68 -68
  123. package/{content → stacks/blazor-azure}/.morph/templates/story.md +143 -143
  124. package/{content → stacks/blazor-azure}/.morph/templates/test.cs +239 -239
  125. package/{content → stacks/blazor-azure}/.morph/templates/ui-design-system.md +286 -286
  126. package/{content → stacks/blazor-azure}/.morph/templates/ui-flows.md +336 -336
  127. package/{content → stacks/blazor-azure}/.morph/templates/ui-mockups.md +133 -133
  128. package/{content → stacks/blazor-azure}/.morph/test-infra/example.bicep +59 -59
  129. package/{content → stacks/blazor-azure}/README.md +79 -79
  130. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/backend/dotnet-supabase.md +244 -0
  131. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/frontend/nextjs-supabase.md +335 -0
  132. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/infrastructure/easypanel-deployer.md +189 -0
  133. package/stacks/nextjs-supabase/.claude/skills/level-2-domains/integrations/supabase-expert.md +170 -0
  134. package/stacks/nextjs-supabase/.morph/config/agents.json +345 -0
  135. package/stacks/nextjs-supabase/.morph/config/config.template.json +92 -0
  136. package/stacks/nextjs-supabase/.morph/docs/easypanel-setup.md +169 -0
  137. package/stacks/nextjs-supabase/.morph/docs/supabase-mcp-setup.md +247 -0
  138. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/README.md +697 -0
  139. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/spec.md +85 -0
  140. package/stacks/nextjs-supabase/.morph/examples/crud-nextjs-supabase/tasks.md +86 -0
  141. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/README.md +498 -0
  142. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/decisions.md +121 -0
  143. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/spec.md +138 -0
  144. package/stacks/nextjs-supabase/.morph/examples/saas-nextjs-supabase/tasks.md +162 -0
  145. package/stacks/nextjs-supabase/.morph/project.md +168 -0
  146. package/stacks/nextjs-supabase/.morph/standards/easypanel-deploy.md +191 -0
  147. package/stacks/nextjs-supabase/.morph/standards/nextjs-patterns.md +193 -0
  148. package/stacks/nextjs-supabase/.morph/standards/supabase-auth.md +171 -0
  149. package/stacks/nextjs-supabase/.morph/standards/supabase-pgvector.md +164 -0
  150. package/stacks/nextjs-supabase/.morph/standards/supabase-rls.md +179 -0
  151. package/stacks/nextjs-supabase/.morph/standards/supabase-storage.md +148 -0
  152. package/stacks/nextjs-supabase/.morph/templates/contracts.cs +173 -0
  153. package/stacks/nextjs-supabase/.morph/templates/contracts.ts +168 -0
  154. package/stacks/nextjs-supabase/.morph/templates/decisions.md +115 -0
  155. package/stacks/nextjs-supabase/.morph/templates/dockerfile-api.dockerfile +38 -0
  156. package/stacks/nextjs-supabase/.morph/templates/dockerfile-web.dockerfile +48 -0
  157. package/stacks/nextjs-supabase/.morph/templates/proposal.md +145 -0
  158. package/stacks/nextjs-supabase/.morph/templates/recap.md +134 -0
  159. package/stacks/nextjs-supabase/.morph/templates/rls-policy.sql +57 -0
  160. package/stacks/nextjs-supabase/.morph/templates/spec.md +231 -0
  161. package/stacks/nextjs-supabase/.morph/templates/supabase-migration.sql +100 -0
  162. package/stacks/nextjs-supabase/.morph/templates/tasks.md +257 -0
  163. package/stacks/nextjs-supabase/CLAUDE.md +149 -0
  164. package/stacks/nextjs-supabase/README.md +112 -0
  165. /package/{detectors → src/lib/detectors}/structure-detector.js +0 -0
  166. /package/{content → stacks/blazor-azure}/.claude/commands/morph-apply.md +0 -0
  167. /package/{content → stacks/blazor-azure}/.claude/commands/morph-preflight.md +0 -0
  168. /package/{content → stacks/blazor-azure}/.claude/commands/morph-proposal.md +0 -0
  169. /package/{content → stacks/blazor-azure}/.claude/commands/morph-status.md +0 -0
  170. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/README.md +0 -0
  171. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/code-review.md +0 -0
  172. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/morph-checklist.md +0 -0
  173. /package/{content → stacks/blazor-azure}/.claude/skills/level-0-meta/simulation-checklist.md +0 -0
  174. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/README.md +0 -0
  175. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/morph-replicate.md +0 -0
  176. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-clarify.md +0 -0
  177. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-design.md +0 -0
  178. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-tasks.md +0 -0
  179. /package/{content → stacks/blazor-azure}/.claude/skills/level-1-workflows/phase-uiux.md +0 -0
  180. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/README.md +0 -0
  181. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/ai-agents/ai-system-architect.md +0 -0
  182. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/architecture/po-pm-advisor.md +0 -0
  183. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/architecture/standards-architect.md +0 -0
  184. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/dotnet-senior.md +0 -0
  185. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/ef-modeler.md +0 -0
  186. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/hangfire-orchestrator.md +0 -0
  187. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/backend/ms-agent-expert.md +0 -0
  188. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/blazor-builder.md +0 -0
  189. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/nextjs-expert.md +0 -0
  190. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/frontend/ui-ux-designer.md +0 -0
  191. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/azure-architect.md +0 -0
  192. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/bicep-architect.md +0 -0
  193. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/container-specialist.md +0 -0
  194. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/infrastructure/devops-engineer.md +0 -0
  195. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/asaas-financial.md +0 -0
  196. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/azure-identity.md +0 -0
  197. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/clerk-auth.md +0 -0
  198. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/integrations/resend-email.md +0 -0
  199. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/quality/code-analyzer.md +0 -0
  200. /package/{content → stacks/blazor-azure}/.claude/skills/level-2-domains/quality/testing-specialist.md +0 -0
  201. /package/{content → stacks/blazor-azure}/.claude/skills/level-3-technologies/README.md +0 -0
  202. /package/{content → stacks/blazor-azure}/.claude/skills/level-4-patterns/README.md +0 -0
  203. /package/{content → stacks/blazor-azure}/.morph/config/config.template.json +0 -0
  204. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/design-impl.md +0 -0
  205. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/fast-track.md +0 -0
  206. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/full-morph.md +0 -0
  207. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/standard.md +0 -0
  208. /package/{content → stacks/blazor-azure}/.morph/docs/workflows/ui-refresh.md +0 -0
  209. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/decisions.md +0 -0
  210. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/proposal.md +0 -0
  211. /package/{content → stacks/blazor-azure}/.morph/examples/scheduled-reports/spec.md +0 -0
  212. /package/{content → stacks/blazor-azure}/.morph/hooks/pre-commit-agents.sh +0 -0
  213. /package/{content → stacks/blazor-azure}/.morph/hooks/task-completed.js +0 -0
  214. /package/{content → stacks/blazor-azure}/.morph/hooks/teammate-idle.js +0 -0
  215. /package/{content → stacks/blazor-azure}/.morph/schemas/tasks.schema.json +0 -0
  216. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-blazor-ui.md +0 -0
  217. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-production.md +0 -0
  218. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-setup.md +0 -0
  219. /package/{content → stacks/blazor-azure}/.morph/standards/agent-framework-workflows.md +0 -0
  220. /package/{content → stacks/blazor-azure}/.morph/standards/architecture.md +0 -0
  221. /package/{content → stacks/blazor-azure}/.morph/standards/azure.md +0 -0
  222. /package/{content → stacks/blazor-azure}/.morph/standards/dotnet10-migration.md +0 -0
  223. /package/{content → stacks/blazor-azure}/.morph/templates/CONTEXT-FEATURE.md +0 -0
  224. /package/{content → stacks/blazor-azure}/.morph/templates/CONTEXT.md +0 -0
  225. /package/{content → stacks/blazor-azure}/.morph/templates/agent.cs +0 -0
  226. /package/{content → stacks/blazor-azure}/.morph/templates/clarify-questions.md +0 -0
  227. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Commands.cs +0 -0
  228. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Entities.cs +0 -0
  229. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/Queries.cs +0 -0
  230. /package/{content → stacks/blazor-azure}/.morph/templates/contracts/README.md +0 -0
  231. /package/{content → stacks/blazor-azure}/.morph/templates/decisions.md +0 -0
  232. /package/{content → stacks/blazor-azure}/.morph/templates/infra/deploy-checklist.md +0 -0
  233. /package/{content → stacks/blazor-azure}/.morph/templates/proposal.md +0 -0
  234. /package/{content → stacks/blazor-azure}/.morph/templates/recap.md +0 -0
  235. /package/{content → stacks/blazor-azure}/.morph/templates/simulation.md +0 -0
  236. /package/{content → stacks/blazor-azure}/.morph/templates/spec.md +0 -0
  237. /package/{content → stacks/blazor-azure}/.morph/templates/state.template.json +0 -0
  238. /package/{content → stacks/blazor-azure}/.morph/templates/tasks.md +0 -0
  239. /package/{content → stacks/blazor-azure}/.morph/templates/ui-components.md +0 -0
  240. /package/{content → stacks/blazor-azure}/CLAUDE.md +0 -0
@@ -1,171 +1,171 @@
1
- // ============================================================
2
- // HANGFIRE JOB TEMPLATE
3
- // Generated by MORPH Framework
4
- // ============================================================
5
-
6
- using Hangfire;
7
- using Microsoft.Extensions.Logging;
8
-
9
- namespace MyProject.Application.Features.{Feature}.Jobs;
10
-
11
- /// <summary>
12
- /// Background job for processing {Feature}.
13
- /// Uses Hangfire for scheduling and retry handling.
14
- /// </summary>
15
- public class {Feature}ProcessorJob(
16
- I{Feature}Service service,
17
- I{Feature}AnalyzerAgent analyzer,
18
- ILogger<{Feature}ProcessorJob> logger) : I{Feature}ProcessorJob
19
- {
20
- /// <summary>
21
- /// Executes the {Feature} processing job.
22
- /// </summary>
23
- /// <param name="id">The {Feature} ID to process</param>
24
- /// <param name="cancellationToken">Cancellation token</param>
25
- [AutomaticRetry(Attempts = 3, DelaysInSeconds = new[] { 60, 300, 900 })]
26
- [Queue("default")]
27
- [JobDisplayName("{Feature} Processing - ID: {0}")]
28
- public async Task ExecuteAsync(int id, CancellationToken cancellationToken)
29
- {
30
- logger.LogInformation("Starting {Feature} processing for ID {Id}", id);
31
-
32
- try
33
- {
34
- // Get the entity
35
- var item = await service.GetByIdAsync(id, cancellationToken);
36
- if (item is null)
37
- {
38
- logger.LogWarning("{Feature} with ID {Id} not found, skipping", id);
39
- return;
40
- }
41
-
42
- // Check if already processed
43
- if (item.Status == {Feature}Status.Completed)
44
- {
45
- logger.LogInformation("{Feature} {Id} already completed, skipping", id);
46
- return;
47
- }
48
-
49
- // Perform analysis (if applicable)
50
- var analysisData = new {Feature}Data(item.Name);
51
- var analysis = await analyzer.AnalyzeAsync(analysisData, cancellationToken);
52
-
53
- logger.LogInformation(
54
- "{Feature} {Id} analyzed. Confidence: {Confidence:P0}",
55
- id, analysis.ConfidenceScore);
56
-
57
- // Update status
58
- // Note: You might need to add a method to update with analysis results
59
- // await service.CompleteWithAnalysisAsync(id, analysis, cancellationToken);
60
-
61
- logger.LogInformation("Completed {Feature} processing for ID {Id}", id);
62
- }
63
- catch (Exception ex)
64
- {
65
- logger.LogError(ex, "Failed to process {Feature} {Id}", id);
66
- throw; // Re-throw to trigger Hangfire retry
67
- }
68
- }
69
- }
70
-
71
- // ============================================================
72
- // RECURRING JOB CONFIGURATION
73
- // ============================================================
74
- //
75
- // For scheduled/recurring jobs, configure in Program.cs:
76
- //
77
- // // Run every hour
78
- // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
79
- // "{feature}-processor",
80
- // job => job.ExecuteAsync(0, CancellationToken.None),
81
- // Cron.Hourly);
82
- //
83
- // // Run daily at midnight
84
- // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
85
- // "{feature}-daily-processor",
86
- // job => job.ExecuteAsync(0, CancellationToken.None),
87
- // Cron.Daily);
88
- //
89
- // // Custom cron expression (every 15 minutes)
90
- // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
91
- // "{feature}-frequent-processor",
92
- // job => job.ExecuteAsync(0, CancellationToken.None),
93
- // "*/15 * * * *");
94
- //
95
- // ============================================================
96
-
97
- // ============================================================
98
- // BATCH JOB TEMPLATE
99
- // ============================================================
100
-
101
- /// <summary>
102
- /// Batch job for processing multiple {Feature}s.
103
- /// </summary>
104
- public class {Feature}BatchProcessorJob(
105
- I{Feature}Service service,
106
- I{Feature}ProcessorJob itemProcessor,
107
- ILogger<{Feature}BatchProcessorJob> logger)
108
- {
109
- /// <summary>
110
- /// Processes all pending {Feature}s.
111
- /// </summary>
112
- [AutomaticRetry(Attempts = 1)]
113
- [Queue("batch")]
114
- [JobDisplayName("{Feature} Batch Processing")]
115
- public async Task ProcessAllPendingAsync(CancellationToken cancellationToken)
116
- {
117
- logger.LogInformation("Starting batch processing for pending {Feature}s");
118
-
119
- var items = await service.GetAllAsync(cancellationToken);
120
- var pending = items.Where(x => x.Status == {Feature}Status.Pending).ToList();
121
-
122
- logger.LogInformation("Found {Count} pending {Feature}s to process", pending.Count);
123
-
124
- foreach (var item in pending)
125
- {
126
- // Enqueue individual processing jobs
127
- BackgroundJob.Enqueue<I{Feature}ProcessorJob>(
128
- job => job.ExecuteAsync(item.Id, CancellationToken.None));
129
- }
130
-
131
- logger.LogInformation("Enqueued {Count} processing jobs", pending.Count);
132
- }
133
- }
134
-
135
- // ============================================================
136
- // HANGFIRE CONFIGURATION
137
- // ============================================================
138
- //
139
- // In Program.cs:
140
- //
141
- // // Add Hangfire services
142
- // builder.Services.AddHangfire(config => config
143
- // .SetDataCompatibilityLevel(CompatibilityLevel.Version_180)
144
- // .UseSimpleAssemblyNameTypeSerializer()
145
- // .UseRecommendedSerializerSettings()
146
- // .UseSqlServerStorage(connectionString, new SqlServerStorageOptions
147
- // {
148
- // CommandBatchMaxTimeout = TimeSpan.FromMinutes(5),
149
- // SlidingInvisibilityTimeout = TimeSpan.FromMinutes(5),
150
- // QueuePollInterval = TimeSpan.Zero,
151
- // UseRecommendedIsolationLevel = true,
152
- // DisableGlobalLocks = true
153
- // }));
154
- //
155
- // builder.Services.AddHangfireServer(options =>
156
- // {
157
- // options.Queues = new[] { "default", "batch" };
158
- // options.WorkerCount = Environment.ProcessorCount * 2;
159
- // });
160
- //
161
- // // Register jobs
162
- // builder.Services.AddScoped<I{Feature}ProcessorJob, {Feature}ProcessorJob>();
163
- // builder.Services.AddScoped<{Feature}BatchProcessorJob>();
164
- //
165
- // // In app pipeline
166
- // app.UseHangfireDashboard("/hangfire", new DashboardOptions
167
- // {
168
- // Authorization = new[] { new HangfireAuthorizationFilter() }
169
- // });
170
- //
171
- // ============================================================
1
+ // ============================================================
2
+ // HANGFIRE JOB TEMPLATE
3
+ // Generated by MORPH Framework
4
+ // ============================================================
5
+
6
+ using Hangfire;
7
+ using Microsoft.Extensions.Logging;
8
+
9
+ namespace MyProject.Application.Features.{Feature}.Jobs;
10
+
11
+ /// <summary>
12
+ /// Background job for processing {Feature}.
13
+ /// Uses Hangfire for scheduling and retry handling.
14
+ /// </summary>
15
+ public class {Feature}ProcessorJob(
16
+ I{Feature}Service service,
17
+ I{Feature}AnalyzerAgent analyzer,
18
+ ILogger<{Feature}ProcessorJob> logger) : I{Feature}ProcessorJob
19
+ {
20
+ /// <summary>
21
+ /// Executes the {Feature} processing job.
22
+ /// </summary>
23
+ /// <param name="id">The {Feature} ID to process</param>
24
+ /// <param name="cancellationToken">Cancellation token</param>
25
+ [AutomaticRetry(Attempts = 3, DelaysInSeconds = new[] { 60, 300, 900 })]
26
+ [Queue("default")]
27
+ [JobDisplayName("{Feature} Processing - ID: {0}")]
28
+ public async Task ExecuteAsync(int id, CancellationToken cancellationToken)
29
+ {
30
+ logger.LogInformation("Starting {Feature} processing for ID {Id}", id);
31
+
32
+ try
33
+ {
34
+ // Get the entity
35
+ var item = await service.GetByIdAsync(id, cancellationToken);
36
+ if (item is null)
37
+ {
38
+ logger.LogWarning("{Feature} with ID {Id} not found, skipping", id);
39
+ return;
40
+ }
41
+
42
+ // Check if already processed
43
+ if (item.Status == {Feature}Status.Completed)
44
+ {
45
+ logger.LogInformation("{Feature} {Id} already completed, skipping", id);
46
+ return;
47
+ }
48
+
49
+ // Perform analysis (if applicable)
50
+ var analysisData = new {Feature}Data(item.Name);
51
+ var analysis = await analyzer.AnalyzeAsync(analysisData, cancellationToken);
52
+
53
+ logger.LogInformation(
54
+ "{Feature} {Id} analyzed. Confidence: {Confidence:P0}",
55
+ id, analysis.ConfidenceScore);
56
+
57
+ // Update status
58
+ // Note: You might need to add a method to update with analysis results
59
+ // await service.CompleteWithAnalysisAsync(id, analysis, cancellationToken);
60
+
61
+ logger.LogInformation("Completed {Feature} processing for ID {Id}", id);
62
+ }
63
+ catch (Exception ex)
64
+ {
65
+ logger.LogError(ex, "Failed to process {Feature} {Id}", id);
66
+ throw; // Re-throw to trigger Hangfire retry
67
+ }
68
+ }
69
+ }
70
+
71
+ // ============================================================
72
+ // RECURRING JOB CONFIGURATION
73
+ // ============================================================
74
+ //
75
+ // For scheduled/recurring jobs, configure in Program.cs:
76
+ //
77
+ // // Run every hour
78
+ // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
79
+ // "{feature}-processor",
80
+ // job => job.ExecuteAsync(0, CancellationToken.None),
81
+ // Cron.Hourly);
82
+ //
83
+ // // Run daily at midnight
84
+ // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
85
+ // "{feature}-daily-processor",
86
+ // job => job.ExecuteAsync(0, CancellationToken.None),
87
+ // Cron.Daily);
88
+ //
89
+ // // Custom cron expression (every 15 minutes)
90
+ // RecurringJob.AddOrUpdate<I{Feature}ProcessorJob>(
91
+ // "{feature}-frequent-processor",
92
+ // job => job.ExecuteAsync(0, CancellationToken.None),
93
+ // "*/15 * * * *");
94
+ //
95
+ // ============================================================
96
+
97
+ // ============================================================
98
+ // BATCH JOB TEMPLATE
99
+ // ============================================================
100
+
101
+ /// <summary>
102
+ /// Batch job for processing multiple {Feature}s.
103
+ /// </summary>
104
+ public class {Feature}BatchProcessorJob(
105
+ I{Feature}Service service,
106
+ I{Feature}ProcessorJob itemProcessor,
107
+ ILogger<{Feature}BatchProcessorJob> logger)
108
+ {
109
+ /// <summary>
110
+ /// Processes all pending {Feature}s.
111
+ /// </summary>
112
+ [AutomaticRetry(Attempts = 1)]
113
+ [Queue("batch")]
114
+ [JobDisplayName("{Feature} Batch Processing")]
115
+ public async Task ProcessAllPendingAsync(CancellationToken cancellationToken)
116
+ {
117
+ logger.LogInformation("Starting batch processing for pending {Feature}s");
118
+
119
+ var items = await service.GetAllAsync(cancellationToken);
120
+ var pending = items.Where(x => x.Status == {Feature}Status.Pending).ToList();
121
+
122
+ logger.LogInformation("Found {Count} pending {Feature}s to process", pending.Count);
123
+
124
+ foreach (var item in pending)
125
+ {
126
+ // Enqueue individual processing jobs
127
+ BackgroundJob.Enqueue<I{Feature}ProcessorJob>(
128
+ job => job.ExecuteAsync(item.Id, CancellationToken.None));
129
+ }
130
+
131
+ logger.LogInformation("Enqueued {Count} processing jobs", pending.Count);
132
+ }
133
+ }
134
+
135
+ // ============================================================
136
+ // HANGFIRE CONFIGURATION
137
+ // ============================================================
138
+ //
139
+ // In Program.cs:
140
+ //
141
+ // // Add Hangfire services
142
+ // builder.Services.AddHangfire(config => config
143
+ // .SetDataCompatibilityLevel(CompatibilityLevel.Version_180)
144
+ // .UseSimpleAssemblyNameTypeSerializer()
145
+ // .UseRecommendedSerializerSettings()
146
+ // .UseSqlServerStorage(connectionString, new SqlServerStorageOptions
147
+ // {
148
+ // CommandBatchMaxTimeout = TimeSpan.FromMinutes(5),
149
+ // SlidingInvisibilityTimeout = TimeSpan.FromMinutes(5),
150
+ // QueuePollInterval = TimeSpan.Zero,
151
+ // UseRecommendedIsolationLevel = true,
152
+ // DisableGlobalLocks = true
153
+ // }));
154
+ //
155
+ // builder.Services.AddHangfireServer(options =>
156
+ // {
157
+ // options.Queues = new[] { "default", "batch" };
158
+ // options.WorkerCount = Environment.ProcessorCount * 2;
159
+ // });
160
+ //
161
+ // // Register jobs
162
+ // builder.Services.AddScoped<I{Feature}ProcessorJob, {Feature}ProcessorJob>();
163
+ // builder.Services.AddScoped<{Feature}BatchProcessorJob>();
164
+ //
165
+ // // In app pipeline
166
+ // app.UseHangfireDashboard("/hangfire", new DashboardOptions
167
+ // {
168
+ // Authorization = new[] { new HangfireAuthorizationFilter() }
169
+ // });
170
+ //
171
+ // ============================================================
@@ -1,83 +1,83 @@
1
- // ============================================================
2
- // EF CORE MIGRATION TEMPLATE
3
- // Generated by MORPH Framework
4
- // ============================================================
5
-
6
- using Microsoft.EntityFrameworkCore.Migrations;
7
-
8
- #nullable disable
9
-
10
- namespace MyProject.Infrastructure.Data.Migrations;
11
-
12
- /// <inheritdoc />
13
- public partial class Add{Feature} : Migration
14
- {
15
- /// <inheritdoc />
16
- protected override void Up(MigrationBuilder migrationBuilder)
17
- {
18
- migrationBuilder.CreateTable(
19
- name: "{Feature}s",
20
- columns: table => new
21
- {
22
- Id = table.Column<int>(type: "int", nullable: false)
23
- .Annotation("SqlServer:Identity", "1, 1"),
24
- Name = table.Column<string>(type: "nvarchar(200)", maxLength: 200, nullable: false),
25
- Status = table.Column<string>(type: "nvarchar(50)", maxLength: 50, nullable: false),
26
- CreatedAt = table.Column<DateTime>(type: "datetime2", nullable: false,
27
- defaultValueSql: "GETUTCDATE()"),
28
- UpdatedAt = table.Column<DateTime>(type: "datetime2", nullable: true)
29
- },
30
- constraints: table =>
31
- {
32
- table.PrimaryKey("PK_{Feature}s", x => x.Id);
33
- });
34
-
35
- // Indexes
36
- migrationBuilder.CreateIndex(
37
- name: "IX_{Feature}s_Name",
38
- table: "{Feature}s",
39
- column: "Name");
40
-
41
- migrationBuilder.CreateIndex(
42
- name: "IX_{Feature}s_Status",
43
- table: "{Feature}s",
44
- column: "Status");
45
-
46
- migrationBuilder.CreateIndex(
47
- name: "IX_{Feature}s_CreatedAt",
48
- table: "{Feature}s",
49
- column: "CreatedAt");
50
- }
51
-
52
- /// <inheritdoc />
53
- protected override void Down(MigrationBuilder migrationBuilder)
54
- {
55
- migrationBuilder.DropTable(
56
- name: "{Feature}s");
57
- }
58
- }
59
-
60
- // ============================================================
61
- // HOW TO CREATE A MIGRATION
62
- // ============================================================
63
- //
64
- // 1. Add your entity to AppDbContext:
65
- //
66
- // public DbSet<{Feature}> {Feature}s => Set<{Feature}>();
67
- //
68
- // 2. Add your configuration:
69
- //
70
- // protected override void OnModelCreating(ModelBuilder modelBuilder)
71
- // {
72
- // modelBuilder.ApplyConfiguration(new {Feature}Configuration());
73
- // }
74
- //
75
- // 3. Run migration command:
76
- //
77
- // dotnet ef migrations add Add{Feature} -p src/MyProject.Infrastructure -s src/MyProject.Web
78
- //
79
- // 4. Apply migration:
80
- //
81
- // dotnet ef database update -p src/MyProject.Infrastructure -s src/MyProject.Web
82
- //
83
- // ============================================================
1
+ // ============================================================
2
+ // EF CORE MIGRATION TEMPLATE
3
+ // Generated by MORPH Framework
4
+ // ============================================================
5
+
6
+ using Microsoft.EntityFrameworkCore.Migrations;
7
+
8
+ #nullable disable
9
+
10
+ namespace MyProject.Infrastructure.Data.Migrations;
11
+
12
+ /// <inheritdoc />
13
+ public partial class Add{Feature} : Migration
14
+ {
15
+ /// <inheritdoc />
16
+ protected override void Up(MigrationBuilder migrationBuilder)
17
+ {
18
+ migrationBuilder.CreateTable(
19
+ name: "{Feature}s",
20
+ columns: table => new
21
+ {
22
+ Id = table.Column<int>(type: "int", nullable: false)
23
+ .Annotation("SqlServer:Identity", "1, 1"),
24
+ Name = table.Column<string>(type: "nvarchar(200)", maxLength: 200, nullable: false),
25
+ Status = table.Column<string>(type: "nvarchar(50)", maxLength: 50, nullable: false),
26
+ CreatedAt = table.Column<DateTime>(type: "datetime2", nullable: false,
27
+ defaultValueSql: "GETUTCDATE()"),
28
+ UpdatedAt = table.Column<DateTime>(type: "datetime2", nullable: true)
29
+ },
30
+ constraints: table =>
31
+ {
32
+ table.PrimaryKey("PK_{Feature}s", x => x.Id);
33
+ });
34
+
35
+ // Indexes
36
+ migrationBuilder.CreateIndex(
37
+ name: "IX_{Feature}s_Name",
38
+ table: "{Feature}s",
39
+ column: "Name");
40
+
41
+ migrationBuilder.CreateIndex(
42
+ name: "IX_{Feature}s_Status",
43
+ table: "{Feature}s",
44
+ column: "Status");
45
+
46
+ migrationBuilder.CreateIndex(
47
+ name: "IX_{Feature}s_CreatedAt",
48
+ table: "{Feature}s",
49
+ column: "CreatedAt");
50
+ }
51
+
52
+ /// <inheritdoc />
53
+ protected override void Down(MigrationBuilder migrationBuilder)
54
+ {
55
+ migrationBuilder.DropTable(
56
+ name: "{Feature}s");
57
+ }
58
+ }
59
+
60
+ // ============================================================
61
+ // HOW TO CREATE A MIGRATION
62
+ // ============================================================
63
+ //
64
+ // 1. Add your entity to AppDbContext:
65
+ //
66
+ // public DbSet<{Feature}> {Feature}s => Set<{Feature}>();
67
+ //
68
+ // 2. Add your configuration:
69
+ //
70
+ // protected override void OnModelCreating(ModelBuilder modelBuilder)
71
+ // {
72
+ // modelBuilder.ApplyConfiguration(new {Feature}Configuration());
73
+ // }
74
+ //
75
+ // 3. Run migration command:
76
+ //
77
+ // dotnet ef migrations add Add{Feature} -p src/MyProject.Infrastructure -s src/MyProject.Web
78
+ //
79
+ // 4. Apply migration:
80
+ //
81
+ // dotnet ef database update -p src/MyProject.Infrastructure -s src/MyProject.Web
82
+ //
83
+ // ============================================================