@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,163 +1,163 @@
1
- /**
2
- * Conversation Analyzer - Analyzes Claude Code conversation history for user preferences
3
- *
4
- * Note: This is a basic implementation. Full conversation analysis would require
5
- * access to Claude Code's message history API or local cache.
6
- */
7
-
8
- import { existsSync, readFileSync } from 'fs';
9
- import { join } from 'path';
10
-
11
- /**
12
- * Analyze conversation history
13
- * @param {string} projectPath - Project root path
14
- * @returns {Promise<Object>} Conversation analysis results
15
- */
16
- export async function analyzeConversation(projectPath) {
17
- const result = {
18
- preferences: {},
19
- decisions: [],
20
- patterns: [],
21
- available: false
22
- };
23
-
24
- // Check if .morph/project/context exists (past decisions)
25
- const contextPath = join(projectPath, '.morph', 'project', 'context');
26
- if (existsSync(contextPath)) {
27
- result.available = true;
28
-
29
- // Read decisions-history.md if exists
30
- const decisionsFile = join(contextPath, 'decisions-history.md');
31
- if (existsSync(decisionsFile)) {
32
- const content = readFileSync(decisionsFile, 'utf8');
33
- result.decisions = extractDecisions(content);
34
- }
35
- }
36
-
37
- // Check .morph/project/outputs/*/decisions.md
38
- const outputsPath = join(projectPath, '.morph', 'project', 'outputs');
39
- if (existsSync(outputsPath)) {
40
- const { readdirSync } = await import('fs');
41
- const features = readdirSync(outputsPath, { withFileTypes: true })
42
- .filter(dirent => dirent.isDirectory())
43
- .map(dirent => dirent.name);
44
-
45
- for (const feature of features) {
46
- const decisionsFile = join(outputsPath, feature, 'decisions.md');
47
- if (existsSync(decisionsFile)) {
48
- const content = readFileSync(decisionsFile, 'utf8');
49
- result.decisions.push(...extractDecisions(content));
50
- }
51
- }
52
- }
53
-
54
- // Extract patterns from decisions
55
- result.patterns = inferPatternsFromDecisions(result.decisions);
56
-
57
- // Extract preferences
58
- result.preferences = extractPreferences(result.decisions);
59
-
60
- return result;
61
- }
62
-
63
- /**
64
- * Extract decisions from markdown
65
- */
66
- function extractDecisions(markdown) {
67
- const decisions = [];
68
- const lines = markdown.split('\n');
69
-
70
- let currentDecision = null;
71
-
72
- for (const line of lines) {
73
- // Look for decision headings
74
- if (line.match(/^##\s+Decision/i) || line.match(/^###\s+ADR/i)) {
75
- if (currentDecision) {
76
- decisions.push(currentDecision);
77
- }
78
- currentDecision = { text: '', category: 'unknown', confidence: 0.5 };
79
- }
80
-
81
- if (currentDecision) {
82
- currentDecision.text += line + '\n';
83
-
84
- // Categorize
85
- if (line.includes('UI') || line.includes('component') || line.includes('Fluent') || line.includes('Mud')) {
86
- currentDecision.category = 'ui-ux';
87
- currentDecision.confidence = 0.8;
88
- } else if (line.includes('pattern') || line.includes('architecture') || line.includes('CQRS')) {
89
- currentDecision.category = 'architecture';
90
- currentDecision.confidence = 0.8;
91
- } else if (line.includes('Azure') || line.includes('Container') || line.includes('deployment')) {
92
- currentDecision.category = 'azure';
93
- currentDecision.confidence = 0.8;
94
- } else if (line.includes('class') || line.includes('method') || line.includes('naming')) {
95
- currentDecision.category = 'coding';
96
- currentDecision.confidence = 0.8;
97
- }
98
- }
99
- }
100
-
101
- if (currentDecision) {
102
- decisions.push(currentDecision);
103
- }
104
-
105
- return decisions;
106
- }
107
-
108
- /**
109
- * Infer patterns from decisions
110
- */
111
- function inferPatternsFromDecisions(decisions) {
112
- const patterns = [];
113
-
114
- for (const decision of decisions) {
115
- const text = decision.text.toLowerCase();
116
-
117
- if (text.includes('always use') || text.includes('sempre usar')) {
118
- patterns.push(`Pattern: ${decision.text.split('\n')[0].substring(0, 60)}...`);
119
- }
120
-
121
- if (text.includes('never') || text.includes('nunca')) {
122
- patterns.push(`Anti-pattern: ${decision.text.split('\n')[0].substring(0, 60)}...`);
123
- }
124
- }
125
-
126
- return patterns;
127
- }
128
-
129
- /**
130
- * Extract user preferences
131
- */
132
- function extractPreferences(decisions) {
133
- const preferences = {
134
- uiLibrary: null,
135
- architecture: null,
136
- testing: null,
137
- deployment: null
138
- };
139
-
140
- for (const decision of decisions) {
141
- const text = decision.text.toLowerCase();
142
-
143
- if (text.includes('fluent ui') && !text.includes('not')) {
144
- preferences.uiLibrary = 'Fluent UI';
145
- } else if (text.includes('mudblazor') && !text.includes('not')) {
146
- preferences.uiLibrary = 'MudBlazor';
147
- }
148
-
149
- if (text.includes('clean architecture')) {
150
- preferences.architecture = 'Clean Architecture';
151
- } else if (text.includes('cqrs')) {
152
- preferences.architecture = 'CQRS';
153
- }
154
-
155
- if (text.includes('container apps')) {
156
- preferences.deployment = 'Azure Container Apps';
157
- } else if (text.includes('app service')) {
158
- preferences.deployment = 'Azure App Service';
159
- }
160
- }
161
-
162
- return preferences;
163
- }
1
+ /**
2
+ * Conversation Analyzer - Analyzes Claude Code conversation history for user preferences
3
+ *
4
+ * Note: This is a basic implementation. Full conversation analysis would require
5
+ * access to Claude Code's message history API or local cache.
6
+ */
7
+
8
+ import { existsSync, readFileSync } from 'fs';
9
+ import { join } from 'path';
10
+
11
+ /**
12
+ * Analyze conversation history
13
+ * @param {string} projectPath - Project root path
14
+ * @returns {Promise<Object>} Conversation analysis results
15
+ */
16
+ export async function analyzeConversation(projectPath) {
17
+ const result = {
18
+ preferences: {},
19
+ decisions: [],
20
+ patterns: [],
21
+ available: false
22
+ };
23
+
24
+ // Check if .morph/project/context exists (past decisions)
25
+ const contextPath = join(projectPath, '.morph', 'project', 'context');
26
+ if (existsSync(contextPath)) {
27
+ result.available = true;
28
+
29
+ // Read decisions-history.md if exists
30
+ const decisionsFile = join(contextPath, 'decisions-history.md');
31
+ if (existsSync(decisionsFile)) {
32
+ const content = readFileSync(decisionsFile, 'utf8');
33
+ result.decisions = extractDecisions(content);
34
+ }
35
+ }
36
+
37
+ // Check .morph/project/outputs/*/decisions.md
38
+ const outputsPath = join(projectPath, '.morph', 'project', 'outputs');
39
+ if (existsSync(outputsPath)) {
40
+ const { readdirSync } = await import('fs');
41
+ const features = readdirSync(outputsPath, { withFileTypes: true })
42
+ .filter(dirent => dirent.isDirectory())
43
+ .map(dirent => dirent.name);
44
+
45
+ for (const feature of features) {
46
+ const decisionsFile = join(outputsPath, feature, 'decisions.md');
47
+ if (existsSync(decisionsFile)) {
48
+ const content = readFileSync(decisionsFile, 'utf8');
49
+ result.decisions.push(...extractDecisions(content));
50
+ }
51
+ }
52
+ }
53
+
54
+ // Extract patterns from decisions
55
+ result.patterns = inferPatternsFromDecisions(result.decisions);
56
+
57
+ // Extract preferences
58
+ result.preferences = extractPreferences(result.decisions);
59
+
60
+ return result;
61
+ }
62
+
63
+ /**
64
+ * Extract decisions from markdown
65
+ */
66
+ function extractDecisions(markdown) {
67
+ const decisions = [];
68
+ const lines = markdown.split('\n');
69
+
70
+ let currentDecision = null;
71
+
72
+ for (const line of lines) {
73
+ // Look for decision headings
74
+ if (line.match(/^##\s+Decision/i) || line.match(/^###\s+ADR/i)) {
75
+ if (currentDecision) {
76
+ decisions.push(currentDecision);
77
+ }
78
+ currentDecision = { text: '', category: 'unknown', confidence: 0.5 };
79
+ }
80
+
81
+ if (currentDecision) {
82
+ currentDecision.text += line + '\n';
83
+
84
+ // Categorize
85
+ if (line.includes('UI') || line.includes('component') || line.includes('Fluent') || line.includes('Mud')) {
86
+ currentDecision.category = 'ui-ux';
87
+ currentDecision.confidence = 0.8;
88
+ } else if (line.includes('pattern') || line.includes('architecture') || line.includes('CQRS')) {
89
+ currentDecision.category = 'architecture';
90
+ currentDecision.confidence = 0.8;
91
+ } else if (line.includes('Azure') || line.includes('Container') || line.includes('deployment')) {
92
+ currentDecision.category = 'azure';
93
+ currentDecision.confidence = 0.8;
94
+ } else if (line.includes('class') || line.includes('method') || line.includes('naming')) {
95
+ currentDecision.category = 'coding';
96
+ currentDecision.confidence = 0.8;
97
+ }
98
+ }
99
+ }
100
+
101
+ if (currentDecision) {
102
+ decisions.push(currentDecision);
103
+ }
104
+
105
+ return decisions;
106
+ }
107
+
108
+ /**
109
+ * Infer patterns from decisions
110
+ */
111
+ function inferPatternsFromDecisions(decisions) {
112
+ const patterns = [];
113
+
114
+ for (const decision of decisions) {
115
+ const text = decision.text.toLowerCase();
116
+
117
+ if (text.includes('always use') || text.includes('sempre usar')) {
118
+ patterns.push(`Pattern: ${decision.text.split('\n')[0].substring(0, 60)}...`);
119
+ }
120
+
121
+ if (text.includes('never') || text.includes('nunca')) {
122
+ patterns.push(`Anti-pattern: ${decision.text.split('\n')[0].substring(0, 60)}...`);
123
+ }
124
+ }
125
+
126
+ return patterns;
127
+ }
128
+
129
+ /**
130
+ * Extract user preferences
131
+ */
132
+ function extractPreferences(decisions) {
133
+ const preferences = {
134
+ uiLibrary: null,
135
+ architecture: null,
136
+ testing: null,
137
+ deployment: null
138
+ };
139
+
140
+ for (const decision of decisions) {
141
+ const text = decision.text.toLowerCase();
142
+
143
+ if (text.includes('fluent ui') && !text.includes('not')) {
144
+ preferences.uiLibrary = 'Fluent UI';
145
+ } else if (text.includes('mudblazor') && !text.includes('not')) {
146
+ preferences.uiLibrary = 'MudBlazor';
147
+ }
148
+
149
+ if (text.includes('clean architecture')) {
150
+ preferences.architecture = 'Clean Architecture';
151
+ } else if (text.includes('cqrs')) {
152
+ preferences.architecture = 'CQRS';
153
+ }
154
+
155
+ if (text.includes('container apps')) {
156
+ preferences.deployment = 'Azure Container Apps';
157
+ } else if (text.includes('app service')) {
158
+ preferences.deployment = 'Azure App Service';
159
+ }
160
+ }
161
+
162
+ return preferences;
163
+ }
@@ -1,84 +1,84 @@
1
- /**
2
- * MORPH-SPEC Detectors - Main Orchestrator
3
- *
4
- * Coordinates all detection modules to build a complete picture of the project.
5
- */
6
-
7
- import { detectStructure } from './structure-detector.js';
8
- import { detectConfig } from './config-detector.js';
9
- import { analyzeConversation } from './conversation-analyzer.js';
10
- import { generateStandards } from './standards-generator.js';
11
-
12
- /**
13
- * Main detection orchestrator
14
- * @param {string} projectPath - Path to project root
15
- * @param {Object} options - Detection options
16
- * @returns {Promise<Object>} Detection results
17
- */
18
- export async function detectProject(projectPath, options = {}) {
19
- const results = {
20
- path: projectPath,
21
- timestamp: new Date().toISOString(),
22
- structure: null,
23
- config: null,
24
- conversation: null,
25
- inferred: null
26
- };
27
-
28
- try {
29
- // 1. Detect structure (stack, architecture, patterns)
30
- if (options.structure !== false) {
31
- results.structure = await detectStructure(projectPath);
32
- }
33
-
34
- // 2. Detect config (technologies, dependencies, versions)
35
- if (options.config !== false) {
36
- results.config = await detectConfig(projectPath);
37
- }
38
-
39
- // 3. Analyze conversation history (if available)
40
- if (options.conversation !== false) {
41
- results.conversation = await analyzeConversation(projectPath);
42
- }
43
-
44
- // 4. Generate inferred standards
45
- if (options.generateStandards !== false) {
46
- results.inferred = await generateStandards(results);
47
- }
48
-
49
- return results;
50
- } catch (error) {
51
- throw new Error(`Detection failed: ${error.message}`);
52
- }
53
- }
54
-
55
- /**
56
- * Get detection summary
57
- * @param {Object} results - Detection results
58
- * @returns {string} Human-readable summary
59
- */
60
- export function getDetectionSummary(results) {
61
- const { structure, config } = results;
62
-
63
- const lines = [
64
- '## Detection Summary',
65
- '',
66
- '### Stack',
67
- `- **Type**: ${structure?.stack || 'unknown'}`,
68
- `- **Architecture**: ${structure?.architecture || 'unknown'}`,
69
- `- **UI Library**: ${structure?.uiLibrary || 'none'}`,
70
- '',
71
- '### Technologies',
72
- `- **Language**: ${config?.language || 'unknown'}`,
73
- `- **Version**: ${config?.version || 'unknown'}`,
74
- `- **Package Manager**: ${config?.packageManager || 'unknown'}`,
75
- '',
76
- '### Patterns Detected',
77
- ...(structure?.patterns || []).map(p => `- ${p}`),
78
- '',
79
- '### Recommendations',
80
- ...(results.inferred?.recommendations || []).map(r => `- ${r}`)
81
- ];
82
-
83
- return lines.join('\n');
84
- }
1
+ /**
2
+ * MORPH-SPEC Detectors - Main Orchestrator
3
+ *
4
+ * Coordinates all detection modules to build a complete picture of the project.
5
+ */
6
+
7
+ import { detectStructure } from './structure-detector.js';
8
+ import { detectConfig } from './config-detector.js';
9
+ import { analyzeConversation } from './conversation-analyzer.js';
10
+ import { generateStandards } from './standards-generator.js';
11
+
12
+ /**
13
+ * Main detection orchestrator
14
+ * @param {string} projectPath - Path to project root
15
+ * @param {Object} options - Detection options
16
+ * @returns {Promise<Object>} Detection results
17
+ */
18
+ export async function detectProject(projectPath, options = {}) {
19
+ const results = {
20
+ path: projectPath,
21
+ timestamp: new Date().toISOString(),
22
+ structure: null,
23
+ config: null,
24
+ conversation: null,
25
+ inferred: null
26
+ };
27
+
28
+ try {
29
+ // 1. Detect structure (stack, architecture, patterns)
30
+ if (options.structure !== false) {
31
+ results.structure = await detectStructure(projectPath);
32
+ }
33
+
34
+ // 2. Detect config (technologies, dependencies, versions)
35
+ if (options.config !== false) {
36
+ results.config = await detectConfig(projectPath);
37
+ }
38
+
39
+ // 3. Analyze conversation history (if available)
40
+ if (options.conversation !== false) {
41
+ results.conversation = await analyzeConversation(projectPath);
42
+ }
43
+
44
+ // 4. Generate inferred standards
45
+ if (options.generateStandards !== false) {
46
+ results.inferred = await generateStandards(results);
47
+ }
48
+
49
+ return results;
50
+ } catch (error) {
51
+ throw new Error(`Detection failed: ${error.message}`);
52
+ }
53
+ }
54
+
55
+ /**
56
+ * Get detection summary
57
+ * @param {Object} results - Detection results
58
+ * @returns {string} Human-readable summary
59
+ */
60
+ export function getDetectionSummary(results) {
61
+ const { structure, config } = results;
62
+
63
+ const lines = [
64
+ '## Detection Summary',
65
+ '',
66
+ '### Stack',
67
+ `- **Type**: ${structure?.stack || 'unknown'}`,
68
+ `- **Architecture**: ${structure?.architecture || 'unknown'}`,
69
+ `- **UI Library**: ${structure?.uiLibrary || 'none'}`,
70
+ '',
71
+ '### Technologies',
72
+ `- **Language**: ${config?.language || 'unknown'}`,
73
+ `- **Version**: ${config?.version || 'unknown'}`,
74
+ `- **Package Manager**: ${config?.packageManager || 'unknown'}`,
75
+ '',
76
+ '### Patterns Detected',
77
+ ...(structure?.patterns || []).map(p => `- ${p}`),
78
+ '',
79
+ '### Recommendations',
80
+ ...(results.inferred?.recommendations || []).map(r => `- ${r}`)
81
+ ];
82
+
83
+ return lines.join('\n');
84
+ }