@hivehub/rulebook 5.7.0 → 5.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (600) hide show
  1. package/.claude/commands/analysis.md +35 -35
  2. package/.claude/commands/continue.md +33 -33
  3. package/.claude/commands/rulebook-decision-create.md +55 -55
  4. package/.claude/commands/rulebook-decision-list.md +15 -15
  5. package/.claude/commands/rulebook-knowledge-add.md +41 -41
  6. package/.claude/commands/rulebook-knowledge-list.md +15 -15
  7. package/.claude/commands/rulebook-memory-save.md +48 -48
  8. package/.claude/commands/rulebook-memory-search.md +47 -47
  9. package/.claude/commands/rulebook-task-apply.md +67 -67
  10. package/.claude/commands/rulebook-task-archive.md +94 -94
  11. package/.claude/commands/rulebook-task-create.md +93 -93
  12. package/.claude/commands/rulebook-task-list.md +42 -42
  13. package/.claude/commands/rulebook-task-show.md +52 -52
  14. package/.claude/commands/rulebook-task-validate.md +53 -53
  15. package/.claude-plugin/marketplace.json +28 -28
  16. package/.claude-plugin/plugin.json +8 -8
  17. package/README.md +86 -0
  18. package/dist/cli/commands/claude.d.ts +17 -0
  19. package/dist/cli/commands/claude.d.ts.map +1 -0
  20. package/dist/cli/commands/claude.js +56 -0
  21. package/dist/cli/commands/claude.js.map +1 -0
  22. package/dist/cli/commands/init.d.ts.map +1 -1
  23. package/dist/cli/commands/init.js +18 -2
  24. package/dist/cli/commands/init.js.map +1 -1
  25. package/dist/cli/commands/update.d.ts.map +1 -1
  26. package/dist/cli/commands/update.js +6 -1
  27. package/dist/cli/commands/update.js.map +1 -1
  28. package/dist/cli/prompts.d.ts +13 -0
  29. package/dist/cli/prompts.d.ts.map +1 -1
  30. package/dist/cli/prompts.js +106 -0
  31. package/dist/cli/prompts.js.map +1 -1
  32. package/dist/core/claude/claude-mcp.d.ts +10 -1
  33. package/dist/core/claude/claude-mcp.d.ts.map +1 -1
  34. package/dist/core/claude/claude-mcp.js +48 -2
  35. package/dist/core/claude/claude-mcp.js.map +1 -1
  36. package/dist/core/claude/claude-settings-manager.d.ts +12 -0
  37. package/dist/core/claude/claude-settings-manager.d.ts.map +1 -1
  38. package/dist/core/claude/claude-settings-manager.js +59 -3
  39. package/dist/core/claude/claude-settings-manager.js.map +1 -1
  40. package/dist/core/detect/detector.d.ts +8 -1
  41. package/dist/core/detect/detector.d.ts.map +1 -1
  42. package/dist/core/detect/detector.js +225 -0
  43. package/dist/core/detect/detector.js.map +1 -1
  44. package/dist/core/detect/library-registry.d.ts +40 -0
  45. package/dist/core/detect/library-registry.d.ts.map +1 -0
  46. package/dist/core/detect/library-registry.js +239 -0
  47. package/dist/core/detect/library-registry.js.map +1 -0
  48. package/dist/core/generators/generator.d.ts +2 -1
  49. package/dist/core/generators/generator.d.ts.map +1 -1
  50. package/dist/core/generators/generator.js +38 -1
  51. package/dist/core/generators/generator.js.map +1 -1
  52. package/dist/core/generators/rules-generator.d.ts +1 -5
  53. package/dist/core/generators/rules-generator.d.ts.map +1 -1
  54. package/dist/core/generators/rules-generator.js +40 -1
  55. package/dist/core/generators/rules-generator.js.map +1 -1
  56. package/dist/index.js +12 -0
  57. package/dist/index.js.map +1 -1
  58. package/dist/mcp/rulebook-server.js +0 -0
  59. package/dist/types.d.ts +13 -0
  60. package/dist/types.d.ts.map +1 -1
  61. package/package.json +23 -22
  62. package/templates/agents/accessibility-reviewer.md +43 -43
  63. package/templates/agents/api-designer.md +42 -42
  64. package/templates/agents/architect.md +51 -51
  65. package/templates/agents/build-engineer.md +36 -36
  66. package/templates/agents/code-reviewer.md +47 -47
  67. package/templates/agents/compiler/codegen-debugger.md +34 -34
  68. package/templates/agents/compiler/stdlib-engineer.md +28 -28
  69. package/templates/agents/compiler/test-coverage-guardian.md +31 -31
  70. package/templates/agents/database-architect.md +41 -41
  71. package/templates/agents/devops-engineer.md +42 -42
  72. package/templates/agents/docs-writer.md +38 -38
  73. package/templates/agents/game-engine/cpp-core-expert.md +35 -35
  74. package/templates/agents/game-engine/render-engineer.md +22 -22
  75. package/templates/agents/game-engine/shader-engineer.md +38 -38
  76. package/templates/agents/game-engine/systems-integration.md +43 -43
  77. package/templates/agents/generic/code-reviewer.md +41 -41
  78. package/templates/agents/generic/docs-writer.md +25 -25
  79. package/templates/agents/generic/project-manager.md +36 -36
  80. package/templates/agents/generic/researcher.md +34 -34
  81. package/templates/agents/generic/test-engineer.md +41 -41
  82. package/templates/agents/i18n-engineer.md +42 -42
  83. package/templates/agents/implementer.md +42 -42
  84. package/templates/agents/migration-engineer.md +42 -42
  85. package/templates/agents/mobile/platform-specialist.md +22 -22
  86. package/templates/agents/mobile/ui-engineer.md +22 -22
  87. package/templates/agents/performance-engineer.md +49 -49
  88. package/templates/agents/project-manager.md +217 -0
  89. package/templates/agents/quality-gatekeeper.md +208 -0
  90. package/templates/agents/refactoring-agent.md +41 -41
  91. package/templates/agents/researcher.md +38 -38
  92. package/templates/agents/security-reviewer.md +40 -40
  93. package/templates/agents/team-lead.md +37 -37
  94. package/templates/agents/tester.md +48 -48
  95. package/templates/agents/ux-reviewer.md +43 -43
  96. package/templates/agents/web-app/api-designer.md +22 -22
  97. package/templates/agents/web-app/backend-engineer.md +30 -30
  98. package/templates/agents/web-app/database-engineer.md +22 -22
  99. package/templates/agents/web-app/frontend-engineer.md +29 -29
  100. package/templates/agents/web-app/security-reviewer.md +32 -32
  101. package/templates/ci/rulebook-review.yml +26 -26
  102. package/templates/claude-workflows/bugfix.js +94 -0
  103. package/templates/claude-workflows/feature-pipeline.js +88 -0
  104. package/templates/claude-workflows/release-gate.js +64 -0
  105. package/templates/claude-workflows/review-fanout.js +125 -0
  106. package/templates/claude-workflows/rulebook-driver.js +382 -0
  107. package/templates/claude-workflows/spec-author.js +133 -0
  108. package/templates/cli/AIDER.md +49 -49
  109. package/templates/cli/AMAZON_Q.md +25 -25
  110. package/templates/cli/AUGGIE.md +32 -32
  111. package/templates/cli/CLAUDE.md +117 -117
  112. package/templates/cli/CLINE.md +99 -99
  113. package/templates/cli/CODEBUDDY.md +20 -20
  114. package/templates/cli/CODEIUM.md +20 -20
  115. package/templates/cli/CODEX.md +21 -21
  116. package/templates/cli/CONTINUE.md +34 -34
  117. package/templates/cli/CURSOR_CLI.md +62 -62
  118. package/templates/cli/FACTORY.md +18 -18
  119. package/templates/cli/GEMINI.md +35 -35
  120. package/templates/cli/KILOCODE.md +18 -18
  121. package/templates/cli/_GENERIC_TEMPLATE.md +29 -29
  122. package/templates/commands/rulebook-decision-create.md +55 -55
  123. package/templates/commands/rulebook-decision-list.md +15 -15
  124. package/templates/commands/rulebook-knowledge-add.md +41 -41
  125. package/templates/commands/rulebook-knowledge-list.md +15 -15
  126. package/templates/commands/rulebook-memory-save.md +48 -48
  127. package/templates/commands/rulebook-memory-search.md +47 -47
  128. package/templates/commands/rulebook-task-apply.md +67 -67
  129. package/templates/commands/rulebook-task-archive.md +94 -94
  130. package/templates/commands/rulebook-task-create.md +93 -93
  131. package/templates/commands/rulebook-task-list.md +42 -42
  132. package/templates/commands/rulebook-task-show.md +52 -52
  133. package/templates/commands/rulebook-task-validate.md +53 -53
  134. package/templates/compact-context/_default.md +23 -23
  135. package/templates/compact-context/cpp.md +26 -26
  136. package/templates/compact-context/go.md +26 -26
  137. package/templates/compact-context/python.md +26 -26
  138. package/templates/compact-context/rust.md +28 -28
  139. package/templates/compact-context/typescript.md +29 -29
  140. package/templates/core/AGENTS_OVERRIDE.md +16 -16
  141. package/templates/core/AGENT_AUTOMATION.md +296 -296
  142. package/templates/core/CLAUDE_MD_v2.md +90 -90
  143. package/templates/core/DAG.md +304 -304
  144. package/templates/core/DECISIONS.md +38 -38
  145. package/templates/core/DOCUMENTATION_RULES.md +36 -36
  146. package/templates/core/KNOWLEDGE.md +49 -49
  147. package/templates/core/MULTI_AGENT.md +74 -74
  148. package/templates/core/PLANS.md +28 -28
  149. package/templates/core/QUALITY_ENFORCEMENT.md +68 -68
  150. package/templates/core/RULEBOOK.md +1947 -1947
  151. package/templates/core/TIER1_PROHIBITIONS.md +154 -154
  152. package/templates/core/TOKEN_OPTIMIZATION.md +49 -49
  153. package/templates/git/CI_CD_PATTERNS.md +661 -661
  154. package/templates/git/GITHUB_ACTIONS.md +728 -728
  155. package/templates/git/GITLAB_CI.md +730 -730
  156. package/templates/git/GIT_WORKFLOW.md +1192 -1192
  157. package/templates/git/SECRETS_MANAGEMENT.md +585 -585
  158. package/templates/hooks/COMMIT_MSG.md +530 -530
  159. package/templates/hooks/POST_CHECKOUT.md +546 -546
  160. package/templates/hooks/PREPARE_COMMIT_MSG.md +619 -619
  161. package/templates/hooks/PRE_COMMIT.md +414 -414
  162. package/templates/hooks/PRE_PUSH.md +601 -601
  163. package/templates/hooks/check-context-and-handoff.sh +16 -6
  164. package/templates/hooks/update-check.ps1 +84 -0
  165. package/templates/hooks/update-check.sh +103 -0
  166. package/templates/ides/CONTINUE_RULES.md +16 -16
  167. package/templates/ides/COPILOT_INSTRUCTIONS.md +23 -23
  168. package/templates/ides/GEMINI_RULES.md +17 -17
  169. package/templates/ides/WINDSURF_RULES.md +14 -14
  170. package/templates/languages/C.md +333 -333
  171. package/templates/languages/CPP.md +743 -743
  172. package/templates/languages/CSHARP.md +417 -417
  173. package/templates/languages/ELIXIR.md +454 -454
  174. package/templates/languages/ERLANG.md +361 -361
  175. package/templates/languages/GO.md +645 -645
  176. package/templates/languages/HASKELL.md +177 -177
  177. package/templates/languages/JAVA.md +607 -607
  178. package/templates/languages/JAVASCRIPT.md +631 -631
  179. package/templates/languages/JULIA.md +97 -97
  180. package/templates/languages/KOTLIN.md +511 -511
  181. package/templates/languages/LISP.md +100 -100
  182. package/templates/languages/LUA.md +74 -74
  183. package/templates/languages/OBJECTIVEC.md +90 -90
  184. package/templates/languages/PHP.md +416 -416
  185. package/templates/languages/PYTHON.md +682 -682
  186. package/templates/languages/RUBY.md +421 -421
  187. package/templates/languages/RUST.md +477 -477
  188. package/templates/languages/SAS.md +73 -73
  189. package/templates/languages/SCALA.md +348 -348
  190. package/templates/languages/SOLIDITY.md +580 -580
  191. package/templates/languages/SQL.md +137 -137
  192. package/templates/languages/SWIFT.md +466 -466
  193. package/templates/languages/TYPESCRIPT.md +591 -591
  194. package/templates/languages/ZIG.md +265 -265
  195. package/templates/libraries/go/ECHO.md +18 -0
  196. package/templates/libraries/go/GIN.md +18 -0
  197. package/templates/libraries/go/GORM.md +18 -0
  198. package/templates/libraries/python/DJANGO.md +18 -0
  199. package/templates/libraries/python/FASTAPI.md +18 -0
  200. package/templates/libraries/python/FLASK.md +17 -0
  201. package/templates/libraries/python/PYDANTIC.md +17 -0
  202. package/templates/libraries/python/PYTEST.md +17 -0
  203. package/templates/libraries/python/SQLALCHEMY.md +17 -0
  204. package/templates/libraries/rust/ACTIX.md +17 -0
  205. package/templates/libraries/rust/AXUM.md +18 -0
  206. package/templates/libraries/rust/SERDE.md +16 -0
  207. package/templates/libraries/rust/SQLX.md +17 -0
  208. package/templates/libraries/rust/TOKIO.md +16 -0
  209. package/templates/libraries/typescript/ANGULAR.md +17 -0
  210. package/templates/libraries/typescript/DRIZZLE.md +16 -0
  211. package/templates/libraries/typescript/EXPRESS.md +17 -0
  212. package/templates/libraries/typescript/HEROUI.md +16 -0
  213. package/templates/libraries/typescript/JEST.md +17 -0
  214. package/templates/libraries/typescript/NESTJS.md +17 -0
  215. package/templates/libraries/typescript/NEXT.md +18 -0
  216. package/templates/libraries/typescript/PRISMA.md +16 -0
  217. package/templates/libraries/typescript/RADIX.md +16 -0
  218. package/templates/libraries/typescript/REACT.md +18 -0
  219. package/templates/libraries/typescript/SHADCN.md +16 -0
  220. package/templates/libraries/typescript/SVELTE.md +16 -0
  221. package/templates/libraries/typescript/TAILWIND.md +16 -0
  222. package/templates/libraries/typescript/TRPC.md +16 -0
  223. package/templates/libraries/typescript/VITEST.md +17 -0
  224. package/templates/libraries/typescript/VUE.md +17 -0
  225. package/templates/libraries/typescript/ZOD.md +17 -0
  226. package/templates/modules/ATLASSIAN.md +255 -255
  227. package/templates/modules/CONTEXT7.md +54 -54
  228. package/templates/modules/FIGMA.md +267 -267
  229. package/templates/modules/GITHUB_MCP.md +64 -64
  230. package/templates/modules/GRAFANA.md +328 -328
  231. package/templates/modules/MEMORY.md +126 -126
  232. package/templates/modules/NOTION.md +247 -247
  233. package/templates/modules/PLAYWRIGHT.md +90 -90
  234. package/templates/modules/RULEBOOK_MCP.md +208 -208
  235. package/templates/modules/SERENA.md +337 -337
  236. package/templates/modules/SUPABASE.md +223 -223
  237. package/templates/modules/SYNAP.md +69 -69
  238. package/templates/modules/VECTORIZER.md +63 -63
  239. package/templates/modules/sequential-thinking.md +42 -42
  240. package/templates/rules/consult-analysis-before-implementing.md +23 -23
  241. package/templates/rules/cpp.md +46 -46
  242. package/templates/rules/csharp.md +44 -44
  243. package/templates/rules/diagnostic-first.md +39 -39
  244. package/templates/rules/fail-twice-escalate.md +46 -46
  245. package/templates/rules/follow-task-sequence.md +36 -36
  246. package/templates/rules/git-safety.md +29 -29
  247. package/templates/rules/go.md +40 -40
  248. package/templates/rules/incremental-implementation.md +56 -56
  249. package/templates/rules/incremental-tests.md +29 -29
  250. package/templates/rules/java.md +43 -43
  251. package/templates/rules/javascript.md +39 -39
  252. package/templates/rules/knowledge-base-usage.md +41 -41
  253. package/templates/rules/multi-agent-teams.md +75 -75
  254. package/templates/rules/no-deferred.md +31 -31
  255. package/templates/rules/no-shortcuts.md +30 -30
  256. package/templates/rules/python.md +43 -43
  257. package/templates/rules/research-first.md +30 -30
  258. package/templates/rules/respect-handoff-trigger.md +41 -41
  259. package/templates/rules/rust.md +40 -40
  260. package/templates/rules/sequential-editing.md +21 -21
  261. package/templates/rules/session-workflow.md +24 -24
  262. package/templates/rules/task-decomposition.md +32 -32
  263. package/templates/rules/typescript.md +40 -40
  264. package/templates/skills/cli/aider/SKILL.md +59 -59
  265. package/templates/skills/cli/amazon-q/SKILL.md +35 -35
  266. package/templates/skills/cli/auggie/SKILL.md +42 -42
  267. package/templates/skills/cli/claude/SKILL.md +42 -42
  268. package/templates/skills/cli/cline/SKILL.md +42 -42
  269. package/templates/skills/cli/codebuddy/SKILL.md +30 -30
  270. package/templates/skills/cli/codeium/SKILL.md +30 -30
  271. package/templates/skills/cli/codex/SKILL.md +31 -31
  272. package/templates/skills/cli/continue/SKILL.md +44 -44
  273. package/templates/skills/cli/cursor-cli/SKILL.md +38 -38
  274. package/templates/skills/cli/factory/SKILL.md +28 -28
  275. package/templates/skills/cli/gemini/SKILL.md +45 -45
  276. package/templates/skills/cli/kilocode/SKILL.md +28 -28
  277. package/templates/skills/core/agent-automation/SKILL.md +194 -194
  278. package/templates/skills/core/dag/SKILL.md +314 -314
  279. package/templates/skills/core/documentation-rules/SKILL.md +46 -46
  280. package/templates/skills/core/quality-enforcement/SKILL.md +78 -78
  281. package/templates/skills/core/rulebook/SKILL.md +176 -176
  282. package/templates/skills/core/rulebook-terse/SKILL.md +116 -116
  283. package/templates/skills/core/rulebook-terse-commit/SKILL.md +96 -96
  284. package/templates/skills/core/rulebook-terse-review/SKILL.md +112 -112
  285. package/templates/skills/dev/accessibility/SKILL.md +17 -17
  286. package/templates/skills/dev/analysis/SKILL.md +19 -19
  287. package/templates/skills/dev/api-design/SKILL.md +15 -15
  288. package/templates/skills/dev/architect/SKILL.md +17 -17
  289. package/templates/skills/dev/build-fix/SKILL.md +17 -17
  290. package/templates/skills/dev/db-design/SKILL.md +15 -15
  291. package/templates/skills/dev/debug/SKILL.md +16 -16
  292. package/templates/skills/dev/deploy/SKILL.md +17 -17
  293. package/templates/skills/dev/docs/SKILL.md +17 -17
  294. package/templates/skills/dev/handoff/SKILL.md +27 -27
  295. package/templates/skills/dev/migrate/SKILL.md +15 -15
  296. package/templates/skills/dev/perf/SKILL.md +17 -17
  297. package/templates/skills/dev/refactor/SKILL.md +17 -17
  298. package/templates/skills/dev/research/SKILL.md +14 -14
  299. package/templates/skills/dev/review/SKILL.md +18 -18
  300. package/templates/skills/dev/security-audit/SKILL.md +17 -17
  301. package/templates/skills/dev/spec/SKILL.md +65 -0
  302. package/templates/skills/ides/copilot/SKILL.md +47 -47
  303. package/templates/skills/ides/cursor/SKILL.md +53 -53
  304. package/templates/skills/ides/jetbrains-ai/SKILL.md +45 -45
  305. package/templates/skills/ides/replit/SKILL.md +46 -46
  306. package/templates/skills/ides/tabnine/SKILL.md +39 -39
  307. package/templates/skills/ides/vscode/SKILL.md +50 -50
  308. package/templates/skills/ides/windsurf/SKILL.md +46 -46
  309. package/templates/skills/ides/zed/SKILL.md +42 -42
  310. package/templates/skills/languages/c/SKILL.md +343 -343
  311. package/templates/skills/languages/cpp/SKILL.md +753 -753
  312. package/templates/skills/languages/csharp/SKILL.md +427 -427
  313. package/templates/skills/languages/elixir/SKILL.md +464 -464
  314. package/templates/skills/languages/erlang/SKILL.md +371 -371
  315. package/templates/skills/languages/go/SKILL.md +655 -655
  316. package/templates/skills/languages/haskell/SKILL.md +187 -187
  317. package/templates/skills/languages/java/SKILL.md +617 -617
  318. package/templates/skills/languages/javascript/SKILL.md +641 -641
  319. package/templates/skills/languages/julia/SKILL.md +107 -107
  320. package/templates/skills/languages/kotlin/SKILL.md +521 -521
  321. package/templates/skills/languages/lisp/SKILL.md +110 -110
  322. package/templates/skills/languages/lua/SKILL.md +84 -84
  323. package/templates/skills/languages/objectivec/SKILL.md +100 -100
  324. package/templates/skills/languages/php/SKILL.md +426 -426
  325. package/templates/skills/languages/python/SKILL.md +692 -692
  326. package/templates/skills/languages/ruby/SKILL.md +431 -431
  327. package/templates/skills/languages/rust/SKILL.md +487 -487
  328. package/templates/skills/languages/sas/SKILL.md +83 -83
  329. package/templates/skills/languages/scala/SKILL.md +358 -358
  330. package/templates/skills/languages/solidity/SKILL.md +590 -590
  331. package/templates/skills/languages/sql/SKILL.md +147 -147
  332. package/templates/skills/languages/swift/SKILL.md +476 -476
  333. package/templates/skills/languages/typescript/SKILL.md +302 -302
  334. package/templates/skills/languages/zig/SKILL.md +275 -275
  335. package/templates/skills/modules/atlassian/SKILL.md +265 -265
  336. package/templates/skills/modules/context7/SKILL.md +64 -64
  337. package/templates/skills/modules/figma/SKILL.md +277 -277
  338. package/templates/skills/modules/github-mcp/SKILL.md +74 -74
  339. package/templates/skills/modules/grafana/SKILL.md +338 -338
  340. package/templates/skills/modules/memory/SKILL.md +73 -73
  341. package/templates/skills/modules/notion/SKILL.md +257 -257
  342. package/templates/skills/modules/playwright/SKILL.md +100 -100
  343. package/templates/skills/modules/rulebook-mcp/SKILL.md +166 -166
  344. package/templates/skills/modules/serena/SKILL.md +347 -347
  345. package/templates/skills/modules/supabase/SKILL.md +233 -233
  346. package/templates/skills/modules/synap/SKILL.md +79 -79
  347. package/templates/skills/modules/vectorizer/SKILL.md +73 -73
  348. package/templates/workflows/typescript-test.yml +9 -3
  349. package/dist/agents/ralph-parser.d.ts +0 -91
  350. package/dist/agents/ralph-parser.d.ts.map +0 -1
  351. package/dist/agents/ralph-parser.js +0 -415
  352. package/dist/agents/ralph-parser.js.map +0 -1
  353. package/dist/cli/commands/analysis.d.ts +0 -8
  354. package/dist/cli/commands/analysis.d.ts.map +0 -1
  355. package/dist/cli/commands/analysis.js +0 -78
  356. package/dist/cli/commands/analysis.js.map +0 -1
  357. package/dist/cli/commands/compress.d.ts +0 -18
  358. package/dist/cli/commands/compress.d.ts.map +0 -1
  359. package/dist/cli/commands/compress.js +0 -100
  360. package/dist/cli/commands/compress.js.map +0 -1
  361. package/dist/cli/commands/ralph.d.ts +0 -45
  362. package/dist/cli/commands/ralph.d.ts.map +0 -1
  363. package/dist/cli/commands/ralph.js +0 -694
  364. package/dist/cli/commands/ralph.js.map +0 -1
  365. package/dist/cli/docs-prompts.d.ts +0 -3
  366. package/dist/cli/docs-prompts.d.ts.map +0 -1
  367. package/dist/cli/docs-prompts.js +0 -45
  368. package/dist/cli/docs-prompts.js.map +0 -1
  369. package/dist/core/agent-manager.d.ts +0 -69
  370. package/dist/core/agent-manager.d.ts.map +0 -1
  371. package/dist/core/agent-manager.js +0 -476
  372. package/dist/core/agent-manager.js.map +0 -1
  373. package/dist/core/agent-template-engine.d.ts +0 -51
  374. package/dist/core/agent-template-engine.d.ts.map +0 -1
  375. package/dist/core/agent-template-engine.js +0 -291
  376. package/dist/core/agent-template-engine.js.map +0 -1
  377. package/dist/core/analysis-manager.d.ts +0 -56
  378. package/dist/core/analysis-manager.d.ts.map +0 -1
  379. package/dist/core/analysis-manager.js +0 -218
  380. package/dist/core/analysis-manager.js.map +0 -1
  381. package/dist/core/auto-fixer.d.ts +0 -14
  382. package/dist/core/auto-fixer.d.ts.map +0 -1
  383. package/dist/core/auto-fixer.js +0 -207
  384. package/dist/core/auto-fixer.js.map +0 -1
  385. package/dist/core/changelog-generator.d.ts +0 -44
  386. package/dist/core/changelog-generator.d.ts.map +0 -1
  387. package/dist/core/changelog-generator.js +0 -222
  388. package/dist/core/changelog-generator.js.map +0 -1
  389. package/dist/core/claude-mcp.d.ts +0 -59
  390. package/dist/core/claude-mcp.d.ts.map +0 -1
  391. package/dist/core/claude-mcp.js +0 -220
  392. package/dist/core/claude-mcp.js.map +0 -1
  393. package/dist/core/claude-md-generator.d.ts +0 -52
  394. package/dist/core/claude-md-generator.d.ts.map +0 -1
  395. package/dist/core/claude-md-generator.js +0 -104
  396. package/dist/core/claude-md-generator.js.map +0 -1
  397. package/dist/core/claude-settings-manager.d.ts +0 -44
  398. package/dist/core/claude-settings-manager.d.ts.map +0 -1
  399. package/dist/core/claude-settings-manager.js +0 -191
  400. package/dist/core/claude-settings-manager.js.map +0 -1
  401. package/dist/core/cli-bridge.d.ts +0 -113
  402. package/dist/core/cli-bridge.d.ts.map +0 -1
  403. package/dist/core/cli-bridge.js +0 -1094
  404. package/dist/core/cli-bridge.js.map +0 -1
  405. package/dist/core/compact-context-manager.d.ts +0 -34
  406. package/dist/core/compact-context-manager.d.ts.map +0 -1
  407. package/dist/core/compact-context-manager.js +0 -60
  408. package/dist/core/compact-context-manager.js.map +0 -1
  409. package/dist/core/complexity-detector.d.ts +0 -36
  410. package/dist/core/complexity-detector.d.ts.map +0 -1
  411. package/dist/core/complexity-detector.js +0 -334
  412. package/dist/core/complexity-detector.js.map +0 -1
  413. package/dist/core/compress/compressor.d.ts +0 -60
  414. package/dist/core/compress/compressor.d.ts.map +0 -1
  415. package/dist/core/compress/compressor.js +0 -232
  416. package/dist/core/compress/compressor.js.map +0 -1
  417. package/dist/core/compress/discover.d.ts +0 -19
  418. package/dist/core/compress/discover.d.ts.map +0 -1
  419. package/dist/core/compress/discover.js +0 -100
  420. package/dist/core/compress/discover.js.map +0 -1
  421. package/dist/core/compress/validator.d.ts +0 -47
  422. package/dist/core/compress/validator.d.ts.map +0 -1
  423. package/dist/core/compress/validator.js +0 -131
  424. package/dist/core/compress/validator.js.map +0 -1
  425. package/dist/core/config-manager.d.ts +0 -86
  426. package/dist/core/config-manager.d.ts.map +0 -1
  427. package/dist/core/config-manager.js +0 -621
  428. package/dist/core/config-manager.js.map +0 -1
  429. package/dist/core/coverage-checker.d.ts +0 -14
  430. package/dist/core/coverage-checker.d.ts.map +0 -1
  431. package/dist/core/coverage-checker.js +0 -176
  432. package/dist/core/coverage-checker.js.map +0 -1
  433. package/dist/core/cursor-mdc-generator.d.ts +0 -30
  434. package/dist/core/cursor-mdc-generator.d.ts.map +0 -1
  435. package/dist/core/cursor-mdc-generator.js +0 -98
  436. package/dist/core/cursor-mdc-generator.js.map +0 -1
  437. package/dist/core/decision-manager.d.ts +0 -25
  438. package/dist/core/decision-manager.d.ts.map +0 -1
  439. package/dist/core/decision-manager.js +0 -183
  440. package/dist/core/decision-manager.js.map +0 -1
  441. package/dist/core/dependency-checker.d.ts +0 -21
  442. package/dist/core/dependency-checker.d.ts.map +0 -1
  443. package/dist/core/dependency-checker.js +0 -247
  444. package/dist/core/dependency-checker.js.map +0 -1
  445. package/dist/core/detector.d.ts +0 -27
  446. package/dist/core/detector.d.ts.map +0 -1
  447. package/dist/core/detector.js +0 -1763
  448. package/dist/core/detector.js.map +0 -1
  449. package/dist/core/docs-generator.d.ts +0 -9
  450. package/dist/core/docs-generator.d.ts.map +0 -1
  451. package/dist/core/docs-generator.js +0 -531
  452. package/dist/core/docs-generator.js.map +0 -1
  453. package/dist/core/doctor.d.ts +0 -19
  454. package/dist/core/doctor.d.ts.map +0 -1
  455. package/dist/core/doctor.js +0 -229
  456. package/dist/core/doctor.js.map +0 -1
  457. package/dist/core/generator.d.ts +0 -56
  458. package/dist/core/generator.d.ts.map +0 -1
  459. package/dist/core/generator.js +0 -1193
  460. package/dist/core/generator.js.map +0 -1
  461. package/dist/core/github-issues-importer.d.ts +0 -82
  462. package/dist/core/github-issues-importer.d.ts.map +0 -1
  463. package/dist/core/github-issues-importer.js +0 -161
  464. package/dist/core/github-issues-importer.js.map +0 -1
  465. package/dist/core/gitignore-generator.d.ts +0 -13
  466. package/dist/core/gitignore-generator.d.ts.map +0 -1
  467. package/dist/core/gitignore-generator.js +0 -307
  468. package/dist/core/gitignore-generator.js.map +0 -1
  469. package/dist/core/health-scorer.d.ts +0 -61
  470. package/dist/core/health-scorer.d.ts.map +0 -1
  471. package/dist/core/health-scorer.js +0 -638
  472. package/dist/core/health-scorer.js.map +0 -1
  473. package/dist/core/iteration-tracker.d.ts +0 -85
  474. package/dist/core/iteration-tracker.d.ts.map +0 -1
  475. package/dist/core/iteration-tracker.js +0 -295
  476. package/dist/core/iteration-tracker.js.map +0 -1
  477. package/dist/core/knowledge-manager.d.ts +0 -24
  478. package/dist/core/knowledge-manager.d.ts.map +0 -1
  479. package/dist/core/knowledge-manager.js +0 -173
  480. package/dist/core/knowledge-manager.js.map +0 -1
  481. package/dist/core/learn-manager.d.ts +0 -29
  482. package/dist/core/learn-manager.d.ts.map +0 -1
  483. package/dist/core/learn-manager.js +0 -159
  484. package/dist/core/learn-manager.js.map +0 -1
  485. package/dist/core/mcp-reference-generator.d.ts +0 -13
  486. package/dist/core/mcp-reference-generator.d.ts.map +0 -1
  487. package/dist/core/mcp-reference-generator.js +0 -66
  488. package/dist/core/mcp-reference-generator.js.map +0 -1
  489. package/dist/core/minimal-scaffolder.d.ts +0 -8
  490. package/dist/core/minimal-scaffolder.d.ts.map +0 -1
  491. package/dist/core/minimal-scaffolder.js +0 -51
  492. package/dist/core/minimal-scaffolder.js.map +0 -1
  493. package/dist/core/modern-console.d.ts +0 -98
  494. package/dist/core/modern-console.d.ts.map +0 -1
  495. package/dist/core/modern-console.js +0 -556
  496. package/dist/core/modern-console.js.map +0 -1
  497. package/dist/core/multi-tool-generator.d.ts +0 -59
  498. package/dist/core/multi-tool-generator.d.ts.map +0 -1
  499. package/dist/core/multi-tool-generator.js +0 -157
  500. package/dist/core/multi-tool-generator.js.map +0 -1
  501. package/dist/core/override-manager.d.ts +0 -23
  502. package/dist/core/override-manager.d.ts.map +0 -1
  503. package/dist/core/override-manager.js +0 -82
  504. package/dist/core/override-manager.js.map +0 -1
  505. package/dist/core/plans-manager.d.ts +0 -46
  506. package/dist/core/plans-manager.d.ts.map +0 -1
  507. package/dist/core/plans-manager.js +0 -158
  508. package/dist/core/plans-manager.js.map +0 -1
  509. package/dist/core/prd-generator.d.ts +0 -48
  510. package/dist/core/prd-generator.d.ts.map +0 -1
  511. package/dist/core/prd-generator.js +0 -233
  512. package/dist/core/prd-generator.js.map +0 -1
  513. package/dist/core/ralph-manager.d.ts +0 -163
  514. package/dist/core/ralph-manager.d.ts.map +0 -1
  515. package/dist/core/ralph-manager.js +0 -555
  516. package/dist/core/ralph-manager.js.map +0 -1
  517. package/dist/core/ralph-parallel.d.ts +0 -55
  518. package/dist/core/ralph-parallel.d.ts.map +0 -1
  519. package/dist/core/ralph-parallel.js +0 -201
  520. package/dist/core/ralph-parallel.js.map +0 -1
  521. package/dist/core/ralph-plan-checkpoint.d.ts +0 -58
  522. package/dist/core/ralph-plan-checkpoint.d.ts.map +0 -1
  523. package/dist/core/ralph-plan-checkpoint.js +0 -154
  524. package/dist/core/ralph-plan-checkpoint.js.map +0 -1
  525. package/dist/core/ralph-scripts.d.ts +0 -12
  526. package/dist/core/ralph-scripts.d.ts.map +0 -1
  527. package/dist/core/ralph-scripts.js +0 -50
  528. package/dist/core/ralph-scripts.js.map +0 -1
  529. package/dist/core/review-manager.d.ts +0 -74
  530. package/dist/core/review-manager.d.ts.map +0 -1
  531. package/dist/core/review-manager.js +0 -371
  532. package/dist/core/review-manager.js.map +0 -1
  533. package/dist/core/rules-generator.d.ts +0 -73
  534. package/dist/core/rules-generator.d.ts.map +0 -1
  535. package/dist/core/rules-generator.js +0 -202
  536. package/dist/core/rules-generator.js.map +0 -1
  537. package/dist/core/skills-manager.d.ts +0 -126
  538. package/dist/core/skills-manager.d.ts.map +0 -1
  539. package/dist/core/skills-manager.js +0 -654
  540. package/dist/core/skills-manager.js.map +0 -1
  541. package/dist/core/state-writer.d.ts +0 -35
  542. package/dist/core/state-writer.d.ts.map +0 -1
  543. package/dist/core/state-writer.js +0 -81
  544. package/dist/core/state-writer.js.map +0 -1
  545. package/dist/core/task-manager.d.ts +0 -127
  546. package/dist/core/task-manager.d.ts.map +0 -1
  547. package/dist/core/task-manager.js +0 -607
  548. package/dist/core/task-manager.js.map +0 -1
  549. package/dist/core/telemetry.d.ts +0 -29
  550. package/dist/core/telemetry.d.ts.map +0 -1
  551. package/dist/core/telemetry.js +0 -57
  552. package/dist/core/telemetry.js.map +0 -1
  553. package/dist/core/validator.d.ts +0 -21
  554. package/dist/core/validator.d.ts.map +0 -1
  555. package/dist/core/validator.js +0 -177
  556. package/dist/core/validator.js.map +0 -1
  557. package/dist/core/version-bumper.d.ts +0 -19
  558. package/dist/core/version-bumper.d.ts.map +0 -1
  559. package/dist/core/version-bumper.js +0 -180
  560. package/dist/core/version-bumper.js.map +0 -1
  561. package/dist/core/watcher.d.ts +0 -9
  562. package/dist/core/watcher.d.ts.map +0 -1
  563. package/dist/core/watcher.js +0 -22
  564. package/dist/core/watcher.js.map +0 -1
  565. package/dist/core/workflow-generator.d.ts +0 -15
  566. package/dist/core/workflow-generator.d.ts.map +0 -1
  567. package/dist/core/workflow-generator.js +0 -391
  568. package/dist/core/workflow-generator.js.map +0 -1
  569. package/dist/hooks/terse-activate.d.ts +0 -59
  570. package/dist/hooks/terse-activate.d.ts.map +0 -1
  571. package/dist/hooks/terse-activate.js +0 -149
  572. package/dist/hooks/terse-activate.js.map +0 -1
  573. package/dist/hooks/terse-config.d.ts +0 -51
  574. package/dist/hooks/terse-config.d.ts.map +0 -1
  575. package/dist/hooks/terse-config.js +0 -130
  576. package/dist/hooks/terse-config.js.map +0 -1
  577. package/dist/hooks/terse-mode-tracker.d.ts +0 -78
  578. package/dist/hooks/terse-mode-tracker.d.ts.map +0 -1
  579. package/dist/hooks/terse-mode-tracker.js +0 -213
  580. package/dist/hooks/terse-mode-tracker.js.map +0 -1
  581. package/dist/memory/hnsw-index.d.ts +0 -68
  582. package/dist/memory/hnsw-index.d.ts.map +0 -1
  583. package/dist/memory/hnsw-index.js +0 -544
  584. package/dist/memory/hnsw-index.js.map +0 -1
  585. package/dist/memory/memory-cache.d.ts +0 -33
  586. package/dist/memory/memory-cache.d.ts.map +0 -1
  587. package/dist/memory/memory-cache.js +0 -85
  588. package/dist/memory/memory-cache.js.map +0 -1
  589. package/dist/memory/memory-search.d.ts +0 -42
  590. package/dist/memory/memory-search.d.ts.map +0 -1
  591. package/dist/memory/memory-search.js +0 -180
  592. package/dist/memory/memory-search.js.map +0 -1
  593. package/dist/memory/memory-store.d.ts +0 -84
  594. package/dist/memory/memory-store.d.ts.map +0 -1
  595. package/dist/memory/memory-store.js +0 -566
  596. package/dist/memory/memory-store.js.map +0 -1
  597. package/dist/memory/memory-vectorizer.d.ts +0 -29
  598. package/dist/memory/memory-vectorizer.d.ts.map +0 -1
  599. package/dist/memory/memory-vectorizer.js +0 -199
  600. package/dist/memory/memory-vectorizer.js.map +0 -1
@@ -1,1763 +0,0 @@
1
- import path from 'path';
2
- import { existsSync } from 'fs';
3
- import { readdir } from 'fs/promises';
4
- import { fileExists, findFiles, readFile, readJsonFile } from '../utils/file-system.js';
5
- export async function detectProject(cwd = process.cwd()) {
6
- const languages = await detectLanguages(cwd);
7
- const modules = await detectModules(cwd);
8
- const frameworks = await detectFrameworks(cwd, languages);
9
- const services = await detectServices(cwd);
10
- const existingAgents = await detectExistingAgents(cwd);
11
- const gitHooks = await detectGitHooks(cwd);
12
- const monorepo = await detectMonorepo(cwd);
13
- const cursor = await detectCursor(cwd);
14
- const geminiCli = await detectGeminiCli(cwd);
15
- const continueDev = await detectContinueDev(cwd);
16
- const windsurf = await detectWindsurf(cwd);
17
- const githubCopilot = await detectGithubCopilot(cwd);
18
- return {
19
- languages,
20
- modules,
21
- frameworks,
22
- services,
23
- existingAgents,
24
- gitHooks,
25
- monorepo,
26
- cursor,
27
- geminiCli,
28
- continueDev,
29
- windsurf,
30
- githubCopilot,
31
- };
32
- }
33
- /**
34
- * Detect Cursor IDE presence and configuration status.
35
- */
36
- export async function detectCursor(cwd) {
37
- const cursorDir = path.join(cwd, '.cursor');
38
- const cursorrules = path.join(cwd, '.cursorrules');
39
- const rulesDir = path.join(cursorDir, 'rules');
40
- const hasCursorDir = existsSync(cursorDir);
41
- const hasCursorrules = existsSync(cursorrules);
42
- const detected = hasCursorDir || hasCursorrules;
43
- let hasMdcRules = false;
44
- if (existsSync(rulesDir)) {
45
- try {
46
- const files = await readdir(rulesDir);
47
- hasMdcRules = files.some((f) => f.endsWith('.mdc'));
48
- }
49
- catch {
50
- // ignore
51
- }
52
- }
53
- return { detected, hasCursorrules, hasMdcRules };
54
- }
55
- /**
56
- * Detect Gemini CLI presence by checking for GEMINI.md in the project root.
57
- */
58
- export async function detectGeminiCli(cwd) {
59
- const geminiMd = path.join(cwd, 'GEMINI.md');
60
- const detected = existsSync(geminiMd);
61
- return { detected };
62
- }
63
- /**
64
- * Detect Continue.dev IDE extension by checking for the .continue/ directory.
65
- */
66
- export async function detectContinueDev(cwd) {
67
- const continueDir = path.join(cwd, '.continue');
68
- const rulesDir = path.join(continueDir, 'rules');
69
- const detected = existsSync(continueDir);
70
- return { detected, rulesDir };
71
- }
72
- /**
73
- * Detect Windsurf IDE by checking for .windsurfrules in the project root.
74
- */
75
- export async function detectWindsurf(cwd) {
76
- const windsurfrules = path.join(cwd, '.windsurfrules');
77
- const detected = existsSync(windsurfrules);
78
- return { detected };
79
- }
80
- /**
81
- * Detect GitHub Copilot by checking for .github/copilot-instructions.md.
82
- */
83
- export async function detectGithubCopilot(cwd) {
84
- const copilotInstructions = path.join(cwd, '.github', 'copilot-instructions.md');
85
- const detected = existsSync(copilotInstructions);
86
- return { detected };
87
- }
88
- /**
89
- * Detect monorepo structure: Turborepo, Nx, pnpm workspaces, Lerna, or manual.
90
- */
91
- export async function detectMonorepo(cwd) {
92
- // Turborepo
93
- if (existsSync(path.join(cwd, 'turbo.json'))) {
94
- const packages = await discoverPackages(cwd);
95
- return { detected: true, tool: 'turborepo', packages };
96
- }
97
- // Nx
98
- if (existsSync(path.join(cwd, 'nx.json'))) {
99
- const packages = await discoverPackages(cwd);
100
- return { detected: true, tool: 'nx', packages };
101
- }
102
- // pnpm workspaces
103
- if (existsSync(path.join(cwd, 'pnpm-workspace.yaml'))) {
104
- const packages = await discoverPackages(cwd);
105
- return { detected: true, tool: 'pnpm', packages };
106
- }
107
- // Lerna
108
- if (existsSync(path.join(cwd, 'lerna.json'))) {
109
- const packages = await discoverPackages(cwd);
110
- return { detected: true, tool: 'lerna', packages };
111
- }
112
- // Manual monorepo — packages/ or apps/ directory with multiple package.json files
113
- const packages = await discoverPackages(cwd);
114
- if (packages.length >= 2) {
115
- return { detected: true, tool: 'manual', packages };
116
- }
117
- return { detected: false, tool: null, packages: [] };
118
- }
119
- /**
120
- * Discover package directories by looking for package.json in packages/ and apps/.
121
- */
122
- async function discoverPackages(cwd) {
123
- const packageDirs = [];
124
- const searchDirs = ['packages', 'apps', 'libs', 'services'];
125
- for (const searchDir of searchDirs) {
126
- const absDir = path.join(cwd, searchDir);
127
- if (!existsSync(absDir))
128
- continue;
129
- try {
130
- const entries = await readdir(absDir, { withFileTypes: true });
131
- for (const entry of entries) {
132
- if (!entry.isDirectory())
133
- continue;
134
- const pkgJson = path.join(absDir, entry.name, 'package.json');
135
- if (existsSync(pkgJson)) {
136
- packageDirs.push(`${searchDir}/${entry.name}`);
137
- }
138
- }
139
- }
140
- catch {
141
- // ignore unreadable dirs
142
- }
143
- }
144
- return packageDirs;
145
- }
146
- async function detectLanguages(cwd) {
147
- const detections = [];
148
- // Detect Rust
149
- const cargoToml = path.join(cwd, 'Cargo.toml');
150
- if (await fileExists(cargoToml)) {
151
- const rsFiles = await findFiles('**/*.rs', cwd);
152
- detections.push({
153
- language: 'rust',
154
- confidence: rsFiles.length > 0 ? 1.0 : 0.8,
155
- indicators: ['Cargo.toml', `${rsFiles.length} .rs files`],
156
- });
157
- }
158
- // Detect TypeScript
159
- const packageJson = path.join(cwd, 'package.json');
160
- const tsConfig = path.join(cwd, 'tsconfig.json');
161
- if ((await fileExists(packageJson)) || (await fileExists(tsConfig))) {
162
- const tsFiles = await findFiles('**/*.ts', cwd);
163
- detections.push({
164
- language: 'typescript',
165
- confidence: tsFiles.length > 0 ? 1.0 : 0.7,
166
- indicators: [
167
- (await fileExists(packageJson)) ? 'package.json' : '',
168
- (await fileExists(tsConfig)) ? 'tsconfig.json' : '',
169
- `${tsFiles.length} .ts files`,
170
- ].filter(Boolean),
171
- });
172
- }
173
- // Detect Python
174
- const pyprojectToml = path.join(cwd, 'pyproject.toml');
175
- const requirementsTxt = path.join(cwd, 'requirements.txt');
176
- const setupPy = path.join(cwd, 'setup.py');
177
- if ((await fileExists(pyprojectToml)) ||
178
- (await fileExists(requirementsTxt)) ||
179
- (await fileExists(setupPy))) {
180
- const pyFiles = await findFiles('**/*.py', cwd);
181
- detections.push({
182
- language: 'python',
183
- confidence: pyFiles.length > 0 ? 1.0 : 0.7,
184
- indicators: [
185
- (await fileExists(pyprojectToml)) ? 'pyproject.toml' : '',
186
- (await fileExists(requirementsTxt)) ? 'requirements.txt' : '',
187
- (await fileExists(setupPy)) ? 'setup.py' : '',
188
- `${pyFiles.length} .py files`,
189
- ].filter(Boolean),
190
- });
191
- }
192
- // Detect Go
193
- const goMod = path.join(cwd, 'go.mod');
194
- if (await fileExists(goMod)) {
195
- const goFiles = await findFiles('**/*.go', cwd);
196
- detections.push({
197
- language: 'go',
198
- confidence: goFiles.length > 0 ? 1.0 : 0.8,
199
- indicators: ['go.mod', `${goFiles.length} .go files`],
200
- });
201
- }
202
- // Detect Java
203
- const pomXml = path.join(cwd, 'pom.xml');
204
- const buildGradle = path.join(cwd, 'build.gradle');
205
- const buildGradleKts = path.join(cwd, 'build.gradle.kts');
206
- if ((await fileExists(pomXml)) ||
207
- (await fileExists(buildGradle)) ||
208
- (await fileExists(buildGradleKts))) {
209
- const javaFiles = await findFiles('**/*.java', cwd);
210
- detections.push({
211
- language: 'java',
212
- confidence: javaFiles.length > 0 ? 1.0 : 0.7,
213
- indicators: [
214
- (await fileExists(pomXml)) ? 'pom.xml' : '',
215
- (await fileExists(buildGradle)) ? 'build.gradle' : '',
216
- (await fileExists(buildGradleKts)) ? 'build.gradle.kts' : '',
217
- `${javaFiles.length} .java files`,
218
- ].filter(Boolean),
219
- });
220
- }
221
- // Detect C/C++
222
- const cmakeLists = path.join(cwd, 'CMakeLists.txt');
223
- const makeFile = path.join(cwd, 'Makefile');
224
- if ((await fileExists(cmakeLists)) || (await fileExists(makeFile))) {
225
- const cppFiles = await findFiles('**/*.{cpp,hpp,cc,h,c}', cwd);
226
- detections.push({
227
- language: 'cpp',
228
- confidence: cppFiles.length > 0 ? 1.0 : 0.8,
229
- indicators: [
230
- (await fileExists(cmakeLists)) ? 'CMakeLists.txt' : '',
231
- (await fileExists(makeFile)) ? 'Makefile' : '',
232
- `${cppFiles.length} C/C++ files`,
233
- ].filter(Boolean),
234
- });
235
- }
236
- // Detect Solidity
237
- const hardhatConfig = path.join(cwd, 'hardhat.config.js');
238
- const foundryToml = path.join(cwd, 'foundry.toml');
239
- if ((await fileExists(hardhatConfig)) || (await fileExists(foundryToml))) {
240
- const solFiles = await findFiles('**/*.sol', cwd);
241
- detections.push({
242
- language: 'solidity',
243
- confidence: solFiles.length > 0 ? 1.0 : 0.8,
244
- indicators: [
245
- (await fileExists(hardhatConfig)) ? 'hardhat.config.js' : '',
246
- (await fileExists(foundryToml)) ? 'foundry.toml' : '',
247
- `${solFiles.length} .sol files`,
248
- ].filter(Boolean),
249
- });
250
- }
251
- // Detect Zig
252
- const buildZig = path.join(cwd, 'build.zig');
253
- if (await fileExists(buildZig)) {
254
- const zigFiles = await findFiles('**/*.zig', cwd);
255
- detections.push({
256
- language: 'zig',
257
- confidence: zigFiles.length > 0 ? 1.0 : 0.9,
258
- indicators: ['build.zig', `${zigFiles.length} .zig files`],
259
- });
260
- }
261
- // Detect Erlang
262
- const rebarConfig = path.join(cwd, 'rebar.config');
263
- if (await fileExists(rebarConfig)) {
264
- const erlFiles = await findFiles('**/*.erl', cwd);
265
- detections.push({
266
- language: 'erlang',
267
- confidence: erlFiles.length > 0 ? 1.0 : 0.8,
268
- indicators: ['rebar.config', `${erlFiles.length} .erl files`],
269
- });
270
- }
271
- // Detect JavaScript (pure, not TypeScript)
272
- if (await fileExists(packageJson)) {
273
- const jsFiles = await findFiles('**/*.js', cwd);
274
- const hasTS = detections.some((d) => d.language === 'typescript');
275
- if (!hasTS && jsFiles.length > 0) {
276
- const pkg = await readJsonFile(packageJson);
277
- detections.push({
278
- language: 'javascript',
279
- confidence: 0.9,
280
- indicators: [
281
- 'package.json',
282
- `${jsFiles.length} .js files`,
283
- pkg?.type === 'module' ? 'ESM' : '',
284
- ].filter(Boolean),
285
- });
286
- }
287
- }
288
- // Detect Dart
289
- const pubspecYaml = path.join(cwd, 'pubspec.yaml');
290
- if (await fileExists(pubspecYaml)) {
291
- const dartFiles = await findFiles('**/*.dart', cwd);
292
- detections.push({
293
- language: 'dart',
294
- confidence: dartFiles.length > 0 ? 1.0 : 0.8,
295
- indicators: ['pubspec.yaml', `${dartFiles.length} .dart files`],
296
- });
297
- }
298
- // Detect Ruby
299
- const gemfile = path.join(cwd, 'Gemfile');
300
- const gemspec = await findFiles('**/*.gemspec', cwd);
301
- if ((await fileExists(gemfile)) || gemspec.length > 0) {
302
- const rbFiles = await findFiles('**/*.rb', cwd);
303
- detections.push({
304
- language: 'ruby',
305
- confidence: rbFiles.length > 0 ? 1.0 : 0.7,
306
- indicators: [
307
- (await fileExists(gemfile)) ? 'Gemfile' : '',
308
- gemspec.length > 0 ? `${gemspec.length} .gemspec` : '',
309
- `${rbFiles.length} .rb files`,
310
- ].filter(Boolean),
311
- });
312
- }
313
- // Detect Scala
314
- const buildSbt = path.join(cwd, 'build.sbt');
315
- if (await fileExists(buildSbt)) {
316
- const scalaFiles = await findFiles('**/*.scala', cwd);
317
- detections.push({
318
- language: 'scala',
319
- confidence: scalaFiles.length > 0 ? 1.0 : 0.8,
320
- indicators: ['build.sbt', `${scalaFiles.length} .scala files`],
321
- });
322
- }
323
- // Detect R
324
- const descriptionFile = path.join(cwd, 'DESCRIPTION');
325
- if (await fileExists(descriptionFile)) {
326
- const rFiles = await findFiles('**/*.R', cwd);
327
- detections.push({
328
- language: 'r',
329
- confidence: rFiles.length > 0 ? 1.0 : 0.8,
330
- indicators: ['DESCRIPTION', `${rFiles.length} .R files`],
331
- });
332
- }
333
- // Detect Haskell
334
- const stackYaml = path.join(cwd, 'stack.yaml');
335
- const cabalFiles = await findFiles('**/*.cabal', cwd);
336
- if ((await fileExists(stackYaml)) || cabalFiles.length > 0) {
337
- const hsFiles = await findFiles('**/*.hs', cwd);
338
- detections.push({
339
- language: 'haskell',
340
- confidence: hsFiles.length > 0 ? 1.0 : 0.8,
341
- indicators: [
342
- (await fileExists(stackYaml)) ? 'stack.yaml' : '',
343
- cabalFiles.length > 0 ? `${cabalFiles.length} .cabal` : '',
344
- `${hsFiles.length} .hs files`,
345
- ].filter(Boolean),
346
- });
347
- }
348
- // Detect Julia
349
- const projectToml = path.join(cwd, 'Project.toml');
350
- if (await fileExists(projectToml)) {
351
- const jlFiles = await findFiles('**/*.jl', cwd);
352
- detections.push({
353
- language: 'julia',
354
- confidence: jlFiles.length > 0 ? 1.0 : 0.8,
355
- indicators: ['Project.toml', `${jlFiles.length} .jl files`],
356
- });
357
- }
358
- // Detect Lua
359
- const luaFiles = await findFiles('**/*.lua', cwd);
360
- if (luaFiles.length > 5) {
361
- detections.push({
362
- language: 'lua',
363
- confidence: 0.9,
364
- indicators: [`${luaFiles.length} .lua files`],
365
- });
366
- }
367
- // Sort by confidence
368
- return detections.sort((a, b) => b.confidence - a.confidence);
369
- }
370
- async function detectFrameworks(cwd, languages) {
371
- const packageJsonPath = path.join(cwd, 'package.json');
372
- let packageJson = null;
373
- if (await fileExists(packageJsonPath)) {
374
- try {
375
- packageJson = await readJsonFile(packageJsonPath);
376
- }
377
- catch {
378
- packageJson = null;
379
- }
380
- }
381
- const composerJsonPath = path.join(cwd, 'composer.json');
382
- let composerJson = null;
383
- if (await fileExists(composerJsonPath)) {
384
- try {
385
- composerJson = await readJsonFile(composerJsonPath);
386
- }
387
- catch {
388
- composerJson = null;
389
- }
390
- }
391
- const languageSet = new Set(languages.map((l) => l.language));
392
- const npmDeps = {
393
- ...(packageJson && typeof packageJson === 'object'
394
- ? (packageJson.dependencies ?? {})
395
- : {}),
396
- ...(packageJson && typeof packageJson === 'object'
397
- ? (packageJson.devDependencies ?? {})
398
- : {}),
399
- };
400
- const phpDeps = {
401
- ...(composerJson && typeof composerJson === 'object'
402
- ? (composerJson.require ?? {})
403
- : {}),
404
- ...(composerJson && typeof composerJson === 'object'
405
- ? (composerJson['require-dev'] ?? {})
406
- : {}),
407
- };
408
- const frameworkDefinitions = [
409
- {
410
- id: 'nestjs',
411
- label: 'NestJS',
412
- languages: ['typescript', 'javascript'],
413
- detect: async () => {
414
- if (npmDeps['@nestjs/core'] || npmDeps['@nestjs/common']) {
415
- return {
416
- detected: true,
417
- confidence: 0.95,
418
- indicators: ['package.json:@nestjs/core'],
419
- };
420
- }
421
- const nestCli = path.join(cwd, 'nest-cli.json');
422
- if (await fileExists(nestCli)) {
423
- return {
424
- detected: true,
425
- confidence: 0.9,
426
- indicators: ['nest-cli.json'],
427
- };
428
- }
429
- return { detected: false, confidence: 0, indicators: [] };
430
- },
431
- },
432
- {
433
- id: 'spring',
434
- label: 'Spring Boot',
435
- languages: ['java', 'kotlin'],
436
- detect: async () => {
437
- const pomPath = path.join(cwd, 'pom.xml');
438
- if (await fileExists(pomPath)) {
439
- const content = await readFile(pomPath);
440
- if (content.includes('spring-boot-starter')) {
441
- return {
442
- detected: true,
443
- confidence: 0.95,
444
- indicators: ['pom.xml:spring-boot-starter'],
445
- };
446
- }
447
- }
448
- const gradlePath = await findFirstExisting([
449
- path.join(cwd, 'build.gradle'),
450
- path.join(cwd, 'build.gradle.kts'),
451
- ]);
452
- if (gradlePath) {
453
- const content = await readFile(gradlePath);
454
- if (content.includes('spring-boot-starter')) {
455
- return {
456
- detected: true,
457
- confidence: 0.9,
458
- indicators: [`${path.basename(gradlePath)}:spring-boot-starter`],
459
- };
460
- }
461
- }
462
- return { detected: false, confidence: 0, indicators: [] };
463
- },
464
- },
465
- {
466
- id: 'laravel',
467
- label: 'Laravel',
468
- languages: ['php'],
469
- detect: async () => {
470
- if (phpDeps['laravel/framework']) {
471
- return {
472
- detected: true,
473
- confidence: 0.95,
474
- indicators: ['composer.json:laravel/framework'],
475
- };
476
- }
477
- const artisanPath = path.join(cwd, 'artisan');
478
- if (await fileExists(artisanPath)) {
479
- return {
480
- detected: true,
481
- confidence: 0.9,
482
- indicators: ['artisan'],
483
- };
484
- }
485
- return { detected: false, confidence: 0, indicators: [] };
486
- },
487
- },
488
- {
489
- id: 'angular',
490
- label: 'Angular',
491
- languages: ['typescript', 'javascript'],
492
- detect: async () => {
493
- if (npmDeps['@angular/core']) {
494
- return {
495
- detected: true,
496
- confidence: 0.95,
497
- indicators: ['package.json:@angular/core'],
498
- };
499
- }
500
- const angularConfig = path.join(cwd, 'angular.json');
501
- if (await fileExists(angularConfig)) {
502
- return {
503
- detected: true,
504
- confidence: 0.9,
505
- indicators: ['angular.json'],
506
- };
507
- }
508
- return { detected: false, confidence: 0, indicators: [] };
509
- },
510
- },
511
- {
512
- id: 'react',
513
- label: 'React',
514
- languages: ['typescript', 'javascript'],
515
- detect: async () => {
516
- if (npmDeps.react && (npmDeps['react-dom'] || npmDeps['react-native'])) {
517
- const indicator = npmDeps['react-dom'] ? 'react-dom' : 'react-native';
518
- return {
519
- detected: true,
520
- confidence: 0.9,
521
- indicators: [`package.json:react`, `package.json:${indicator}`],
522
- };
523
- }
524
- return { detected: false, confidence: 0, indicators: [] };
525
- },
526
- },
527
- {
528
- id: 'vue',
529
- label: 'Vue.js',
530
- languages: ['typescript', 'javascript'],
531
- detect: async () => {
532
- if (npmDeps.vue) {
533
- return {
534
- detected: true,
535
- confidence: 0.9,
536
- indicators: ['package.json:vue'],
537
- };
538
- }
539
- const vueConfig = await findFirstExisting([
540
- path.join(cwd, 'vite.config.ts'),
541
- path.join(cwd, 'vite.config.js'),
542
- ]);
543
- if (vueConfig) {
544
- const content = await readFile(vueConfig);
545
- if (content.includes('@vitejs/plugin-vue') || content.includes('vue()')) {
546
- return {
547
- detected: true,
548
- confidence: 0.8,
549
- indicators: [path.basename(vueConfig)],
550
- };
551
- }
552
- }
553
- return { detected: false, confidence: 0, indicators: [] };
554
- },
555
- },
556
- {
557
- id: 'nuxt',
558
- label: 'Nuxt',
559
- languages: ['typescript', 'javascript'],
560
- detect: async () => {
561
- if (npmDeps.nuxt) {
562
- return {
563
- detected: true,
564
- confidence: 0.9,
565
- indicators: ['package.json:nuxt'],
566
- };
567
- }
568
- const nuxtConfig = await findFiles('nuxt.config.*', cwd);
569
- if (nuxtConfig.length > 0) {
570
- return {
571
- detected: true,
572
- confidence: 0.85,
573
- indicators: [path.basename(nuxtConfig[0])],
574
- };
575
- }
576
- return { detected: false, confidence: 0, indicators: [] };
577
- },
578
- },
579
- {
580
- id: 'django',
581
- label: 'Django',
582
- languages: ['python'],
583
- detect: async () => {
584
- const requirementsPath = path.join(cwd, 'requirements.txt');
585
- if (await fileExists(requirementsPath)) {
586
- const content = await readFile(requirementsPath);
587
- if (content.includes('Django')) {
588
- return {
589
- detected: true,
590
- confidence: 0.95,
591
- indicators: ['requirements.txt:Django'],
592
- };
593
- }
594
- }
595
- const managePy = path.join(cwd, 'manage.py');
596
- if (await fileExists(managePy)) {
597
- const content = await readFile(managePy);
598
- if (content.includes('django')) {
599
- return {
600
- detected: true,
601
- confidence: 0.9,
602
- indicators: ['manage.py'],
603
- };
604
- }
605
- }
606
- return { detected: false, confidence: 0, indicators: [] };
607
- },
608
- },
609
- {
610
- id: 'flask',
611
- label: 'Flask',
612
- languages: ['python'],
613
- detect: async () => {
614
- const requirementsPath = path.join(cwd, 'requirements.txt');
615
- if (await fileExists(requirementsPath)) {
616
- const content = await readFile(requirementsPath);
617
- if (content.includes('Flask')) {
618
- return {
619
- detected: true,
620
- confidence: 0.95,
621
- indicators: ['requirements.txt:Flask'],
622
- };
623
- }
624
- }
625
- return { detected: false, confidence: 0, indicators: [] };
626
- },
627
- },
628
- {
629
- id: 'rails',
630
- label: 'Ruby on Rails',
631
- languages: ['ruby'],
632
- detect: async () => {
633
- const gemfilePath = path.join(cwd, 'Gemfile');
634
- if (await fileExists(gemfilePath)) {
635
- const content = await readFile(gemfilePath);
636
- if (content.includes('rails')) {
637
- return {
638
- detected: true,
639
- confidence: 0.95,
640
- indicators: ['Gemfile:rails'],
641
- };
642
- }
643
- }
644
- const railsPath = path.join(cwd, 'bin', 'rails');
645
- if (await fileExists(railsPath)) {
646
- return {
647
- detected: true,
648
- confidence: 0.9,
649
- indicators: ['bin/rails'],
650
- };
651
- }
652
- return { detected: false, confidence: 0, indicators: [] };
653
- },
654
- },
655
- {
656
- id: 'symfony',
657
- label: 'Symfony',
658
- languages: ['php'],
659
- detect: async () => {
660
- if (phpDeps['symfony/framework-bundle']) {
661
- return {
662
- detected: true,
663
- confidence: 0.95,
664
- indicators: ['composer.json:symfony/framework-bundle'],
665
- };
666
- }
667
- const symfonyConfig = path.join(cwd, 'symfony.lock');
668
- if (await fileExists(symfonyConfig)) {
669
- return {
670
- detected: true,
671
- confidence: 0.9,
672
- indicators: ['symfony.lock'],
673
- };
674
- }
675
- return { detected: false, confidence: 0, indicators: [] };
676
- },
677
- },
678
- {
679
- id: 'zend',
680
- label: 'Zend Framework',
681
- languages: ['php'],
682
- detect: async () => {
683
- if (phpDeps['zendframework/zendframework'] || phpDeps['laminas/laminas-mvc']) {
684
- return {
685
- detected: true,
686
- confidence: 0.95,
687
- indicators: ['composer.json:zendframework or laminas'],
688
- };
689
- }
690
- return { detected: false, confidence: 0, indicators: [] };
691
- },
692
- },
693
- {
694
- id: 'jquery',
695
- label: 'jQuery',
696
- languages: ['javascript', 'typescript'],
697
- detect: async () => {
698
- if (npmDeps['jquery']) {
699
- return {
700
- detected: true,
701
- confidence: 0.95,
702
- indicators: ['package.json:jquery'],
703
- };
704
- }
705
- // Check for jQuery in HTML files
706
- const indexHtml = path.join(cwd, 'index.html');
707
- if (await fileExists(indexHtml)) {
708
- const content = await readFile(indexHtml);
709
- if (content.includes('jquery') || content.includes('jQuery')) {
710
- return {
711
- detected: true,
712
- confidence: 0.8,
713
- indicators: ['index.html:jquery'],
714
- };
715
- }
716
- }
717
- return { detected: false, confidence: 0, indicators: [] };
718
- },
719
- },
720
- {
721
- id: 'reactnative',
722
- label: 'React Native',
723
- languages: ['javascript', 'typescript'],
724
- detect: async () => {
725
- if (npmDeps['react-native']) {
726
- return {
727
- detected: true,
728
- confidence: 0.95,
729
- indicators: ['package.json:react-native'],
730
- };
731
- }
732
- const appJson = path.join(cwd, 'app.json');
733
- if (await fileExists(appJson)) {
734
- const content = await readFile(appJson);
735
- if (content.includes('react-native') || content.includes('expo')) {
736
- return {
737
- detected: true,
738
- confidence: 0.9,
739
- indicators: ['app.json'],
740
- };
741
- }
742
- }
743
- return { detected: false, confidence: 0, indicators: [] };
744
- },
745
- },
746
- {
747
- id: 'flutter',
748
- label: 'Flutter',
749
- languages: ['dart'],
750
- detect: async () => {
751
- const pubspecPath = path.join(cwd, 'pubspec.yaml');
752
- if (await fileExists(pubspecPath)) {
753
- const content = await readFile(pubspecPath);
754
- if (content.includes('flutter:')) {
755
- return {
756
- detected: true,
757
- confidence: 0.95,
758
- indicators: ['pubspec.yaml:flutter'],
759
- };
760
- }
761
- }
762
- return { detected: false, confidence: 0, indicators: [] };
763
- },
764
- },
765
- {
766
- id: 'nextjs',
767
- label: 'Next.js',
768
- languages: ['typescript', 'javascript'],
769
- detect: async () => {
770
- if (npmDeps['next']) {
771
- return {
772
- detected: true,
773
- confidence: 0.95,
774
- indicators: ['package.json:next'],
775
- };
776
- }
777
- const nextConfig = await findFirstExisting([
778
- path.join(cwd, 'next.config.js'),
779
- path.join(cwd, 'next.config.mjs'),
780
- path.join(cwd, 'next.config.ts'),
781
- ]);
782
- if (nextConfig) {
783
- return {
784
- detected: true,
785
- confidence: 0.9,
786
- indicators: [path.basename(nextConfig)],
787
- };
788
- }
789
- return { detected: false, confidence: 0, indicators: [] };
790
- },
791
- },
792
- {
793
- id: 'electron',
794
- label: 'Electron',
795
- languages: ['typescript', 'javascript'],
796
- detect: async () => {
797
- if (npmDeps['electron']) {
798
- return {
799
- detected: true,
800
- confidence: 0.95,
801
- indicators: ['package.json:electron'],
802
- };
803
- }
804
- // Check for electron-builder or electron-forge
805
- if (npmDeps['electron-builder'] || npmDeps['@electron-forge/cli']) {
806
- return {
807
- detected: true,
808
- confidence: 0.9,
809
- indicators: ['package.json:electron-builder or electron-forge'],
810
- };
811
- }
812
- return { detected: false, confidence: 0, indicators: [] };
813
- },
814
- },
815
- ];
816
- const detections = [];
817
- for (const definition of frameworkDefinitions) {
818
- const match = await definition.detect();
819
- const availableLanguages = definition.languages.filter((lang) => languageSet.has(lang));
820
- const languagesForFramework = availableLanguages.length > 0 ? availableLanguages : definition.languages;
821
- detections.push({
822
- framework: definition.id,
823
- detected: match.detected,
824
- languages: languagesForFramework,
825
- confidence: match.confidence,
826
- indicators: match.indicators,
827
- });
828
- }
829
- return detections.sort((a, b) => b.confidence - a.confidence);
830
- }
831
- async function detectModules(cwd) {
832
- const modules = [];
833
- // Check for MCP configuration files
834
- const mcpConfigPaths = [
835
- path.join(cwd, 'mcp.json'),
836
- path.join(cwd, 'mcp-config.json'),
837
- path.join(cwd, '.cursor', 'mcp.json'),
838
- ];
839
- for (const mcpPath of mcpConfigPaths) {
840
- if (await fileExists(mcpPath)) {
841
- try {
842
- const config = await readJsonFile(mcpPath);
843
- if (config) {
844
- // Check for Vectorizer
845
- if (config.mcpServers?.vectorizer || config.servers?.vectorizer) {
846
- modules.push({
847
- module: 'vectorizer',
848
- detected: true,
849
- source: mcpPath,
850
- });
851
- }
852
- // Check for Synap
853
- if (config.mcpServers?.synap || config.servers?.synap) {
854
- modules.push({
855
- module: 'synap',
856
- detected: true,
857
- source: mcpPath,
858
- });
859
- }
860
- // Check for Context7
861
- if (config.mcpServers?.context7 || config.servers?.context7) {
862
- modules.push({
863
- module: 'context7',
864
- detected: true,
865
- source: mcpPath,
866
- });
867
- }
868
- // Check for GitHub MCP Server
869
- if (config.mcpServers?.github || config.servers?.github) {
870
- modules.push({
871
- module: 'github',
872
- detected: true,
873
- source: mcpPath,
874
- });
875
- }
876
- // Check for Playwright MCP Server
877
- if (config.mcpServers?.playwright || config.servers?.playwright) {
878
- modules.push({
879
- module: 'playwright',
880
- detected: true,
881
- source: mcpPath,
882
- });
883
- }
884
- // Check for Rulebook MCP Server
885
- if (config.mcpServers?.rulebook || config.servers?.rulebook) {
886
- modules.push({
887
- module: 'rulebook_mcp',
888
- detected: true,
889
- source: mcpPath,
890
- });
891
- }
892
- // Check for Sequential Thinking MCP Server (various key names)
893
- const seqKeys = ['sequential-thinking', 'sequential_thinking', 'sequentialThinking'];
894
- const hasSeqThinking = seqKeys.some((k) => config.mcpServers?.[k] || config.servers?.[k]) ||
895
- Object.entries(config.mcpServers ?? {}).some(([, v]) => typeof v === 'object' &&
896
- v !== null &&
897
- 'args' in v &&
898
- Array.isArray(v.args) &&
899
- v.args.some((a) => typeof a === 'string' && a.includes('sequential-thinking')));
900
- if (hasSeqThinking) {
901
- modules.push({
902
- module: 'sequential_thinking',
903
- detected: true,
904
- source: mcpPath,
905
- });
906
- }
907
- }
908
- }
909
- catch {
910
- // Ignore JSON parse errors
911
- }
912
- }
913
- }
914
- // Add undetected modules
915
- const detectedModules = new Set(modules.map((m) => m.module));
916
- const allModules = [
917
- 'vectorizer',
918
- 'synap',
919
- 'context7',
920
- 'github',
921
- 'playwright',
922
- 'rulebook_mcp',
923
- 'sequential_thinking',
924
- ];
925
- for (const module of allModules) {
926
- if (!detectedModules.has(module)) {
927
- modules.push({
928
- module,
929
- detected: false,
930
- });
931
- }
932
- }
933
- return modules;
934
- }
935
- async function detectExistingAgents(cwd) {
936
- const agentsPath = path.join(cwd, 'AGENTS.md');
937
- if (!(await fileExists(agentsPath))) {
938
- return null;
939
- }
940
- const content = await readFile(agentsPath);
941
- const blocks = parseAgentBlocks(content);
942
- return {
943
- exists: true,
944
- path: agentsPath,
945
- content,
946
- blocks,
947
- };
948
- }
949
- function parseAgentBlocks(content) {
950
- const blocks = [];
951
- const lines = content.split('\n');
952
- let currentBlock = null;
953
- for (let i = 0; i < lines.length; i++) {
954
- const line = lines[i];
955
- const startMatch = line.match(/<!--\s*([A-Z_]+):START\s*-->/);
956
- const endMatch = line.match(/<!--\s*([A-Z_]+):END\s*-->/);
957
- if (startMatch) {
958
- currentBlock = {
959
- name: startMatch[1],
960
- startLine: i,
961
- content: [line],
962
- };
963
- }
964
- else if (endMatch && currentBlock) {
965
- currentBlock.content.push(line);
966
- blocks.push({
967
- name: currentBlock.name,
968
- startLine: currentBlock.startLine,
969
- endLine: i,
970
- content: currentBlock.content.join('\n'),
971
- });
972
- currentBlock = null;
973
- }
974
- else if (currentBlock) {
975
- currentBlock.content.push(line);
976
- }
977
- }
978
- return blocks;
979
- }
980
- async function detectGitHooks(cwd) {
981
- const preCommitPath = path.join(cwd, '.git', 'hooks', 'pre-commit');
982
- const prePushPath = path.join(cwd, '.git', 'hooks', 'pre-push');
983
- return {
984
- preCommitExists: await fileExists(preCommitPath),
985
- prePushExists: await fileExists(prePushPath),
986
- };
987
- }
988
- async function findFirstExisting(pathsToCheck) {
989
- for (const filePath of pathsToCheck) {
990
- if (await fileExists(filePath)) {
991
- return filePath;
992
- }
993
- }
994
- return null;
995
- }
996
- async function detectServices(cwd) {
997
- const services = [];
998
- const packageJson = path.join(cwd, 'package.json');
999
- const envFile = path.join(cwd, '.env');
1000
- const dockerCompose = path.join(cwd, 'docker-compose.yml');
1001
- const dockerComposeYaml = path.join(cwd, 'docker-compose.yaml');
1002
- // Check package.json for database drivers
1003
- if (await fileExists(packageJson)) {
1004
- try {
1005
- const pkg = await readJsonFile(packageJson);
1006
- const allDeps = { ...(pkg?.dependencies || {}), ...(pkg?.devDependencies || {}) };
1007
- // PostgreSQL
1008
- if (allDeps['pg'] || allDeps['postgres'] || allDeps['@prisma/client']) {
1009
- services.push({
1010
- service: 'postgresql',
1011
- detected: true,
1012
- confidence: 0.9,
1013
- indicators: ['pg', 'postgres', '@prisma/client'].filter((dep) => allDeps[dep]),
1014
- source: packageJson,
1015
- });
1016
- }
1017
- // MySQL/MariaDB
1018
- if (allDeps['mysql2'] || allDeps['mysql'] || allDeps['mariadb']) {
1019
- services.push({
1020
- service: allDeps['mariadb'] ? 'mariadb' : 'mysql',
1021
- detected: true,
1022
- confidence: 0.9,
1023
- indicators: ['mysql2', 'mysql', 'mariadb'].filter((dep) => allDeps[dep]),
1024
- source: packageJson,
1025
- });
1026
- }
1027
- // MongoDB
1028
- if (allDeps['mongodb'] || allDeps['mongoose']) {
1029
- services.push({
1030
- service: 'mongodb',
1031
- detected: true,
1032
- confidence: 0.9,
1033
- indicators: ['mongodb', 'mongoose'].filter((dep) => allDeps[dep]),
1034
- source: packageJson,
1035
- });
1036
- }
1037
- // Redis
1038
- if (allDeps['redis'] || allDeps['ioredis'] || allDeps['@redis/client']) {
1039
- services.push({
1040
- service: 'redis',
1041
- detected: true,
1042
- confidence: 0.9,
1043
- indicators: ['redis', 'ioredis', '@redis/client'].filter((dep) => allDeps[dep]),
1044
- source: packageJson,
1045
- });
1046
- }
1047
- // Memcached
1048
- if (allDeps['memcached']) {
1049
- services.push({
1050
- service: 'memcached',
1051
- detected: true,
1052
- confidence: 0.9,
1053
- indicators: ['memcached'],
1054
- source: packageJson,
1055
- });
1056
- }
1057
- // Elasticsearch
1058
- if (allDeps['@elastic/elasticsearch'] || allDeps['elasticsearch']) {
1059
- services.push({
1060
- service: 'elasticsearch',
1061
- detected: true,
1062
- confidence: 0.9,
1063
- indicators: ['@elastic/elasticsearch', 'elasticsearch'].filter((dep) => allDeps[dep]),
1064
- source: packageJson,
1065
- });
1066
- }
1067
- // Neo4j
1068
- if (allDeps['neo4j-driver']) {
1069
- services.push({
1070
- service: 'neo4j',
1071
- detected: true,
1072
- confidence: 0.9,
1073
- indicators: ['neo4j-driver'],
1074
- source: packageJson,
1075
- });
1076
- }
1077
- // InfluxDB
1078
- if (allDeps['@influxdata/influxdb-client']) {
1079
- services.push({
1080
- service: 'influxdb',
1081
- detected: true,
1082
- confidence: 0.9,
1083
- indicators: ['@influxdata/influxdb-client'],
1084
- source: packageJson,
1085
- });
1086
- }
1087
- // RabbitMQ
1088
- if (allDeps['amqplib'] || allDeps['amqp-connection-manager']) {
1089
- services.push({
1090
- service: 'rabbitmq',
1091
- detected: true,
1092
- confidence: 0.9,
1093
- indicators: ['amqplib', 'amqp-connection-manager'].filter((dep) => allDeps[dep]),
1094
- source: packageJson,
1095
- });
1096
- }
1097
- // Kafka
1098
- if (allDeps['kafkajs'] || allDeps['node-rdkafka']) {
1099
- services.push({
1100
- service: 'kafka',
1101
- detected: true,
1102
- confidence: 0.9,
1103
- indicators: ['kafkajs', 'node-rdkafka'].filter((dep) => allDeps[dep]),
1104
- source: packageJson,
1105
- });
1106
- }
1107
- // AWS S3
1108
- if (allDeps['@aws-sdk/client-s3'] || allDeps['aws-sdk']) {
1109
- services.push({
1110
- service: 's3',
1111
- detected: true,
1112
- confidence: 0.8,
1113
- indicators: ['@aws-sdk/client-s3', 'aws-sdk'].filter((dep) => allDeps[dep]),
1114
- source: packageJson,
1115
- });
1116
- }
1117
- // Azure Blob
1118
- if (allDeps['@azure/storage-blob']) {
1119
- services.push({
1120
- service: 'azure_blob',
1121
- detected: true,
1122
- confidence: 0.9,
1123
- indicators: ['@azure/storage-blob'],
1124
- source: packageJson,
1125
- });
1126
- }
1127
- // Google Cloud Storage
1128
- if (allDeps['@google-cloud/storage']) {
1129
- services.push({
1130
- service: 'gcs',
1131
- detected: true,
1132
- confidence: 0.9,
1133
- indicators: ['@google-cloud/storage'],
1134
- source: packageJson,
1135
- });
1136
- }
1137
- // MinIO
1138
- if (allDeps['minio']) {
1139
- services.push({
1140
- service: 'minio',
1141
- detected: true,
1142
- confidence: 0.9,
1143
- indicators: ['minio'],
1144
- source: packageJson,
1145
- });
1146
- }
1147
- // SQLite
1148
- if (allDeps['better-sqlite3'] || allDeps['sqlite3']) {
1149
- services.push({
1150
- service: 'sqlite',
1151
- detected: true,
1152
- confidence: 0.9,
1153
- indicators: ['better-sqlite3', 'sqlite3'].filter((dep) => allDeps[dep]),
1154
- source: packageJson,
1155
- });
1156
- }
1157
- // Cassandra
1158
- if (allDeps['cassandra-driver']) {
1159
- services.push({
1160
- service: 'cassandra',
1161
- detected: true,
1162
- confidence: 0.9,
1163
- indicators: ['cassandra-driver'],
1164
- source: packageJson,
1165
- });
1166
- }
1167
- // DynamoDB
1168
- if (allDeps['@aws-sdk/client-dynamodb'] || allDeps['aws-sdk']) {
1169
- services.push({
1170
- service: 'dynamodb',
1171
- detected: true,
1172
- confidence: 0.8,
1173
- indicators: ['@aws-sdk/client-dynamodb', 'aws-sdk'].filter((dep) => allDeps[dep]),
1174
- source: packageJson,
1175
- });
1176
- }
1177
- // SQL Server
1178
- if (allDeps['mssql'] || allDeps['tedious']) {
1179
- services.push({
1180
- service: 'sqlserver',
1181
- detected: true,
1182
- confidence: 0.9,
1183
- indicators: ['mssql', 'tedious'].filter((dep) => allDeps[dep]),
1184
- source: packageJson,
1185
- });
1186
- }
1187
- // Oracle
1188
- if (allDeps['oracledb']) {
1189
- services.push({
1190
- service: 'oracle',
1191
- detected: true,
1192
- confidence: 0.9,
1193
- indicators: ['oracledb'],
1194
- source: packageJson,
1195
- });
1196
- }
1197
- // Sentry
1198
- if (allDeps['@sentry/node'] ||
1199
- allDeps['@sentry/react'] ||
1200
- allDeps['@sentry/nextjs'] ||
1201
- allDeps['@sentry/browser']) {
1202
- services.push({
1203
- service: 'sentry',
1204
- detected: true,
1205
- confidence: 0.9,
1206
- indicators: ['@sentry/node', '@sentry/react', '@sentry/nextjs', '@sentry/browser'].filter((dep) => allDeps[dep]),
1207
- source: packageJson,
1208
- });
1209
- }
1210
- // OpenTelemetry
1211
- if (allDeps['@opentelemetry/sdk-node'] ||
1212
- allDeps['@opentelemetry/api'] ||
1213
- allDeps['@opentelemetry/auto-instrumentations-node']) {
1214
- services.push({
1215
- service: 'opentelemetry',
1216
- detected: true,
1217
- confidence: 0.9,
1218
- indicators: [
1219
- '@opentelemetry/sdk-node',
1220
- '@opentelemetry/api',
1221
- '@opentelemetry/auto-instrumentations-node',
1222
- ].filter((dep) => allDeps[dep]),
1223
- source: packageJson,
1224
- });
1225
- }
1226
- // Datadog
1227
- if (allDeps['dd-trace'] || allDeps['datadog-lambda-js']) {
1228
- services.push({
1229
- service: 'datadog',
1230
- detected: true,
1231
- confidence: 0.9,
1232
- indicators: ['dd-trace', 'datadog-lambda-js'].filter((dep) => allDeps[dep]),
1233
- source: packageJson,
1234
- });
1235
- }
1236
- // Pino
1237
- if (allDeps['pino'] || allDeps['pino-http']) {
1238
- services.push({
1239
- service: 'pino',
1240
- detected: true,
1241
- confidence: 0.9,
1242
- indicators: ['pino', 'pino-http'].filter((dep) => allDeps[dep]),
1243
- source: packageJson,
1244
- });
1245
- }
1246
- // Winston
1247
- if (allDeps['winston'] || allDeps['winston-transport']) {
1248
- services.push({
1249
- service: 'winston',
1250
- detected: true,
1251
- confidence: 0.9,
1252
- indicators: ['winston', 'winston-transport'].filter((dep) => allDeps[dep]),
1253
- source: packageJson,
1254
- });
1255
- }
1256
- // Prometheus
1257
- if (allDeps['prom-client'] || allDeps['express-prometheus-middleware']) {
1258
- services.push({
1259
- service: 'prometheus',
1260
- detected: true,
1261
- confidence: 0.9,
1262
- indicators: ['prom-client', 'express-prometheus-middleware'].filter((dep) => allDeps[dep]),
1263
- source: packageJson,
1264
- });
1265
- }
1266
- }
1267
- catch {
1268
- // Ignore JSON parse errors
1269
- }
1270
- }
1271
- // Check .env file for service connection strings
1272
- if (await fileExists(envFile)) {
1273
- try {
1274
- const envContent = await readFile(envFile);
1275
- const envLines = envContent.split('\n');
1276
- for (const line of envLines) {
1277
- const upperLine = line.toUpperCase();
1278
- // PostgreSQL
1279
- if (upperLine.includes('POSTGRES') ||
1280
- (upperLine.includes('DATABASE_URL') && upperLine.includes('POSTGRES'))) {
1281
- if (!services.find((s) => s.service === 'postgresql')) {
1282
- services.push({
1283
- service: 'postgresql',
1284
- detected: true,
1285
- confidence: 0.8,
1286
- indicators: ['DATABASE_URL or POSTGRES_* env vars'],
1287
- source: envFile,
1288
- });
1289
- }
1290
- }
1291
- // MySQL
1292
- if (upperLine.includes('MYSQL') && !upperLine.includes('MARIADB')) {
1293
- if (!services.find((s) => s.service === 'mysql')) {
1294
- services.push({
1295
- service: 'mysql',
1296
- detected: true,
1297
- confidence: 0.8,
1298
- indicators: ['MYSQL_* env vars'],
1299
- source: envFile,
1300
- });
1301
- }
1302
- }
1303
- // MariaDB
1304
- if (upperLine.includes('MARIADB')) {
1305
- if (!services.find((s) => s.service === 'mariadb')) {
1306
- services.push({
1307
- service: 'mariadb',
1308
- detected: true,
1309
- confidence: 0.8,
1310
- indicators: ['MARIADB_* env vars'],
1311
- source: envFile,
1312
- });
1313
- }
1314
- }
1315
- // MongoDB
1316
- if (upperLine.includes('MONGODB')) {
1317
- if (!services.find((s) => s.service === 'mongodb')) {
1318
- services.push({
1319
- service: 'mongodb',
1320
- detected: true,
1321
- confidence: 0.8,
1322
- indicators: ['MONGODB_* env vars'],
1323
- source: envFile,
1324
- });
1325
- }
1326
- }
1327
- // Redis
1328
- if (upperLine.includes('REDIS')) {
1329
- if (!services.find((s) => s.service === 'redis')) {
1330
- services.push({
1331
- service: 'redis',
1332
- detected: true,
1333
- confidence: 0.8,
1334
- indicators: ['REDIS_* env vars'],
1335
- source: envFile,
1336
- });
1337
- }
1338
- }
1339
- // Elasticsearch
1340
- if (upperLine.includes('ELASTICSEARCH')) {
1341
- if (!services.find((s) => s.service === 'elasticsearch')) {
1342
- services.push({
1343
- service: 'elasticsearch',
1344
- detected: true,
1345
- confidence: 0.8,
1346
- indicators: ['ELASTICSEARCH_* env vars'],
1347
- source: envFile,
1348
- });
1349
- }
1350
- }
1351
- // RabbitMQ
1352
- if (upperLine.includes('RABBITMQ') || upperLine.includes('AMQP')) {
1353
- if (!services.find((s) => s.service === 'rabbitmq')) {
1354
- services.push({
1355
- service: 'rabbitmq',
1356
- detected: true,
1357
- confidence: 0.8,
1358
- indicators: ['RABBITMQ_* or AMQP_* env vars'],
1359
- source: envFile,
1360
- });
1361
- }
1362
- }
1363
- // Kafka
1364
- if (upperLine.includes('KAFKA')) {
1365
- if (!services.find((s) => s.service === 'kafka')) {
1366
- services.push({
1367
- service: 'kafka',
1368
- detected: true,
1369
- confidence: 0.8,
1370
- indicators: ['KAFKA_* env vars'],
1371
- source: envFile,
1372
- });
1373
- }
1374
- }
1375
- // AWS S3
1376
- if (upperLine.includes('S3_') || (upperLine.includes('AWS_') && upperLine.includes('S3'))) {
1377
- if (!services.find((s) => s.service === 's3')) {
1378
- services.push({
1379
- service: 's3',
1380
- detected: true,
1381
- confidence: 0.7,
1382
- indicators: ['S3_* or AWS_* env vars'],
1383
- source: envFile,
1384
- });
1385
- }
1386
- }
1387
- // Azure Blob
1388
- if (upperLine.includes('AZURE_STORAGE') || upperLine.includes('AZURE_BLOB')) {
1389
- if (!services.find((s) => s.service === 'azure_blob')) {
1390
- services.push({
1391
- service: 'azure_blob',
1392
- detected: true,
1393
- confidence: 0.8,
1394
- indicators: ['AZURE_STORAGE_* or AZURE_BLOB_* env vars'],
1395
- source: envFile,
1396
- });
1397
- }
1398
- }
1399
- // Google Cloud Storage
1400
- if (upperLine.includes('GCS_') || upperLine.includes('GCP_STORAGE')) {
1401
- if (!services.find((s) => s.service === 'gcs')) {
1402
- services.push({
1403
- service: 'gcs',
1404
- detected: true,
1405
- confidence: 0.8,
1406
- indicators: ['GCS_* or GCP_STORAGE_* env vars'],
1407
- source: envFile,
1408
- });
1409
- }
1410
- }
1411
- // MinIO
1412
- if (upperLine.includes('MINIO')) {
1413
- if (!services.find((s) => s.service === 'minio')) {
1414
- services.push({
1415
- service: 'minio',
1416
- detected: true,
1417
- confidence: 0.8,
1418
- indicators: ['MINIO_* env vars'],
1419
- source: envFile,
1420
- });
1421
- }
1422
- }
1423
- // InfluxDB
1424
- if (upperLine.includes('INFLUXDB')) {
1425
- if (!services.find((s) => s.service === 'influxdb')) {
1426
- services.push({
1427
- service: 'influxdb',
1428
- detected: true,
1429
- confidence: 0.8,
1430
- indicators: ['INFLUXDB_* env vars'],
1431
- source: envFile,
1432
- });
1433
- }
1434
- }
1435
- // Neo4j
1436
- if (upperLine.includes('NEO4J')) {
1437
- if (!services.find((s) => s.service === 'neo4j')) {
1438
- services.push({
1439
- service: 'neo4j',
1440
- detected: true,
1441
- confidence: 0.8,
1442
- indicators: ['NEO4J_* env vars'],
1443
- source: envFile,
1444
- });
1445
- }
1446
- }
1447
- // SQL Server
1448
- if (upperLine.includes('SQL_SERVER') || upperLine.includes('MSSQL')) {
1449
- if (!services.find((s) => s.service === 'sqlserver')) {
1450
- services.push({
1451
- service: 'sqlserver',
1452
- detected: true,
1453
- confidence: 0.8,
1454
- indicators: ['SQL_SERVER_* or MSSQL_* env vars'],
1455
- source: envFile,
1456
- });
1457
- }
1458
- }
1459
- // Oracle
1460
- if (upperLine.includes('ORACLE')) {
1461
- if (!services.find((s) => s.service === 'oracle')) {
1462
- services.push({
1463
- service: 'oracle',
1464
- detected: true,
1465
- confidence: 0.8,
1466
- indicators: ['ORACLE_* env vars'],
1467
- source: envFile,
1468
- });
1469
- }
1470
- }
1471
- }
1472
- }
1473
- catch {
1474
- // Ignore file read errors
1475
- }
1476
- }
1477
- // Check docker-compose.yml for services
1478
- const composeFiles = [dockerCompose, dockerComposeYaml];
1479
- for (const composeFile of composeFiles) {
1480
- if (await fileExists(composeFile)) {
1481
- try {
1482
- const composeContent = await readFile(composeFile);
1483
- const upperContent = composeContent.toUpperCase();
1484
- // PostgreSQL
1485
- if (upperContent.includes('POSTGRES') &&
1486
- !services.find((s) => s.service === 'postgresql')) {
1487
- services.push({
1488
- service: 'postgresql',
1489
- detected: true,
1490
- confidence: 0.7,
1491
- indicators: ['docker-compose.yml postgres service'],
1492
- source: composeFile,
1493
- });
1494
- }
1495
- // MySQL
1496
- if (upperContent.includes('MYSQL') &&
1497
- !upperContent.includes('MARIADB') &&
1498
- !services.find((s) => s.service === 'mysql')) {
1499
- services.push({
1500
- service: 'mysql',
1501
- detected: true,
1502
- confidence: 0.7,
1503
- indicators: ['docker-compose.yml mysql service'],
1504
- source: composeFile,
1505
- });
1506
- }
1507
- // MariaDB
1508
- if (upperContent.includes('MARIADB') && !services.find((s) => s.service === 'mariadb')) {
1509
- services.push({
1510
- service: 'mariadb',
1511
- detected: true,
1512
- confidence: 0.7,
1513
- indicators: ['docker-compose.yml mariadb service'],
1514
- source: composeFile,
1515
- });
1516
- }
1517
- // MongoDB
1518
- if (upperContent.includes('MONGO') && !services.find((s) => s.service === 'mongodb')) {
1519
- services.push({
1520
- service: 'mongodb',
1521
- detected: true,
1522
- confidence: 0.7,
1523
- indicators: ['docker-compose.yml mongo service'],
1524
- source: composeFile,
1525
- });
1526
- }
1527
- // Redis
1528
- if (upperContent.includes('REDIS') && !services.find((s) => s.service === 'redis')) {
1529
- services.push({
1530
- service: 'redis',
1531
- detected: true,
1532
- confidence: 0.7,
1533
- indicators: ['docker-compose.yml redis service'],
1534
- source: composeFile,
1535
- });
1536
- }
1537
- // Memcached
1538
- if (upperContent.includes('MEMCACHED') &&
1539
- !services.find((s) => s.service === 'memcached')) {
1540
- services.push({
1541
- service: 'memcached',
1542
- detected: true,
1543
- confidence: 0.7,
1544
- indicators: ['docker-compose.yml memcached service'],
1545
- source: composeFile,
1546
- });
1547
- }
1548
- // Elasticsearch
1549
- if (upperContent.includes('ELASTICSEARCH') &&
1550
- !services.find((s) => s.service === 'elasticsearch')) {
1551
- services.push({
1552
- service: 'elasticsearch',
1553
- detected: true,
1554
- confidence: 0.7,
1555
- indicators: ['docker-compose.yml elasticsearch service'],
1556
- source: composeFile,
1557
- });
1558
- }
1559
- // Neo4j
1560
- if (upperContent.includes('NEO4J') && !services.find((s) => s.service === 'neo4j')) {
1561
- services.push({
1562
- service: 'neo4j',
1563
- detected: true,
1564
- confidence: 0.7,
1565
- indicators: ['docker-compose.yml neo4j service'],
1566
- source: composeFile,
1567
- });
1568
- }
1569
- // InfluxDB
1570
- if (upperContent.includes('INFLUXDB') && !services.find((s) => s.service === 'influxdb')) {
1571
- services.push({
1572
- service: 'influxdb',
1573
- detected: true,
1574
- confidence: 0.7,
1575
- indicators: ['docker-compose.yml influxdb service'],
1576
- source: composeFile,
1577
- });
1578
- }
1579
- // RabbitMQ
1580
- if (upperContent.includes('RABBITMQ') && !services.find((s) => s.service === 'rabbitmq')) {
1581
- services.push({
1582
- service: 'rabbitmq',
1583
- detected: true,
1584
- confidence: 0.7,
1585
- indicators: ['docker-compose.yml rabbitmq service'],
1586
- source: composeFile,
1587
- });
1588
- }
1589
- // Kafka
1590
- if (upperContent.includes('KAFKA') && !services.find((s) => s.service === 'kafka')) {
1591
- services.push({
1592
- service: 'kafka',
1593
- detected: true,
1594
- confidence: 0.7,
1595
- indicators: ['docker-compose.yml kafka service'],
1596
- source: composeFile,
1597
- });
1598
- }
1599
- // MinIO
1600
- if (upperContent.includes('MINIO') && !services.find((s) => s.service === 'minio')) {
1601
- services.push({
1602
- service: 'minio',
1603
- detected: true,
1604
- confidence: 0.7,
1605
- indicators: ['docker-compose.yml minio service'],
1606
- source: composeFile,
1607
- });
1608
- }
1609
- // SQL Server
1610
- if ((upperContent.includes('SQLSERVER') || upperContent.includes('MSSQL')) &&
1611
- !services.find((s) => s.service === 'sqlserver')) {
1612
- services.push({
1613
- service: 'sqlserver',
1614
- detected: true,
1615
- confidence: 0.7,
1616
- indicators: ['docker-compose.yml sqlserver service'],
1617
- source: composeFile,
1618
- });
1619
- }
1620
- }
1621
- catch {
1622
- // Ignore file read errors
1623
- }
1624
- }
1625
- }
1626
- // Detect container/orchestration services (Docker, Docker Compose, Kubernetes, Helm)
1627
- const dockerfile = path.join(cwd, 'Dockerfile');
1628
- const dockerignore = path.join(cwd, '.dockerignore');
1629
- const k8sDir = path.join(cwd, 'k8s');
1630
- const kubernetesDir = path.join(cwd, 'kubernetes');
1631
- const chartYaml = path.join(cwd, 'Chart.yaml');
1632
- const chartsDir = path.join(cwd, 'charts');
1633
- // Docker
1634
- if ((await fileExists(dockerfile)) || (await fileExists(dockerignore))) {
1635
- const indicators = [];
1636
- if (await fileExists(dockerfile))
1637
- indicators.push('Dockerfile');
1638
- if (await fileExists(dockerignore))
1639
- indicators.push('.dockerignore');
1640
- services.push({
1641
- service: 'docker',
1642
- detected: true,
1643
- confidence: 0.95,
1644
- indicators,
1645
- source: indicators[0],
1646
- });
1647
- }
1648
- // Docker Compose
1649
- if ((await fileExists(dockerCompose)) || (await fileExists(dockerComposeYaml))) {
1650
- const indicators = [];
1651
- if (await fileExists(dockerCompose))
1652
- indicators.push('docker-compose.yml');
1653
- if (await fileExists(dockerComposeYaml))
1654
- indicators.push('docker-compose.yaml');
1655
- services.push({
1656
- service: 'docker-compose',
1657
- detected: true,
1658
- confidence: 0.95,
1659
- indicators,
1660
- source: indicators[0],
1661
- });
1662
- }
1663
- // Kubernetes
1664
- const k8sDirExists = existsSync(k8sDir);
1665
- const kubernetesDirExists = existsSync(kubernetesDir);
1666
- let k8sYamlDetected = false;
1667
- if (!k8sDirExists && !kubernetesDirExists) {
1668
- // Scan root-level YAML files for Kubernetes resource kinds
1669
- try {
1670
- const rootFiles = await readdir(cwd);
1671
- const yamlFiles = rootFiles.filter((f) => f.endsWith('.yml') || f.endsWith('.yaml'));
1672
- for (const yamlFile of yamlFiles) {
1673
- const content = await readFile(path.join(cwd, yamlFile));
1674
- if (content.includes('kind: Deployment') ||
1675
- content.includes('kind: Service') ||
1676
- content.includes('kind: Ingress')) {
1677
- k8sYamlDetected = true;
1678
- break;
1679
- }
1680
- }
1681
- }
1682
- catch {
1683
- // Ignore read errors
1684
- }
1685
- }
1686
- if (k8sDirExists || kubernetesDirExists || k8sYamlDetected) {
1687
- const indicators = [];
1688
- if (k8sDirExists)
1689
- indicators.push('k8s/ directory');
1690
- if (kubernetesDirExists)
1691
- indicators.push('kubernetes/ directory');
1692
- if (k8sYamlDetected)
1693
- indicators.push('YAML with kind: Deployment/Service/Ingress');
1694
- services.push({
1695
- service: 'kubernetes',
1696
- detected: true,
1697
- confidence: 0.9,
1698
- indicators,
1699
- });
1700
- }
1701
- // Helm
1702
- const chartYamlExists = await fileExists(chartYaml);
1703
- const chartsDirExists = existsSync(chartsDir);
1704
- if (chartYamlExists || chartsDirExists) {
1705
- const indicators = [];
1706
- if (chartYamlExists)
1707
- indicators.push('Chart.yaml');
1708
- if (chartsDirExists)
1709
- indicators.push('charts/ directory');
1710
- services.push({
1711
- service: 'helm',
1712
- detected: true,
1713
- confidence: 0.9,
1714
- indicators,
1715
- });
1716
- }
1717
- // Add undetected services (for manual selection)
1718
- const detectedServices = new Set(services.map((s) => s.service));
1719
- const allServices = [
1720
- 'postgresql',
1721
- 'mysql',
1722
- 'mariadb',
1723
- 'sqlserver',
1724
- 'oracle',
1725
- 'sqlite',
1726
- 'mongodb',
1727
- 'cassandra',
1728
- 'dynamodb',
1729
- 'redis',
1730
- 'memcached',
1731
- 'elasticsearch',
1732
- 'neo4j',
1733
- 'influxdb',
1734
- 'rabbitmq',
1735
- 'kafka',
1736
- 's3',
1737
- 'azure_blob',
1738
- 'gcs',
1739
- 'minio',
1740
- 'docker',
1741
- 'docker-compose',
1742
- 'kubernetes',
1743
- 'helm',
1744
- 'sentry',
1745
- 'opentelemetry',
1746
- 'datadog',
1747
- 'pino',
1748
- 'winston',
1749
- 'prometheus',
1750
- ];
1751
- for (const service of allServices) {
1752
- if (!detectedServices.has(service)) {
1753
- services.push({
1754
- service,
1755
- detected: false,
1756
- confidence: 0,
1757
- indicators: [],
1758
- });
1759
- }
1760
- }
1761
- return services;
1762
- }
1763
- //# sourceMappingURL=detector.js.map