cgraphx 1.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 (936) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +243 -0
  3. package/dist/.claude-template/commands/my-commit.md +9 -0
  4. package/dist/.claude-template/commands/my-query.md +4 -0
  5. package/dist/.claude-template/hooks/context-monitor/context-monitor.cjs +216 -0
  6. package/dist/.claude-template/plugins/claude-hud/dist/claude-config-dir.d.ts +4 -0
  7. package/dist/.claude-template/plugins/claude-hud/dist/claude-config-dir.d.ts.map +1 -0
  8. package/dist/.claude-template/plugins/claude-hud/dist/claude-config-dir.js +24 -0
  9. package/dist/.claude-template/plugins/claude-hud/dist/claude-config-dir.js.map +1 -0
  10. package/dist/.claude-template/plugins/claude-hud/dist/config-reader.d.ts +8 -0
  11. package/dist/.claude-template/plugins/claude-hud/dist/config-reader.d.ts.map +1 -0
  12. package/dist/.claude-template/plugins/claude-hud/dist/config-reader.js +204 -0
  13. package/dist/.claude-template/plugins/claude-hud/dist/config-reader.js.map +1 -0
  14. package/dist/.claude-template/plugins/claude-hud/dist/config.d.ts +46 -0
  15. package/dist/.claude-template/plugins/claude-hud/dist/config.d.ts.map +1 -0
  16. package/dist/.claude-template/plugins/claude-hud/dist/config.js +220 -0
  17. package/dist/.claude-template/plugins/claude-hud/dist/config.js.map +1 -0
  18. package/dist/.claude-template/plugins/claude-hud/dist/constants.d.ts +10 -0
  19. package/dist/.claude-template/plugins/claude-hud/dist/constants.d.ts.map +1 -0
  20. package/dist/.claude-template/plugins/claude-hud/dist/constants.js +10 -0
  21. package/dist/.claude-template/plugins/claude-hud/dist/constants.js.map +1 -0
  22. package/dist/.claude-template/plugins/claude-hud/dist/debug.d.ts +6 -0
  23. package/dist/.claude-template/plugins/claude-hud/dist/debug.d.ts.map +1 -0
  24. package/dist/.claude-template/plugins/claude-hud/dist/debug.js +15 -0
  25. package/dist/.claude-template/plugins/claude-hud/dist/debug.js.map +1 -0
  26. package/dist/.claude-template/plugins/claude-hud/dist/extra-cmd.d.ts +23 -0
  27. package/dist/.claude-template/plugins/claude-hud/dist/extra-cmd.d.ts.map +1 -0
  28. package/dist/.claude-template/plugins/claude-hud/dist/extra-cmd.js +103 -0
  29. package/dist/.claude-template/plugins/claude-hud/dist/extra-cmd.js.map +1 -0
  30. package/dist/.claude-template/plugins/claude-hud/dist/git.d.ts +16 -0
  31. package/dist/.claude-template/plugins/claude-hud/dist/git.d.ts.map +1 -0
  32. package/dist/.claude-template/plugins/claude-hud/dist/git.js +86 -0
  33. package/dist/.claude-template/plugins/claude-hud/dist/git.js.map +1 -0
  34. package/dist/.claude-template/plugins/claude-hud/dist/index.d.ts +24 -0
  35. package/dist/.claude-template/plugins/claude-hud/dist/index.d.ts.map +1 -0
  36. package/dist/.claude-template/plugins/claude-hud/dist/index.js +97 -0
  37. package/dist/.claude-template/plugins/claude-hud/dist/index.js.map +1 -0
  38. package/dist/.claude-template/plugins/claude-hud/dist/render/agents-line.d.ts +3 -0
  39. package/dist/.claude-template/plugins/claude-hud/dist/render/agents-line.d.ts.map +1 -0
  40. package/dist/.claude-template/plugins/claude-hud/dist/render/agents-line.js +44 -0
  41. package/dist/.claude-template/plugins/claude-hud/dist/render/agents-line.js.map +1 -0
  42. package/dist/.claude-template/plugins/claude-hud/dist/render/colors.d.ts +12 -0
  43. package/dist/.claude-template/plugins/claude-hud/dist/render/colors.d.ts.map +1 -0
  44. package/dist/.claude-template/plugins/claude-hud/dist/render/colors.js +58 -0
  45. package/dist/.claude-template/plugins/claude-hud/dist/render/colors.js.map +1 -0
  46. package/dist/.claude-template/plugins/claude-hud/dist/render/index.d.ts +3 -0
  47. package/dist/.claude-template/plugins/claude-hud/dist/render/index.d.ts.map +1 -0
  48. package/dist/.claude-template/plugins/claude-hud/dist/render/index.js +379 -0
  49. package/dist/.claude-template/plugins/claude-hud/dist/render/index.js.map +1 -0
  50. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/environment.d.ts +3 -0
  51. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/environment.d.ts.map +1 -0
  52. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/environment.js +30 -0
  53. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/environment.js.map +1 -0
  54. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/identity.d.ts +3 -0
  55. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/identity.d.ts.map +1 -0
  56. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/identity.js +52 -0
  57. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/identity.js.map +1 -0
  58. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/index.d.ts +5 -0
  59. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/index.d.ts.map +1 -0
  60. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/index.js +5 -0
  61. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/index.js.map +1 -0
  62. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/project.d.ts +3 -0
  63. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/project.d.ts.map +1 -0
  64. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/project.js +74 -0
  65. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/project.js.map +1 -0
  66. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/usage.d.ts +3 -0
  67. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/usage.d.ts.map +1 -0
  68. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/usage.js +92 -0
  69. package/dist/.claude-template/plugins/claude-hud/dist/render/lines/usage.js.map +1 -0
  70. package/dist/.claude-template/plugins/claude-hud/dist/render/session-line.d.ts +7 -0
  71. package/dist/.claude-template/plugins/claude-hud/dist/render/session-line.d.ts.map +1 -0
  72. package/dist/.claude-template/plugins/claude-hud/dist/render/session-line.js +247 -0
  73. package/dist/.claude-template/plugins/claude-hud/dist/render/session-line.js.map +1 -0
  74. package/dist/.claude-template/plugins/claude-hud/dist/render/todos-line.d.ts +3 -0
  75. package/dist/.claude-template/plugins/claude-hud/dist/render/todos-line.d.ts.map +1 -0
  76. package/dist/.claude-template/plugins/claude-hud/dist/render/todos-line.js +25 -0
  77. package/dist/.claude-template/plugins/claude-hud/dist/render/todos-line.js.map +1 -0
  78. package/dist/.claude-template/plugins/claude-hud/dist/render/tools-line.d.ts +3 -0
  79. package/dist/.claude-template/plugins/claude-hud/dist/render/tools-line.d.ts.map +1 -0
  80. package/dist/.claude-template/plugins/claude-hud/dist/render/tools-line.js +43 -0
  81. package/dist/.claude-template/plugins/claude-hud/dist/render/tools-line.js.map +1 -0
  82. package/dist/.claude-template/plugins/claude-hud/dist/speed-tracker.d.ts +7 -0
  83. package/dist/.claude-template/plugins/claude-hud/dist/speed-tracker.d.ts.map +1 -0
  84. package/dist/.claude-template/plugins/claude-hud/dist/speed-tracker.js +62 -0
  85. package/dist/.claude-template/plugins/claude-hud/dist/speed-tracker.js.map +1 -0
  86. package/dist/.claude-template/plugins/claude-hud/dist/stdin.d.ts +9 -0
  87. package/dist/.claude-template/plugins/claude-hud/dist/stdin.d.ts.map +1 -0
  88. package/dist/.claude-template/plugins/claude-hud/dist/stdin.js +136 -0
  89. package/dist/.claude-template/plugins/claude-hud/dist/stdin.js.map +1 -0
  90. package/dist/.claude-template/plugins/claude-hud/dist/transcript.d.ts +3 -0
  91. package/dist/.claude-template/plugins/claude-hud/dist/transcript.d.ts.map +1 -0
  92. package/dist/.claude-template/plugins/claude-hud/dist/transcript.js +189 -0
  93. package/dist/.claude-template/plugins/claude-hud/dist/transcript.js.map +1 -0
  94. package/dist/.claude-template/plugins/claude-hud/dist/types.d.ts +79 -0
  95. package/dist/.claude-template/plugins/claude-hud/dist/types.d.ts.map +1 -0
  96. package/dist/.claude-template/plugins/claude-hud/dist/types.js +5 -0
  97. package/dist/.claude-template/plugins/claude-hud/dist/types.js.map +1 -0
  98. package/dist/.claude-template/plugins/claude-hud/dist/usage-api.d.ts +59 -0
  99. package/dist/.claude-template/plugins/claude-hud/dist/usage-api.d.ts.map +1 -0
  100. package/dist/.claude-template/plugins/claude-hud/dist/usage-api.js +733 -0
  101. package/dist/.claude-template/plugins/claude-hud/dist/usage-api.js.map +1 -0
  102. package/dist/.claude-template/skills/cgraphx/SKILL.md +143 -0
  103. package/dist/.claude-template/skills/cgraphx/agent-prompt.md +56 -0
  104. package/dist/.claude-template/skills/clarify-requirements/SKILL.md +425 -0
  105. package/dist/.claude-template/skills/code-impact-api/SKILL.md +143 -0
  106. package/dist/.claude-template/skills/code-impact-api/agent-prompt.md +51 -0
  107. package/dist/.claude-template/skills/code-impact-docgen/SKILL.md +366 -0
  108. package/dist/.claude-template/skills/code-impact-docgen/template-business-html.md +242 -0
  109. package/dist/.claude-template/skills/code-impact-docgen/template-business-md.md +107 -0
  110. package/dist/.claude-template/skills/code-impact-docgen/template-technical-html.md +205 -0
  111. package/dist/.claude-template/skills/code-impact-docgen/template-technical-md.md +155 -0
  112. package/dist/.claude-template/skills/code-impact-init/SKILL.md +800 -0
  113. package/dist/.claude-template/skills/code-impact-markdown/SKILL.md +345 -0
  114. package/dist/.claude-template/skills/code-impact-markdown/template-guide.md +68 -0
  115. package/dist/.claude-template/skills/code-impact-markdown/template-memory.md +82 -0
  116. package/dist/.claude-template/skills/code-impact-markdown/template-runbook.md +58 -0
  117. package/dist/.claude-template/skills/db-query/SKILL.md +166 -0
  118. package/dist/.claude-template/skills/db-query/agent-prompt.md +55 -0
  119. package/dist/.claude-template/skills/developer-timeline/SKILL.md +302 -0
  120. package/dist/.claude-template/skills/developer-timeline/demo-single-page-report.html +657 -0
  121. package/dist/.claude-template/skills/implementation/SKILL.md +136 -0
  122. package/dist/.claude-template/skills/subagent-implement/SKILL.md +225 -0
  123. package/dist/.claude-template/skills/subagent-implement/implementer-prompt.md +127 -0
  124. package/dist/.claude-template/skills/subagent-implement/quality-reviewer-prompt.md +130 -0
  125. package/dist/.claude-template/skills/subagent-implement/spec-reviewer-prompt.md +112 -0
  126. package/dist/.claude-template/skills/write-plan/SKILL.md +322 -0
  127. package/dist/.claude-template/skills/write-plan/plan-document-reviewer-prompt.md +134 -0
  128. package/dist/.claude-template/skills/write-prd/SKILL.md +242 -0
  129. package/dist/.claude-template/skills/write-spec/SKILL.md +278 -0
  130. package/dist/bin/codegraph.d.ts +26 -0
  131. package/dist/bin/codegraph.d.ts.map +1 -0
  132. package/dist/bin/codegraph.js +2014 -0
  133. package/dist/bin/codegraph.js.map +1 -0
  134. package/dist/bin/fatal-handler.d.ts +20 -0
  135. package/dist/bin/fatal-handler.d.ts.map +1 -0
  136. package/dist/bin/fatal-handler.js +118 -0
  137. package/dist/bin/fatal-handler.js.map +1 -0
  138. package/dist/bin/node-version-check.d.ts +51 -0
  139. package/dist/bin/node-version-check.d.ts.map +1 -0
  140. package/dist/bin/node-version-check.js +114 -0
  141. package/dist/bin/node-version-check.js.map +1 -0
  142. package/dist/bin/uninstall.d.ts +14 -0
  143. package/dist/bin/uninstall.d.ts.map +1 -0
  144. package/dist/bin/uninstall.js +36 -0
  145. package/dist/bin/uninstall.js.map +1 -0
  146. package/dist/context/formatter.d.ts +30 -0
  147. package/dist/context/formatter.d.ts.map +1 -0
  148. package/dist/context/formatter.js +263 -0
  149. package/dist/context/formatter.js.map +1 -0
  150. package/dist/context/index.d.ts +119 -0
  151. package/dist/context/index.d.ts.map +1 -0
  152. package/dist/context/index.js +1296 -0
  153. package/dist/context/index.js.map +1 -0
  154. package/dist/context/markers.d.ts +19 -0
  155. package/dist/context/markers.d.ts.map +1 -0
  156. package/dist/context/markers.js +22 -0
  157. package/dist/context/markers.js.map +1 -0
  158. package/dist/db/index.d.ts +122 -0
  159. package/dist/db/index.d.ts.map +1 -0
  160. package/dist/db/index.js +296 -0
  161. package/dist/db/index.js.map +1 -0
  162. package/dist/db/migrations.d.ts +44 -0
  163. package/dist/db/migrations.d.ts.map +1 -0
  164. package/dist/db/migrations.js +140 -0
  165. package/dist/db/migrations.js.map +1 -0
  166. package/dist/db/queries.d.ts +401 -0
  167. package/dist/db/queries.d.ts.map +1 -0
  168. package/dist/db/queries.js +1591 -0
  169. package/dist/db/queries.js.map +1 -0
  170. package/dist/db/schema.sql +152 -0
  171. package/dist/db/sqlite-adapter.d.ts +53 -0
  172. package/dist/db/sqlite-adapter.d.ts.map +1 -0
  173. package/dist/db/sqlite-adapter.js +117 -0
  174. package/dist/db/sqlite-adapter.js.map +1 -0
  175. package/dist/dbquery/cli.d.ts +17 -0
  176. package/dist/dbquery/cli.d.ts.map +1 -0
  177. package/dist/dbquery/cli.js +229 -0
  178. package/dist/dbquery/cli.js.map +1 -0
  179. package/dist/dbquery/config.d.ts +38 -0
  180. package/dist/dbquery/config.d.ts.map +1 -0
  181. package/dist/dbquery/config.js +244 -0
  182. package/dist/dbquery/config.js.map +1 -0
  183. package/dist/dbquery/constants.d.ts +40 -0
  184. package/dist/dbquery/constants.d.ts.map +1 -0
  185. package/dist/dbquery/constants.js +65 -0
  186. package/dist/dbquery/constants.js.map +1 -0
  187. package/dist/dbquery/drivers/mysql.d.ts +15 -0
  188. package/dist/dbquery/drivers/mysql.d.ts.map +1 -0
  189. package/dist/dbquery/drivers/mysql.js +102 -0
  190. package/dist/dbquery/drivers/mysql.js.map +1 -0
  191. package/dist/dbquery/drivers/postgres.d.ts +16 -0
  192. package/dist/dbquery/drivers/postgres.d.ts.map +1 -0
  193. package/dist/dbquery/drivers/postgres.js +105 -0
  194. package/dist/dbquery/drivers/postgres.js.map +1 -0
  195. package/dist/dbquery/errors.d.ts +40 -0
  196. package/dist/dbquery/errors.d.ts.map +1 -0
  197. package/dist/dbquery/errors.js +85 -0
  198. package/dist/dbquery/errors.js.map +1 -0
  199. package/dist/dbquery/executor.d.ts +30 -0
  200. package/dist/dbquery/executor.d.ts.map +1 -0
  201. package/dist/dbquery/executor.js +243 -0
  202. package/dist/dbquery/executor.js.map +1 -0
  203. package/dist/dbquery/format.d.ts +18 -0
  204. package/dist/dbquery/format.d.ts.map +1 -0
  205. package/dist/dbquery/format.js +174 -0
  206. package/dist/dbquery/format.js.map +1 -0
  207. package/dist/dbquery/index.d.ts +10 -0
  208. package/dist/dbquery/index.d.ts.map +1 -0
  209. package/dist/dbquery/index.js +23 -0
  210. package/dist/dbquery/index.js.map +1 -0
  211. package/dist/dbquery/init.d.ts +33 -0
  212. package/dist/dbquery/init.d.ts.map +1 -0
  213. package/dist/dbquery/init.js +125 -0
  214. package/dist/dbquery/init.js.map +1 -0
  215. package/dist/dbquery/logging.d.ts +22 -0
  216. package/dist/dbquery/logging.d.ts.map +1 -0
  217. package/dist/dbquery/logging.js +140 -0
  218. package/dist/dbquery/logging.js.map +1 -0
  219. package/dist/dbquery/mcp-tools.d.ts +29 -0
  220. package/dist/dbquery/mcp-tools.d.ts.map +1 -0
  221. package/dist/dbquery/mcp-tools.js +206 -0
  222. package/dist/dbquery/mcp-tools.js.map +1 -0
  223. package/dist/dbquery/queries.d.ts +31 -0
  224. package/dist/dbquery/queries.d.ts.map +1 -0
  225. package/dist/dbquery/queries.js +160 -0
  226. package/dist/dbquery/queries.js.map +1 -0
  227. package/dist/dbquery/safety.d.ts +35 -0
  228. package/dist/dbquery/safety.d.ts.map +1 -0
  229. package/dist/dbquery/safety.js +306 -0
  230. package/dist/dbquery/safety.js.map +1 -0
  231. package/dist/dbquery/types.d.ts +152 -0
  232. package/dist/dbquery/types.d.ts.map +1 -0
  233. package/dist/dbquery/types.js +10 -0
  234. package/dist/dbquery/types.js.map +1 -0
  235. package/dist/directory.d.ts +147 -0
  236. package/dist/directory.d.ts.map +1 -0
  237. package/dist/directory.js +523 -0
  238. package/dist/directory.js.map +1 -0
  239. package/dist/errors.d.ts +136 -0
  240. package/dist/errors.d.ts.map +1 -0
  241. package/dist/errors.js +219 -0
  242. package/dist/errors.js.map +1 -0
  243. package/dist/extraction/astro-extractor.d.ts +79 -0
  244. package/dist/extraction/astro-extractor.d.ts.map +1 -0
  245. package/dist/extraction/astro-extractor.js +320 -0
  246. package/dist/extraction/astro-extractor.js.map +1 -0
  247. package/dist/extraction/dfm-extractor.d.ts +31 -0
  248. package/dist/extraction/dfm-extractor.d.ts.map +1 -0
  249. package/dist/extraction/dfm-extractor.js +151 -0
  250. package/dist/extraction/dfm-extractor.js.map +1 -0
  251. package/dist/extraction/extraction-version.d.ts +25 -0
  252. package/dist/extraction/extraction-version.d.ts.map +1 -0
  253. package/dist/extraction/extraction-version.js +28 -0
  254. package/dist/extraction/extraction-version.js.map +1 -0
  255. package/dist/extraction/function-ref.d.ts +118 -0
  256. package/dist/extraction/function-ref.d.ts.map +1 -0
  257. package/dist/extraction/function-ref.js +727 -0
  258. package/dist/extraction/function-ref.js.map +1 -0
  259. package/dist/extraction/generated-detection.d.ts +30 -0
  260. package/dist/extraction/generated-detection.d.ts.map +1 -0
  261. package/dist/extraction/generated-detection.js +83 -0
  262. package/dist/extraction/generated-detection.js.map +1 -0
  263. package/dist/extraction/grammars.d.ts +114 -0
  264. package/dist/extraction/grammars.d.ts.map +1 -0
  265. package/dist/extraction/grammars.js +477 -0
  266. package/dist/extraction/grammars.js.map +1 -0
  267. package/dist/extraction/index.d.ts +175 -0
  268. package/dist/extraction/index.d.ts.map +1 -0
  269. package/dist/extraction/index.js +1887 -0
  270. package/dist/extraction/index.js.map +1 -0
  271. package/dist/extraction/languages/c-cpp.d.ts +12 -0
  272. package/dist/extraction/languages/c-cpp.d.ts.map +1 -0
  273. package/dist/extraction/languages/c-cpp.js +275 -0
  274. package/dist/extraction/languages/c-cpp.js.map +1 -0
  275. package/dist/extraction/languages/csharp.d.ts +25 -0
  276. package/dist/extraction/languages/csharp.d.ts.map +1 -0
  277. package/dist/extraction/languages/csharp.js +175 -0
  278. package/dist/extraction/languages/csharp.js.map +1 -0
  279. package/dist/extraction/languages/dart.d.ts +3 -0
  280. package/dist/extraction/languages/dart.d.ts.map +1 -0
  281. package/dist/extraction/languages/dart.js +374 -0
  282. package/dist/extraction/languages/dart.js.map +1 -0
  283. package/dist/extraction/languages/go.d.ts +3 -0
  284. package/dist/extraction/languages/go.d.ts.map +1 -0
  285. package/dist/extraction/languages/go.js +111 -0
  286. package/dist/extraction/languages/go.js.map +1 -0
  287. package/dist/extraction/languages/index.d.ts +10 -0
  288. package/dist/extraction/languages/index.d.ts.map +1 -0
  289. package/dist/extraction/languages/index.js +53 -0
  290. package/dist/extraction/languages/index.js.map +1 -0
  291. package/dist/extraction/languages/java.d.ts +3 -0
  292. package/dist/extraction/languages/java.d.ts.map +1 -0
  293. package/dist/extraction/languages/java.js +315 -0
  294. package/dist/extraction/languages/java.js.map +1 -0
  295. package/dist/extraction/languages/javascript.d.ts +3 -0
  296. package/dist/extraction/languages/javascript.d.ts.map +1 -0
  297. package/dist/extraction/languages/javascript.js +106 -0
  298. package/dist/extraction/languages/javascript.js.map +1 -0
  299. package/dist/extraction/languages/kotlin.d.ts +3 -0
  300. package/dist/extraction/languages/kotlin.d.ts.map +1 -0
  301. package/dist/extraction/languages/kotlin.js +379 -0
  302. package/dist/extraction/languages/kotlin.js.map +1 -0
  303. package/dist/extraction/languages/lua.d.ts +3 -0
  304. package/dist/extraction/languages/lua.d.ts.map +1 -0
  305. package/dist/extraction/languages/lua.js +150 -0
  306. package/dist/extraction/languages/lua.js.map +1 -0
  307. package/dist/extraction/languages/luau.d.ts +3 -0
  308. package/dist/extraction/languages/luau.d.ts.map +1 -0
  309. package/dist/extraction/languages/luau.js +37 -0
  310. package/dist/extraction/languages/luau.js.map +1 -0
  311. package/dist/extraction/languages/objc.d.ts +3 -0
  312. package/dist/extraction/languages/objc.d.ts.map +1 -0
  313. package/dist/extraction/languages/objc.js +175 -0
  314. package/dist/extraction/languages/objc.js.map +1 -0
  315. package/dist/extraction/languages/pascal.d.ts +3 -0
  316. package/dist/extraction/languages/pascal.d.ts.map +1 -0
  317. package/dist/extraction/languages/pascal.js +77 -0
  318. package/dist/extraction/languages/pascal.js.map +1 -0
  319. package/dist/extraction/languages/php.d.ts +3 -0
  320. package/dist/extraction/languages/php.d.ts.map +1 -0
  321. package/dist/extraction/languages/php.js +196 -0
  322. package/dist/extraction/languages/php.js.map +1 -0
  323. package/dist/extraction/languages/python.d.ts +3 -0
  324. package/dist/extraction/languages/python.d.ts.map +1 -0
  325. package/dist/extraction/languages/python.js +56 -0
  326. package/dist/extraction/languages/python.js.map +1 -0
  327. package/dist/extraction/languages/r.d.ts +3 -0
  328. package/dist/extraction/languages/r.d.ts.map +1 -0
  329. package/dist/extraction/languages/r.js +314 -0
  330. package/dist/extraction/languages/r.js.map +1 -0
  331. package/dist/extraction/languages/ruby.d.ts +3 -0
  332. package/dist/extraction/languages/ruby.d.ts.map +1 -0
  333. package/dist/extraction/languages/ruby.js +149 -0
  334. package/dist/extraction/languages/ruby.js.map +1 -0
  335. package/dist/extraction/languages/rust.d.ts +3 -0
  336. package/dist/extraction/languages/rust.d.ts.map +1 -0
  337. package/dist/extraction/languages/rust.js +142 -0
  338. package/dist/extraction/languages/rust.js.map +1 -0
  339. package/dist/extraction/languages/scala.d.ts +3 -0
  340. package/dist/extraction/languages/scala.d.ts.map +1 -0
  341. package/dist/extraction/languages/scala.js +209 -0
  342. package/dist/extraction/languages/scala.js.map +1 -0
  343. package/dist/extraction/languages/swift.d.ts +3 -0
  344. package/dist/extraction/languages/swift.d.ts.map +1 -0
  345. package/dist/extraction/languages/swift.js +152 -0
  346. package/dist/extraction/languages/swift.js.map +1 -0
  347. package/dist/extraction/languages/typescript.d.ts +16 -0
  348. package/dist/extraction/languages/typescript.d.ts.map +1 -0
  349. package/dist/extraction/languages/typescript.js +167 -0
  350. package/dist/extraction/languages/typescript.js.map +1 -0
  351. package/dist/extraction/liquid-extractor.d.ts +59 -0
  352. package/dist/extraction/liquid-extractor.d.ts.map +1 -0
  353. package/dist/extraction/liquid-extractor.js +357 -0
  354. package/dist/extraction/liquid-extractor.js.map +1 -0
  355. package/dist/extraction/mybatis-extractor.d.ts +48 -0
  356. package/dist/extraction/mybatis-extractor.d.ts.map +1 -0
  357. package/dist/extraction/mybatis-extractor.js +198 -0
  358. package/dist/extraction/mybatis-extractor.js.map +1 -0
  359. package/dist/extraction/parse-worker.d.ts +8 -0
  360. package/dist/extraction/parse-worker.d.ts.map +1 -0
  361. package/dist/extraction/parse-worker.js +97 -0
  362. package/dist/extraction/parse-worker.js.map +1 -0
  363. package/dist/extraction/razor-extractor.d.ts +42 -0
  364. package/dist/extraction/razor-extractor.d.ts.map +1 -0
  365. package/dist/extraction/razor-extractor.js +285 -0
  366. package/dist/extraction/razor-extractor.js.map +1 -0
  367. package/dist/extraction/svelte-extractor.d.ts +56 -0
  368. package/dist/extraction/svelte-extractor.d.ts.map +1 -0
  369. package/dist/extraction/svelte-extractor.js +275 -0
  370. package/dist/extraction/svelte-extractor.js.map +1 -0
  371. package/dist/extraction/tree-sitter-helpers.d.ts +28 -0
  372. package/dist/extraction/tree-sitter-helpers.d.ts.map +1 -0
  373. package/dist/extraction/tree-sitter-helpers.js +152 -0
  374. package/dist/extraction/tree-sitter-helpers.js.map +1 -0
  375. package/dist/extraction/tree-sitter-types.d.ts +239 -0
  376. package/dist/extraction/tree-sitter-types.d.ts.map +1 -0
  377. package/dist/extraction/tree-sitter-types.js +10 -0
  378. package/dist/extraction/tree-sitter-types.js.map +1 -0
  379. package/dist/extraction/tree-sitter.d.ts +647 -0
  380. package/dist/extraction/tree-sitter.d.ts.map +1 -0
  381. package/dist/extraction/tree-sitter.js +5592 -0
  382. package/dist/extraction/tree-sitter.js.map +1 -0
  383. package/dist/extraction/vue-extractor.d.ts +51 -0
  384. package/dist/extraction/vue-extractor.d.ts.map +1 -0
  385. package/dist/extraction/vue-extractor.js +254 -0
  386. package/dist/extraction/vue-extractor.js.map +1 -0
  387. package/dist/extraction/wasm/tree-sitter-c_sharp.wasm +0 -0
  388. package/dist/extraction/wasm/tree-sitter-lua.wasm +0 -0
  389. package/dist/extraction/wasm/tree-sitter-luau.wasm +0 -0
  390. package/dist/extraction/wasm/tree-sitter-pascal.wasm +0 -0
  391. package/dist/extraction/wasm/tree-sitter-r.wasm +0 -0
  392. package/dist/extraction/wasm/tree-sitter-scala.wasm +0 -0
  393. package/dist/extraction/wasm-runtime-flags.d.ts +38 -0
  394. package/dist/extraction/wasm-runtime-flags.d.ts.map +1 -0
  395. package/dist/extraction/wasm-runtime-flags.js +106 -0
  396. package/dist/extraction/wasm-runtime-flags.js.map +1 -0
  397. package/dist/graph/index.d.ts +8 -0
  398. package/dist/graph/index.d.ts.map +1 -0
  399. package/dist/graph/index.js +13 -0
  400. package/dist/graph/index.js.map +1 -0
  401. package/dist/graph/queries.d.ts +106 -0
  402. package/dist/graph/queries.d.ts.map +1 -0
  403. package/dist/graph/queries.js +339 -0
  404. package/dist/graph/queries.js.map +1 -0
  405. package/dist/graph/traversal.d.ts +127 -0
  406. package/dist/graph/traversal.d.ts.map +1 -0
  407. package/dist/graph/traversal.js +540 -0
  408. package/dist/graph/traversal.js.map +1 -0
  409. package/dist/index.d.ts +563 -0
  410. package/dist/index.d.ts.map +1 -0
  411. package/dist/index.js +1041 -0
  412. package/dist/index.js.map +1 -0
  413. package/dist/installer/claude-assets.d.ts +45 -0
  414. package/dist/installer/claude-assets.d.ts.map +1 -0
  415. package/dist/installer/claude-assets.js +144 -0
  416. package/dist/installer/claude-assets.js.map +1 -0
  417. package/dist/installer/config-writer.d.ts +28 -0
  418. package/dist/installer/config-writer.d.ts.map +1 -0
  419. package/dist/installer/config-writer.js +91 -0
  420. package/dist/installer/config-writer.js.map +1 -0
  421. package/dist/installer/index.d.ts +101 -0
  422. package/dist/installer/index.d.ts.map +1 -0
  423. package/dist/installer/index.js +692 -0
  424. package/dist/installer/index.js.map +1 -0
  425. package/dist/installer/instructions-template.d.ts +41 -0
  426. package/dist/installer/instructions-template.d.ts.map +1 -0
  427. package/dist/installer/instructions-template.js +53 -0
  428. package/dist/installer/instructions-template.js.map +1 -0
  429. package/dist/installer/targets/antigravity.d.ts +57 -0
  430. package/dist/installer/targets/antigravity.d.ts.map +1 -0
  431. package/dist/installer/targets/antigravity.js +308 -0
  432. package/dist/installer/targets/antigravity.js.map +1 -0
  433. package/dist/installer/targets/claude.d.ts +66 -0
  434. package/dist/installer/targets/claude.d.ts.map +1 -0
  435. package/dist/installer/targets/claude.js +564 -0
  436. package/dist/installer/targets/claude.js.map +1 -0
  437. package/dist/installer/targets/codex.d.ts +18 -0
  438. package/dist/installer/targets/codex.d.ts.map +1 -0
  439. package/dist/installer/targets/codex.js +185 -0
  440. package/dist/installer/targets/codex.js.map +1 -0
  441. package/dist/installer/targets/cursor.d.ts +35 -0
  442. package/dist/installer/targets/cursor.d.ts.map +1 -0
  443. package/dist/installer/targets/cursor.js +254 -0
  444. package/dist/installer/targets/cursor.js.map +1 -0
  445. package/dist/installer/targets/gemini.d.ts +26 -0
  446. package/dist/installer/targets/gemini.d.ts.map +1 -0
  447. package/dist/installer/targets/gemini.js +165 -0
  448. package/dist/installer/targets/gemini.js.map +1 -0
  449. package/dist/installer/targets/hermes.d.ts +18 -0
  450. package/dist/installer/targets/hermes.d.ts.map +1 -0
  451. package/dist/installer/targets/hermes.js +359 -0
  452. package/dist/installer/targets/hermes.js.map +1 -0
  453. package/dist/installer/targets/kiro.d.ts +27 -0
  454. package/dist/installer/targets/kiro.d.ts.map +1 -0
  455. package/dist/installer/targets/kiro.js +178 -0
  456. package/dist/installer/targets/kiro.js.map +1 -0
  457. package/dist/installer/targets/opencode.d.ts +38 -0
  458. package/dist/installer/targets/opencode.d.ts.map +1 -0
  459. package/dist/installer/targets/opencode.js +288 -0
  460. package/dist/installer/targets/opencode.js.map +1 -0
  461. package/dist/installer/targets/registry.d.ts +35 -0
  462. package/dist/installer/targets/registry.d.ts.map +1 -0
  463. package/dist/installer/targets/registry.js +91 -0
  464. package/dist/installer/targets/registry.js.map +1 -0
  465. package/dist/installer/targets/shared.d.ts +101 -0
  466. package/dist/installer/targets/shared.d.ts.map +1 -0
  467. package/dist/installer/targets/shared.js +264 -0
  468. package/dist/installer/targets/shared.js.map +1 -0
  469. package/dist/installer/targets/toml.d.ts +52 -0
  470. package/dist/installer/targets/toml.d.ts.map +1 -0
  471. package/dist/installer/targets/toml.js +147 -0
  472. package/dist/installer/targets/toml.js.map +1 -0
  473. package/dist/installer/targets/types.d.ts +108 -0
  474. package/dist/installer/targets/types.d.ts.map +1 -0
  475. package/dist/installer/targets/types.js +16 -0
  476. package/dist/installer/targets/types.js.map +1 -0
  477. package/dist/markdown/cli.d.ts +16 -0
  478. package/dist/markdown/cli.d.ts.map +1 -0
  479. package/dist/markdown/cli.js +533 -0
  480. package/dist/markdown/cli.js.map +1 -0
  481. package/dist/markdown/constants.d.ts +22 -0
  482. package/dist/markdown/constants.d.ts.map +1 -0
  483. package/dist/markdown/constants.js +71 -0
  484. package/dist/markdown/constants.js.map +1 -0
  485. package/dist/markdown/dedup.d.ts +20 -0
  486. package/dist/markdown/dedup.d.ts.map +1 -0
  487. package/dist/markdown/dedup.js +64 -0
  488. package/dist/markdown/dedup.js.map +1 -0
  489. package/dist/markdown/errors.d.ts +22 -0
  490. package/dist/markdown/errors.d.ts.map +1 -0
  491. package/dist/markdown/errors.js +45 -0
  492. package/dist/markdown/errors.js.map +1 -0
  493. package/dist/markdown/extractor.d.ts +43 -0
  494. package/dist/markdown/extractor.d.ts.map +1 -0
  495. package/dist/markdown/extractor.js +152 -0
  496. package/dist/markdown/extractor.js.map +1 -0
  497. package/dist/markdown/frontmatter-parser.d.ts +47 -0
  498. package/dist/markdown/frontmatter-parser.d.ts.map +1 -0
  499. package/dist/markdown/frontmatter-parser.js +199 -0
  500. package/dist/markdown/frontmatter-parser.js.map +1 -0
  501. package/dist/markdown/indexer.d.ts +34 -0
  502. package/dist/markdown/indexer.d.ts.map +1 -0
  503. package/dist/markdown/indexer.js +256 -0
  504. package/dist/markdown/indexer.js.map +1 -0
  505. package/dist/markdown/mcp-tools.d.ts +33 -0
  506. package/dist/markdown/mcp-tools.d.ts.map +1 -0
  507. package/dist/markdown/mcp-tools.js +300 -0
  508. package/dist/markdown/mcp-tools.js.map +1 -0
  509. package/dist/markdown/query.d.ts +108 -0
  510. package/dist/markdown/query.d.ts.map +1 -0
  511. package/dist/markdown/query.js +570 -0
  512. package/dist/markdown/query.js.map +1 -0
  513. package/dist/markdown/schema-bootstrap.d.ts +40 -0
  514. package/dist/markdown/schema-bootstrap.d.ts.map +1 -0
  515. package/dist/markdown/schema-bootstrap.js +85 -0
  516. package/dist/markdown/schema-bootstrap.js.map +1 -0
  517. package/dist/markdown/schema.sql +124 -0
  518. package/dist/markdown/store.d.ts +77 -0
  519. package/dist/markdown/store.d.ts.map +1 -0
  520. package/dist/markdown/store.js +194 -0
  521. package/dist/markdown/store.js.map +1 -0
  522. package/dist/markdown/summary-extractor.d.ts +22 -0
  523. package/dist/markdown/summary-extractor.d.ts.map +1 -0
  524. package/dist/markdown/summary-extractor.js +66 -0
  525. package/dist/markdown/summary-extractor.js.map +1 -0
  526. package/dist/markdown/types.d.ts +159 -0
  527. package/dist/markdown/types.d.ts.map +1 -0
  528. package/dist/markdown/types.js +9 -0
  529. package/dist/markdown/types.js.map +1 -0
  530. package/dist/markdown/validator.d.ts +44 -0
  531. package/dist/markdown/validator.d.ts.map +1 -0
  532. package/dist/markdown/validator.js +95 -0
  533. package/dist/markdown/validator.js.map +1 -0
  534. package/dist/mcp/daemon-manager.d.ts +42 -0
  535. package/dist/mcp/daemon-manager.d.ts.map +1 -0
  536. package/dist/mcp/daemon-manager.js +129 -0
  537. package/dist/mcp/daemon-manager.js.map +1 -0
  538. package/dist/mcp/daemon-paths.d.ts +46 -0
  539. package/dist/mcp/daemon-paths.d.ts.map +1 -0
  540. package/dist/mcp/daemon-paths.js +125 -0
  541. package/dist/mcp/daemon-paths.js.map +1 -0
  542. package/dist/mcp/daemon-registry.d.ts +47 -0
  543. package/dist/mcp/daemon-registry.d.ts.map +1 -0
  544. package/dist/mcp/daemon-registry.js +229 -0
  545. package/dist/mcp/daemon-registry.js.map +1 -0
  546. package/dist/mcp/daemon.d.ts +220 -0
  547. package/dist/mcp/daemon.d.ts.map +1 -0
  548. package/dist/mcp/daemon.js +637 -0
  549. package/dist/mcp/daemon.js.map +1 -0
  550. package/dist/mcp/dynamic-boundaries.d.ts +41 -0
  551. package/dist/mcp/dynamic-boundaries.d.ts.map +1 -0
  552. package/dist/mcp/dynamic-boundaries.js +359 -0
  553. package/dist/mcp/dynamic-boundaries.js.map +1 -0
  554. package/dist/mcp/engine.d.ts +105 -0
  555. package/dist/mcp/engine.d.ts.map +1 -0
  556. package/dist/mcp/engine.js +278 -0
  557. package/dist/mcp/engine.js.map +1 -0
  558. package/dist/mcp/index.d.ts +113 -0
  559. package/dist/mcp/index.d.ts.map +1 -0
  560. package/dist/mcp/index.js +499 -0
  561. package/dist/mcp/index.js.map +1 -0
  562. package/dist/mcp/liveness-watchdog.d.ts +18 -0
  563. package/dist/mcp/liveness-watchdog.d.ts.map +1 -0
  564. package/dist/mcp/liveness-watchdog.js +207 -0
  565. package/dist/mcp/liveness-watchdog.js.map +1 -0
  566. package/dist/mcp/ppid-watchdog.d.ts +44 -0
  567. package/dist/mcp/ppid-watchdog.d.ts.map +1 -0
  568. package/dist/mcp/ppid-watchdog.js +27 -0
  569. package/dist/mcp/ppid-watchdog.js.map +1 -0
  570. package/dist/mcp/proxy.d.ts +87 -0
  571. package/dist/mcp/proxy.d.ts.map +1 -0
  572. package/dist/mcp/proxy.js +641 -0
  573. package/dist/mcp/proxy.js.map +1 -0
  574. package/dist/mcp/server-instructions.d.ts +34 -0
  575. package/dist/mcp/server-instructions.d.ts.map +1 -0
  576. package/dist/mcp/server-instructions.js +106 -0
  577. package/dist/mcp/server-instructions.js.map +1 -0
  578. package/dist/mcp/session.d.ts +79 -0
  579. package/dist/mcp/session.d.ts.map +1 -0
  580. package/dist/mcp/session.js +330 -0
  581. package/dist/mcp/session.js.map +1 -0
  582. package/dist/mcp/stdin-teardown.d.ts +27 -0
  583. package/dist/mcp/stdin-teardown.d.ts.map +1 -0
  584. package/dist/mcp/stdin-teardown.js +49 -0
  585. package/dist/mcp/stdin-teardown.js.map +1 -0
  586. package/dist/mcp/tools.d.ts +547 -0
  587. package/dist/mcp/tools.d.ts.map +1 -0
  588. package/dist/mcp/tools.js +4122 -0
  589. package/dist/mcp/tools.js.map +1 -0
  590. package/dist/mcp/transport.d.ts +188 -0
  591. package/dist/mcp/transport.d.ts.map +1 -0
  592. package/dist/mcp/transport.js +359 -0
  593. package/dist/mcp/transport.js.map +1 -0
  594. package/dist/mcp/version.d.ts +19 -0
  595. package/dist/mcp/version.d.ts.map +1 -0
  596. package/dist/mcp/version.js +71 -0
  597. package/dist/mcp/version.js.map +1 -0
  598. package/dist/project-config.d.ts +36 -0
  599. package/dist/project-config.d.ts.map +1 -0
  600. package/dist/project-config.js +235 -0
  601. package/dist/project-config.js.map +1 -0
  602. package/dist/reasoning/config.d.ts +45 -0
  603. package/dist/reasoning/config.d.ts.map +1 -0
  604. package/dist/reasoning/config.js +171 -0
  605. package/dist/reasoning/config.js.map +1 -0
  606. package/dist/reasoning/credentials.d.ts +5 -0
  607. package/dist/reasoning/credentials.d.ts.map +1 -0
  608. package/dist/reasoning/credentials.js +83 -0
  609. package/dist/reasoning/credentials.js.map +1 -0
  610. package/dist/reasoning/login.d.ts +21 -0
  611. package/dist/reasoning/login.d.ts.map +1 -0
  612. package/dist/reasoning/login.js +85 -0
  613. package/dist/reasoning/login.js.map +1 -0
  614. package/dist/reasoning/reasoner.d.ts +43 -0
  615. package/dist/reasoning/reasoner.d.ts.map +1 -0
  616. package/dist/reasoning/reasoner.js +308 -0
  617. package/dist/reasoning/reasoner.js.map +1 -0
  618. package/dist/resolution/c-fnptr-synthesizer.d.ts +33 -0
  619. package/dist/resolution/c-fnptr-synthesizer.d.ts.map +1 -0
  620. package/dist/resolution/c-fnptr-synthesizer.js +352 -0
  621. package/dist/resolution/c-fnptr-synthesizer.js.map +1 -0
  622. package/dist/resolution/callback-synthesizer.d.ts +15 -0
  623. package/dist/resolution/callback-synthesizer.d.ts.map +1 -0
  624. package/dist/resolution/callback-synthesizer.js +2926 -0
  625. package/dist/resolution/callback-synthesizer.js.map +1 -0
  626. package/dist/resolution/frameworks/astro.d.ts +9 -0
  627. package/dist/resolution/frameworks/astro.d.ts.map +1 -0
  628. package/dist/resolution/frameworks/astro.js +169 -0
  629. package/dist/resolution/frameworks/astro.js.map +1 -0
  630. package/dist/resolution/frameworks/cargo-workspace.d.ts +18 -0
  631. package/dist/resolution/frameworks/cargo-workspace.d.ts.map +1 -0
  632. package/dist/resolution/frameworks/cargo-workspace.js +225 -0
  633. package/dist/resolution/frameworks/cargo-workspace.js.map +1 -0
  634. package/dist/resolution/frameworks/csharp.d.ts +8 -0
  635. package/dist/resolution/frameworks/csharp.d.ts.map +1 -0
  636. package/dist/resolution/frameworks/csharp.js +241 -0
  637. package/dist/resolution/frameworks/csharp.js.map +1 -0
  638. package/dist/resolution/frameworks/drupal.d.ts +51 -0
  639. package/dist/resolution/frameworks/drupal.d.ts.map +1 -0
  640. package/dist/resolution/frameworks/drupal.js +367 -0
  641. package/dist/resolution/frameworks/drupal.js.map +1 -0
  642. package/dist/resolution/frameworks/expo-modules.d.ts +3 -0
  643. package/dist/resolution/frameworks/expo-modules.d.ts.map +1 -0
  644. package/dist/resolution/frameworks/expo-modules.js +148 -0
  645. package/dist/resolution/frameworks/expo-modules.js.map +1 -0
  646. package/dist/resolution/frameworks/express.d.ts +8 -0
  647. package/dist/resolution/frameworks/express.d.ts.map +1 -0
  648. package/dist/resolution/frameworks/express.js +308 -0
  649. package/dist/resolution/frameworks/express.js.map +1 -0
  650. package/dist/resolution/frameworks/fabric.d.ts +3 -0
  651. package/dist/resolution/frameworks/fabric.d.ts.map +1 -0
  652. package/dist/resolution/frameworks/fabric.js +354 -0
  653. package/dist/resolution/frameworks/fabric.js.map +1 -0
  654. package/dist/resolution/frameworks/go.d.ts +8 -0
  655. package/dist/resolution/frameworks/go.d.ts.map +1 -0
  656. package/dist/resolution/frameworks/go.js +161 -0
  657. package/dist/resolution/frameworks/go.js.map +1 -0
  658. package/dist/resolution/frameworks/goframe.d.ts +41 -0
  659. package/dist/resolution/frameworks/goframe.d.ts.map +1 -0
  660. package/dist/resolution/frameworks/goframe.js +112 -0
  661. package/dist/resolution/frameworks/goframe.js.map +1 -0
  662. package/dist/resolution/frameworks/index.d.ts +50 -0
  663. package/dist/resolution/frameworks/index.d.ts.map +1 -0
  664. package/dist/resolution/frameworks/index.js +169 -0
  665. package/dist/resolution/frameworks/index.js.map +1 -0
  666. package/dist/resolution/frameworks/java.d.ts +8 -0
  667. package/dist/resolution/frameworks/java.d.ts.map +1 -0
  668. package/dist/resolution/frameworks/java.js +509 -0
  669. package/dist/resolution/frameworks/java.js.map +1 -0
  670. package/dist/resolution/frameworks/laravel.d.ts +13 -0
  671. package/dist/resolution/frameworks/laravel.d.ts.map +1 -0
  672. package/dist/resolution/frameworks/laravel.js +257 -0
  673. package/dist/resolution/frameworks/laravel.js.map +1 -0
  674. package/dist/resolution/frameworks/nestjs.d.ts +26 -0
  675. package/dist/resolution/frameworks/nestjs.d.ts.map +1 -0
  676. package/dist/resolution/frameworks/nestjs.js +698 -0
  677. package/dist/resolution/frameworks/nestjs.js.map +1 -0
  678. package/dist/resolution/frameworks/play.d.ts +19 -0
  679. package/dist/resolution/frameworks/play.d.ts.map +1 -0
  680. package/dist/resolution/frameworks/play.js +111 -0
  681. package/dist/resolution/frameworks/play.js.map +1 -0
  682. package/dist/resolution/frameworks/python.d.ts +10 -0
  683. package/dist/resolution/frameworks/python.d.ts.map +1 -0
  684. package/dist/resolution/frameworks/python.js +400 -0
  685. package/dist/resolution/frameworks/python.js.map +1 -0
  686. package/dist/resolution/frameworks/react-native.d.ts +3 -0
  687. package/dist/resolution/frameworks/react-native.d.ts.map +1 -0
  688. package/dist/resolution/frameworks/react-native.js +410 -0
  689. package/dist/resolution/frameworks/react-native.js.map +1 -0
  690. package/dist/resolution/frameworks/react.d.ts +8 -0
  691. package/dist/resolution/frameworks/react.d.ts.map +1 -0
  692. package/dist/resolution/frameworks/react.js +334 -0
  693. package/dist/resolution/frameworks/react.js.map +1 -0
  694. package/dist/resolution/frameworks/ruby.d.ts +8 -0
  695. package/dist/resolution/frameworks/ruby.d.ts.map +1 -0
  696. package/dist/resolution/frameworks/ruby.js +302 -0
  697. package/dist/resolution/frameworks/ruby.js.map +1 -0
  698. package/dist/resolution/frameworks/rust.d.ts +8 -0
  699. package/dist/resolution/frameworks/rust.d.ts.map +1 -0
  700. package/dist/resolution/frameworks/rust.js +304 -0
  701. package/dist/resolution/frameworks/rust.js.map +1 -0
  702. package/dist/resolution/frameworks/svelte.d.ts +9 -0
  703. package/dist/resolution/frameworks/svelte.d.ts.map +1 -0
  704. package/dist/resolution/frameworks/svelte.js +253 -0
  705. package/dist/resolution/frameworks/svelte.js.map +1 -0
  706. package/dist/resolution/frameworks/swift-objc.d.ts +37 -0
  707. package/dist/resolution/frameworks/swift-objc.d.ts.map +1 -0
  708. package/dist/resolution/frameworks/swift-objc.js +252 -0
  709. package/dist/resolution/frameworks/swift-objc.js.map +1 -0
  710. package/dist/resolution/frameworks/swift.d.ts +10 -0
  711. package/dist/resolution/frameworks/swift.d.ts.map +1 -0
  712. package/dist/resolution/frameworks/swift.js +400 -0
  713. package/dist/resolution/frameworks/swift.js.map +1 -0
  714. package/dist/resolution/frameworks/vue.d.ts +9 -0
  715. package/dist/resolution/frameworks/vue.d.ts.map +1 -0
  716. package/dist/resolution/frameworks/vue.js +303 -0
  717. package/dist/resolution/frameworks/vue.js.map +1 -0
  718. package/dist/resolution/go-module.d.ts +26 -0
  719. package/dist/resolution/go-module.d.ts.map +1 -0
  720. package/dist/resolution/go-module.js +78 -0
  721. package/dist/resolution/go-module.js.map +1 -0
  722. package/dist/resolution/goframe-synthesizer.d.ts +28 -0
  723. package/dist/resolution/goframe-synthesizer.d.ts.map +1 -0
  724. package/dist/resolution/goframe-synthesizer.js +158 -0
  725. package/dist/resolution/goframe-synthesizer.js.map +1 -0
  726. package/dist/resolution/import-resolver.d.ts +78 -0
  727. package/dist/resolution/import-resolver.d.ts.map +1 -0
  728. package/dist/resolution/import-resolver.js +1849 -0
  729. package/dist/resolution/import-resolver.js.map +1 -0
  730. package/dist/resolution/index.d.ts +196 -0
  731. package/dist/resolution/index.d.ts.map +1 -0
  732. package/dist/resolution/index.js +1328 -0
  733. package/dist/resolution/index.js.map +1 -0
  734. package/dist/resolution/lru-cache.d.ts +24 -0
  735. package/dist/resolution/lru-cache.d.ts.map +1 -0
  736. package/dist/resolution/lru-cache.js +62 -0
  737. package/dist/resolution/lru-cache.js.map +1 -0
  738. package/dist/resolution/name-matcher.d.ts +93 -0
  739. package/dist/resolution/name-matcher.d.ts.map +1 -0
  740. package/dist/resolution/name-matcher.js +1212 -0
  741. package/dist/resolution/name-matcher.js.map +1 -0
  742. package/dist/resolution/path-aliases.d.ts +68 -0
  743. package/dist/resolution/path-aliases.d.ts.map +1 -0
  744. package/dist/resolution/path-aliases.js +238 -0
  745. package/dist/resolution/path-aliases.js.map +1 -0
  746. package/dist/resolution/strip-comments.d.ts +27 -0
  747. package/dist/resolution/strip-comments.d.ts.map +1 -0
  748. package/dist/resolution/strip-comments.js +443 -0
  749. package/dist/resolution/strip-comments.js.map +1 -0
  750. package/dist/resolution/swift-objc-bridge.d.ts +134 -0
  751. package/dist/resolution/swift-objc-bridge.d.ts.map +1 -0
  752. package/dist/resolution/swift-objc-bridge.js +256 -0
  753. package/dist/resolution/swift-objc-bridge.js.map +1 -0
  754. package/dist/resolution/types.d.ts +233 -0
  755. package/dist/resolution/types.d.ts.map +1 -0
  756. package/dist/resolution/types.js +8 -0
  757. package/dist/resolution/types.js.map +1 -0
  758. package/dist/resolution/workspace-packages.d.ts +48 -0
  759. package/dist/resolution/workspace-packages.d.ts.map +1 -0
  760. package/dist/resolution/workspace-packages.js +208 -0
  761. package/dist/resolution/workspace-packages.js.map +1 -0
  762. package/dist/search/query-parser.d.ts +57 -0
  763. package/dist/search/query-parser.d.ts.map +1 -0
  764. package/dist/search/query-parser.js +177 -0
  765. package/dist/search/query-parser.js.map +1 -0
  766. package/dist/search/query-utils.d.ts +87 -0
  767. package/dist/search/query-utils.d.ts.map +1 -0
  768. package/dist/search/query-utils.js +449 -0
  769. package/dist/search/query-utils.js.map +1 -0
  770. package/dist/sync/git-hooks.d.ts +45 -0
  771. package/dist/sync/git-hooks.d.ts.map +1 -0
  772. package/dist/sync/git-hooks.js +225 -0
  773. package/dist/sync/git-hooks.js.map +1 -0
  774. package/dist/sync/index.d.ts +19 -0
  775. package/dist/sync/index.d.ts.map +1 -0
  776. package/dist/sync/index.js +35 -0
  777. package/dist/sync/index.js.map +1 -0
  778. package/dist/sync/watch-policy.d.ts +48 -0
  779. package/dist/sync/watch-policy.d.ts.map +1 -0
  780. package/dist/sync/watch-policy.js +124 -0
  781. package/dist/sync/watch-policy.js.map +1 -0
  782. package/dist/sync/watcher.d.ts +350 -0
  783. package/dist/sync/watcher.d.ts.map +1 -0
  784. package/dist/sync/watcher.js +811 -0
  785. package/dist/sync/watcher.js.map +1 -0
  786. package/dist/sync/worktree.d.ts +54 -0
  787. package/dist/sync/worktree.d.ts.map +1 -0
  788. package/dist/sync/worktree.js +137 -0
  789. package/dist/sync/worktree.js.map +1 -0
  790. package/dist/telemetry/index.d.ts +143 -0
  791. package/dist/telemetry/index.d.ts.map +1 -0
  792. package/dist/telemetry/index.js +541 -0
  793. package/dist/telemetry/index.js.map +1 -0
  794. package/dist/timeline/bash-semantics.d.ts +52 -0
  795. package/dist/timeline/bash-semantics.d.ts.map +1 -0
  796. package/dist/timeline/bash-semantics.js +376 -0
  797. package/dist/timeline/bash-semantics.js.map +1 -0
  798. package/dist/timeline/cli.d.ts +50 -0
  799. package/dist/timeline/cli.d.ts.map +1 -0
  800. package/dist/timeline/cli.js +367 -0
  801. package/dist/timeline/cli.js.map +1 -0
  802. package/dist/timeline/constants.d.ts +62 -0
  803. package/dist/timeline/constants.d.ts.map +1 -0
  804. package/dist/timeline/constants.js +73 -0
  805. package/dist/timeline/constants.js.map +1 -0
  806. package/dist/timeline/errors.d.ts +27 -0
  807. package/dist/timeline/errors.d.ts.map +1 -0
  808. package/dist/timeline/errors.js +51 -0
  809. package/dist/timeline/errors.js.map +1 -0
  810. package/dist/timeline/hook-runner.d.ts +36 -0
  811. package/dist/timeline/hook-runner.d.ts.map +1 -0
  812. package/dist/timeline/hook-runner.js +61 -0
  813. package/dist/timeline/hook-runner.js.map +1 -0
  814. package/dist/timeline/hooks.d.ts +45 -0
  815. package/dist/timeline/hooks.d.ts.map +1 -0
  816. package/dist/timeline/hooks.js +364 -0
  817. package/dist/timeline/hooks.js.map +1 -0
  818. package/dist/timeline/index.d.ts +12 -0
  819. package/dist/timeline/index.d.ts.map +1 -0
  820. package/dist/timeline/index.js +28 -0
  821. package/dist/timeline/index.js.map +1 -0
  822. package/dist/timeline/indexer.d.ts +37 -0
  823. package/dist/timeline/indexer.d.ts.map +1 -0
  824. package/dist/timeline/indexer.js +76 -0
  825. package/dist/timeline/indexer.js.map +1 -0
  826. package/dist/timeline/installer.d.ts +33 -0
  827. package/dist/timeline/installer.d.ts.map +1 -0
  828. package/dist/timeline/installer.js +255 -0
  829. package/dist/timeline/installer.js.map +1 -0
  830. package/dist/timeline/payload.d.ts +31 -0
  831. package/dist/timeline/payload.d.ts.map +1 -0
  832. package/dist/timeline/payload.js +58 -0
  833. package/dist/timeline/payload.js.map +1 -0
  834. package/dist/timeline/post-tool-summary.d.ts +29 -0
  835. package/dist/timeline/post-tool-summary.d.ts.map +1 -0
  836. package/dist/timeline/post-tool-summary.js +190 -0
  837. package/dist/timeline/post-tool-summary.js.map +1 -0
  838. package/dist/timeline/recorder.d.ts +36 -0
  839. package/dist/timeline/recorder.d.ts.map +1 -0
  840. package/dist/timeline/recorder.js +42 -0
  841. package/dist/timeline/recorder.js.map +1 -0
  842. package/dist/timeline/schema-bootstrap.d.ts +42 -0
  843. package/dist/timeline/schema-bootstrap.d.ts.map +1 -0
  844. package/dist/timeline/schema-bootstrap.js +81 -0
  845. package/dist/timeline/schema-bootstrap.js.map +1 -0
  846. package/dist/timeline/schema.sql +37 -0
  847. package/dist/timeline/store.d.ts +69 -0
  848. package/dist/timeline/store.d.ts.map +1 -0
  849. package/dist/timeline/store.js +429 -0
  850. package/dist/timeline/store.js.map +1 -0
  851. package/dist/timeline/types.d.ts +78 -0
  852. package/dist/timeline/types.d.ts.map +1 -0
  853. package/dist/timeline/types.js +9 -0
  854. package/dist/timeline/types.js.map +1 -0
  855. package/dist/types.d.ts +392 -0
  856. package/dist/types.d.ts.map +1 -0
  857. package/dist/types.js +81 -0
  858. package/dist/types.js.map +1 -0
  859. package/dist/ui/glyphs.d.ts +42 -0
  860. package/dist/ui/glyphs.d.ts.map +1 -0
  861. package/dist/ui/glyphs.js +78 -0
  862. package/dist/ui/glyphs.js.map +1 -0
  863. package/dist/ui/shimmer-progress.d.ts +11 -0
  864. package/dist/ui/shimmer-progress.d.ts.map +1 -0
  865. package/dist/ui/shimmer-progress.js +90 -0
  866. package/dist/ui/shimmer-progress.js.map +1 -0
  867. package/dist/ui/shimmer-worker.d.ts +2 -0
  868. package/dist/ui/shimmer-worker.d.ts.map +1 -0
  869. package/dist/ui/shimmer-worker.js +118 -0
  870. package/dist/ui/shimmer-worker.js.map +1 -0
  871. package/dist/ui/types.d.ts +17 -0
  872. package/dist/ui/types.d.ts.map +1 -0
  873. package/dist/ui/types.js +3 -0
  874. package/dist/ui/types.js.map +1 -0
  875. package/dist/upgrade/index.d.ts +132 -0
  876. package/dist/upgrade/index.d.ts.map +1 -0
  877. package/dist/upgrade/index.js +498 -0
  878. package/dist/upgrade/index.js.map +1 -0
  879. package/dist/utils.d.ts +224 -0
  880. package/dist/utils.d.ts.map +1 -0
  881. package/dist/utils.js +583 -0
  882. package/dist/utils.js.map +1 -0
  883. package/package.json +60 -0
  884. package/scripts/add-lang/bench.sh +60 -0
  885. package/scripts/add-lang/check-grammar.mjs +75 -0
  886. package/scripts/add-lang/dump-ast.mjs +103 -0
  887. package/scripts/add-lang/verify-extraction.mjs +70 -0
  888. package/scripts/agent-eval/ab-adoption.sh +91 -0
  889. package/scripts/agent-eval/ab-hook.sh +86 -0
  890. package/scripts/agent-eval/ab-impl.sh +78 -0
  891. package/scripts/agent-eval/ab-new-vs-baseline.sh +102 -0
  892. package/scripts/agent-eval/ab-sufficiency.sh +78 -0
  893. package/scripts/agent-eval/arms-F.sh +21 -0
  894. package/scripts/agent-eval/arms-matrix.sh +37 -0
  895. package/scripts/agent-eval/audit.sh +68 -0
  896. package/scripts/agent-eval/bench-readme.sh +28 -0
  897. package/scripts/agent-eval/bench-why-repo.sh +22 -0
  898. package/scripts/agent-eval/block-read-hook.sh +19 -0
  899. package/scripts/agent-eval/hook-settings.json +15 -0
  900. package/scripts/agent-eval/itrun.sh +120 -0
  901. package/scripts/agent-eval/offload-eval-3arm.sh +72 -0
  902. package/scripts/agent-eval/offload-eval-cost.mjs +133 -0
  903. package/scripts/agent-eval/offload-eval-effort.mjs +108 -0
  904. package/scripts/agent-eval/offload-eval-frontload-matrix.sh +25 -0
  905. package/scripts/agent-eval/offload-eval-frontload.sh +47 -0
  906. package/scripts/agent-eval/offload-eval-ground-truth.json +18 -0
  907. package/scripts/agent-eval/offload-eval-hook.mjs +84 -0
  908. package/scripts/agent-eval/offload-eval-judge.mjs +103 -0
  909. package/scripts/agent-eval/offload-eval-matrix.sh +20 -0
  910. package/scripts/agent-eval/offload-eval-metrics.mjs +94 -0
  911. package/scripts/agent-eval/offload-eval-refs1.sh +50 -0
  912. package/scripts/agent-eval/offload-eval-setup.sh +24 -0
  913. package/scripts/agent-eval/offload-eval-styles.sh +72 -0
  914. package/scripts/agent-eval/offload-eval-summarize.mjs +68 -0
  915. package/scripts/agent-eval/offload-eval.md +76 -0
  916. package/scripts/agent-eval/parse-arms.mjs +116 -0
  917. package/scripts/agent-eval/parse-bench-readme.mjs +84 -0
  918. package/scripts/agent-eval/parse-run.mjs +45 -0
  919. package/scripts/agent-eval/parse-session.mjs +93 -0
  920. package/scripts/agent-eval/probe-context.mjs +21 -0
  921. package/scripts/agent-eval/probe-explore.mjs +40 -0
  922. package/scripts/agent-eval/probe-node.mjs +20 -0
  923. package/scripts/agent-eval/probe-sweep.mjs +119 -0
  924. package/scripts/agent-eval/probe-trace.mjs +20 -0
  925. package/scripts/agent-eval/redirect-read-hook.sh +38 -0
  926. package/scripts/agent-eval/run-agent.sh +34 -0
  927. package/scripts/agent-eval/run-all.sh +69 -0
  928. package/scripts/agent-eval/run-arms.sh +56 -0
  929. package/scripts/agent-eval/seq-matrix.mjs +137 -0
  930. package/scripts/build-bundle.sh +118 -0
  931. package/scripts/extract-release-notes.mjs +130 -0
  932. package/scripts/local-install.sh +41 -0
  933. package/scripts/npm-sdk.js +75 -0
  934. package/scripts/npm-shim.js +246 -0
  935. package/scripts/pack-npm.sh +118 -0
  936. package/scripts/prepare-release.mjs +270 -0
@@ -0,0 +1,1591 @@
1
+ "use strict";
2
+ /**
3
+ * Database Queries
4
+ *
5
+ * Prepared statements for CRUD operations on the knowledge graph.
6
+ */
7
+ Object.defineProperty(exports, "__esModule", { value: true });
8
+ exports.QueryBuilder = void 0;
9
+ const utils_1 = require("../utils");
10
+ const query_utils_1 = require("../search/query-utils");
11
+ const query_parser_1 = require("../search/query-parser");
12
+ const generated_detection_1 = require("../extraction/generated-detection");
13
+ /**
14
+ * Path-only heuristic for files that should not be candidates for
15
+ * "dominant file" detection: test/spec files and tool-generated files.
16
+ * Generated files (`*.pb.go`, `*.pulsar.go`, mock outputs, …) often
17
+ * have huge in-file edge counts that dwarf the real source — etcd's
18
+ * `rpc.pb.go` has 4× the in-file edges of `server.go`.
19
+ */
20
+ function isLowValueFile(filePath) {
21
+ const lp = filePath.toLowerCase();
22
+ return (/(?:^|\/)(tests?|__tests?__|spec)\//.test(lp) ||
23
+ /_test\.go$/.test(lp) ||
24
+ /(?:^|\/)test_[^/]+\.py$/.test(lp) ||
25
+ /_test\.py$/.test(lp) ||
26
+ /_spec\.rb$/.test(lp) ||
27
+ /_test\.rb$/.test(lp) ||
28
+ /\.(test|spec)\.[jt]sx?$/.test(lp) ||
29
+ /(test|spec|tests)\.(java|kt|scala)$/.test(lp) ||
30
+ /(tests?|spec)\.cs$/.test(lp) ||
31
+ /tests?\.swift$/.test(lp) ||
32
+ /_test\.dart$/.test(lp) ||
33
+ (0, generated_detection_1.isGeneratedFile)(filePath));
34
+ }
35
+ const SQLITE_PARAM_CHUNK_SIZE = 500;
36
+ /**
37
+ * Convert database row to Node object
38
+ */
39
+ function rowToNode(row) {
40
+ return {
41
+ id: row.id,
42
+ kind: row.kind,
43
+ name: row.name,
44
+ qualifiedName: row.qualified_name,
45
+ filePath: row.file_path,
46
+ language: row.language,
47
+ startLine: row.start_line,
48
+ endLine: row.end_line,
49
+ startColumn: row.start_column,
50
+ endColumn: row.end_column,
51
+ docstring: row.docstring ?? undefined,
52
+ signature: row.signature ?? undefined,
53
+ visibility: row.visibility,
54
+ isExported: row.is_exported === 1,
55
+ isAsync: row.is_async === 1,
56
+ isStatic: row.is_static === 1,
57
+ isAbstract: row.is_abstract === 1,
58
+ decorators: row.decorators ? (0, utils_1.safeJsonParse)(row.decorators, undefined) : undefined,
59
+ typeParameters: row.type_parameters ? (0, utils_1.safeJsonParse)(row.type_parameters, undefined) : undefined,
60
+ returnType: row.return_type ?? undefined,
61
+ updatedAt: row.updated_at,
62
+ };
63
+ }
64
+ /**
65
+ * Convert database row to Edge object
66
+ */
67
+ function rowToEdge(row) {
68
+ return {
69
+ source: row.source,
70
+ target: row.target,
71
+ kind: row.kind,
72
+ metadata: row.metadata ? (0, utils_1.safeJsonParse)(row.metadata, undefined) : undefined,
73
+ line: row.line ?? undefined,
74
+ column: row.col ?? undefined,
75
+ provenance: row.provenance,
76
+ };
77
+ }
78
+ /**
79
+ * Convert database row to FileRecord object
80
+ */
81
+ function rowToFileRecord(row) {
82
+ return {
83
+ path: row.path,
84
+ contentHash: row.content_hash,
85
+ language: row.language,
86
+ size: row.size,
87
+ modifiedAt: row.modified_at,
88
+ indexedAt: row.indexed_at,
89
+ nodeCount: row.node_count,
90
+ errors: row.errors ? (0, utils_1.safeJsonParse)(row.errors, undefined) : undefined,
91
+ };
92
+ }
93
+ /**
94
+ * Query builder for the knowledge graph database
95
+ */
96
+ class QueryBuilder {
97
+ db;
98
+ // Project-name tokens (go.mod / package.json / repo dir), normalized. A query
99
+ // word matching one is dropped from path-relevance scoring — it names the
100
+ // whole project, not a symbol, so it carries no discriminative signal (#720).
101
+ // Set once by the CodeGraph instance; empty by default (no down-weighting).
102
+ projectNameTokens = new Set();
103
+ // Node cache for frequently accessed nodes (LRU-style, max 1000 entries)
104
+ nodeCache = new Map();
105
+ maxCacheSize = 1000;
106
+ // Prepared statements (lazily initialized)
107
+ stmts = {};
108
+ constructor(db) {
109
+ this.db = db;
110
+ }
111
+ /** Set the normalized project-name tokens used to down-weight non-discriminative
112
+ * query words in path scoring (#720). Called once when the project opens. */
113
+ setProjectNameTokens(tokens) {
114
+ this.projectNameTokens = tokens;
115
+ }
116
+ /** The normalized project-name tokens (#720); empty if none were derived. */
117
+ getProjectNameTokens() {
118
+ return this.projectNameTokens;
119
+ }
120
+ // ===========================================================================
121
+ // Node Operations
122
+ // ===========================================================================
123
+ /**
124
+ * Insert a new node
125
+ */
126
+ insertNode(node) {
127
+ if (!this.stmts.insertNode) {
128
+ this.stmts.insertNode = this.db.prepare(`
129
+ INSERT OR REPLACE INTO nodes (
130
+ id, kind, name, qualified_name, file_path, language,
131
+ start_line, end_line, start_column, end_column,
132
+ docstring, signature, visibility,
133
+ is_exported, is_async, is_static, is_abstract,
134
+ decorators, type_parameters, return_type, updated_at
135
+ ) VALUES (
136
+ @id, @kind, @name, @qualifiedName, @filePath, @language,
137
+ @startLine, @endLine, @startColumn, @endColumn,
138
+ @docstring, @signature, @visibility,
139
+ @isExported, @isAsync, @isStatic, @isAbstract,
140
+ @decorators, @typeParameters, @returnType, @updatedAt
141
+ )
142
+ `);
143
+ }
144
+ // Validate required fields to prevent SQLite bind errors
145
+ if (!node.id || !node.kind || !node.name || !node.filePath || !node.language) {
146
+ console.error('[cgraphx] Skipping node with missing required fields:', {
147
+ id: node.id,
148
+ kind: node.kind,
149
+ name: node.name,
150
+ filePath: node.filePath,
151
+ language: node.language,
152
+ });
153
+ return;
154
+ }
155
+ // INSERT OR REPLACE may overwrite a node we have cached. Drop the
156
+ // stale entry so the next getNodeById sees the new row, not the old
157
+ // one (matches the cache-invalidation pattern used by updateNode and
158
+ // deleteNode below).
159
+ this.nodeCache.delete(node.id);
160
+ this.stmts.insertNode.run({
161
+ id: node.id,
162
+ kind: node.kind,
163
+ name: node.name,
164
+ qualifiedName: node.qualifiedName ?? node.name,
165
+ filePath: node.filePath,
166
+ language: node.language,
167
+ startLine: node.startLine ?? 0,
168
+ endLine: node.endLine ?? 0,
169
+ startColumn: node.startColumn ?? 0,
170
+ endColumn: node.endColumn ?? 0,
171
+ docstring: node.docstring ?? null,
172
+ signature: node.signature ?? null,
173
+ visibility: node.visibility ?? null,
174
+ isExported: node.isExported ? 1 : 0,
175
+ isAsync: node.isAsync ? 1 : 0,
176
+ isStatic: node.isStatic ? 1 : 0,
177
+ isAbstract: node.isAbstract ? 1 : 0,
178
+ decorators: node.decorators ? JSON.stringify(node.decorators) : null,
179
+ typeParameters: node.typeParameters ? JSON.stringify(node.typeParameters) : null,
180
+ returnType: node.returnType ?? null,
181
+ updatedAt: node.updatedAt ?? Date.now(),
182
+ });
183
+ }
184
+ /**
185
+ * Insert multiple nodes in a transaction
186
+ */
187
+ insertNodes(nodes) {
188
+ this.db.transaction(() => {
189
+ for (const node of nodes) {
190
+ this.insertNode(node);
191
+ }
192
+ })();
193
+ }
194
+ /**
195
+ * Update an existing node
196
+ */
197
+ updateNode(node) {
198
+ if (!this.stmts.updateNode) {
199
+ this.stmts.updateNode = this.db.prepare(`
200
+ UPDATE nodes SET
201
+ kind = @kind,
202
+ name = @name,
203
+ qualified_name = @qualifiedName,
204
+ file_path = @filePath,
205
+ language = @language,
206
+ start_line = @startLine,
207
+ end_line = @endLine,
208
+ start_column = @startColumn,
209
+ end_column = @endColumn,
210
+ docstring = @docstring,
211
+ signature = @signature,
212
+ visibility = @visibility,
213
+ is_exported = @isExported,
214
+ is_async = @isAsync,
215
+ is_static = @isStatic,
216
+ is_abstract = @isAbstract,
217
+ decorators = @decorators,
218
+ type_parameters = @typeParameters,
219
+ return_type = @returnType,
220
+ updated_at = @updatedAt
221
+ WHERE id = @id
222
+ `);
223
+ }
224
+ // Invalidate cache before update
225
+ this.nodeCache.delete(node.id);
226
+ // Validate required fields
227
+ if (!node.id || !node.kind || !node.name || !node.filePath || !node.language) {
228
+ console.error('[cgraphx] Skipping node update with missing required fields:', node.id);
229
+ return;
230
+ }
231
+ this.stmts.updateNode.run({
232
+ id: node.id,
233
+ kind: node.kind,
234
+ name: node.name,
235
+ qualifiedName: node.qualifiedName ?? node.name,
236
+ filePath: node.filePath,
237
+ language: node.language,
238
+ startLine: node.startLine ?? 0,
239
+ endLine: node.endLine ?? 0,
240
+ startColumn: node.startColumn ?? 0,
241
+ endColumn: node.endColumn ?? 0,
242
+ docstring: node.docstring ?? null,
243
+ signature: node.signature ?? null,
244
+ visibility: node.visibility ?? null,
245
+ isExported: node.isExported ? 1 : 0,
246
+ isAsync: node.isAsync ? 1 : 0,
247
+ isStatic: node.isStatic ? 1 : 0,
248
+ isAbstract: node.isAbstract ? 1 : 0,
249
+ decorators: node.decorators ? JSON.stringify(node.decorators) : null,
250
+ typeParameters: node.typeParameters ? JSON.stringify(node.typeParameters) : null,
251
+ returnType: node.returnType ?? null,
252
+ updatedAt: node.updatedAt ?? Date.now(),
253
+ });
254
+ }
255
+ /**
256
+ * Delete a node by ID
257
+ */
258
+ deleteNode(id) {
259
+ if (!this.stmts.deleteNode) {
260
+ this.stmts.deleteNode = this.db.prepare('DELETE FROM nodes WHERE id = ?');
261
+ }
262
+ // Invalidate cache
263
+ this.nodeCache.delete(id);
264
+ this.stmts.deleteNode.run(id);
265
+ }
266
+ /**
267
+ * Delete all nodes for a file
268
+ */
269
+ deleteNodesByFile(filePath) {
270
+ if (!this.stmts.deleteNodesByFile) {
271
+ this.stmts.deleteNodesByFile = this.db.prepare('DELETE FROM nodes WHERE file_path = ?');
272
+ }
273
+ // Invalidate cache for nodes in this file
274
+ for (const [id, node] of this.nodeCache) {
275
+ if (node.filePath === filePath) {
276
+ this.nodeCache.delete(id);
277
+ }
278
+ }
279
+ this.stmts.deleteNodesByFile.run(filePath);
280
+ }
281
+ /**
282
+ * Get a node by ID
283
+ */
284
+ getNodeById(id) {
285
+ // Check cache first
286
+ if (this.nodeCache.has(id)) {
287
+ const cached = this.nodeCache.get(id);
288
+ // Move to end to implement LRU (delete and re-add)
289
+ this.nodeCache.delete(id);
290
+ this.nodeCache.set(id, cached);
291
+ return cached;
292
+ }
293
+ if (!this.stmts.getNodeById) {
294
+ this.stmts.getNodeById = this.db.prepare('SELECT * FROM nodes WHERE id = ?');
295
+ }
296
+ const row = this.stmts.getNodeById.get(id);
297
+ if (!row) {
298
+ return null;
299
+ }
300
+ const node = rowToNode(row);
301
+ this.cacheNode(node);
302
+ return node;
303
+ }
304
+ /**
305
+ * Batch lookup: fetch many nodes by ID in a single SQL round-trip.
306
+ *
307
+ * Replaces the N+1 pattern in graph traversal where every edge would
308
+ * trigger its own `getNodeById` call. For a function with 50 callers
309
+ * this collapses 50 point reads into one IN-list query (~10-50x
310
+ * faster end-to-end).
311
+ *
312
+ * Returns a Map keyed by id so callers can preserve their own ordering
313
+ * (typically the order edges were returned from the graph). Missing IDs
314
+ * are simply absent from the map.
315
+ *
316
+ * Cache-aware: ids already in the LRU cache are served from memory and
317
+ * the SQL query only touches the misses.
318
+ */
319
+ getNodesByIds(ids) {
320
+ const out = new Map();
321
+ if (ids.length === 0)
322
+ return out;
323
+ // Serve cache hits first; build the miss list for SQL.
324
+ const misses = [];
325
+ for (const id of ids) {
326
+ const cached = this.nodeCache.get(id);
327
+ if (cached !== undefined) {
328
+ // LRU touch
329
+ this.nodeCache.delete(id);
330
+ this.nodeCache.set(id, cached);
331
+ out.set(id, cached);
332
+ }
333
+ else {
334
+ misses.push(id);
335
+ }
336
+ }
337
+ if (misses.length === 0)
338
+ return out;
339
+ // Chunk under SQLite's parameter limit (default 999, raised to 32766
340
+ // in better-sqlite3 builds — chunk at 500 for safety across both
341
+ // backends and to keep the query plan simple).
342
+ for (let i = 0; i < misses.length; i += SQLITE_PARAM_CHUNK_SIZE) {
343
+ const chunk = misses.slice(i, i + SQLITE_PARAM_CHUNK_SIZE);
344
+ const placeholders = chunk.map(() => '?').join(',');
345
+ const rows = this.db
346
+ .prepare(`SELECT * FROM nodes WHERE id IN (${placeholders})`)
347
+ .all(...chunk);
348
+ for (const row of rows) {
349
+ const node = rowToNode(row);
350
+ out.set(node.id, node);
351
+ this.cacheNode(node);
352
+ }
353
+ }
354
+ return out;
355
+ }
356
+ getExistingNodeIds(ids) {
357
+ const out = new Set();
358
+ if (ids.length === 0)
359
+ return out;
360
+ const uniqueIds = [...new Set(ids)];
361
+ for (let i = 0; i < uniqueIds.length; i += SQLITE_PARAM_CHUNK_SIZE) {
362
+ const chunk = uniqueIds.slice(i, i + SQLITE_PARAM_CHUNK_SIZE);
363
+ const placeholders = chunk.map(() => '?').join(',');
364
+ const rows = this.db
365
+ .prepare(`SELECT id FROM nodes WHERE id IN (${placeholders})`)
366
+ .all(...chunk);
367
+ for (const row of rows) {
368
+ out.add(row.id);
369
+ }
370
+ }
371
+ return out;
372
+ }
373
+ /**
374
+ * Add a node to the cache, evicting oldest if needed
375
+ */
376
+ cacheNode(node) {
377
+ if (this.nodeCache.size >= this.maxCacheSize) {
378
+ // Evict oldest (first) entry
379
+ const firstKey = this.nodeCache.keys().next().value;
380
+ if (firstKey) {
381
+ this.nodeCache.delete(firstKey);
382
+ }
383
+ }
384
+ this.nodeCache.set(node.id, node);
385
+ }
386
+ /**
387
+ * Clear the node cache
388
+ */
389
+ clearCache() {
390
+ this.nodeCache.clear();
391
+ }
392
+ /**
393
+ * Get all nodes in a file
394
+ */
395
+ getNodesByFile(filePath) {
396
+ if (!this.stmts.getNodesByFile) {
397
+ this.stmts.getNodesByFile = this.db.prepare('SELECT * FROM nodes WHERE file_path = ? ORDER BY start_line');
398
+ }
399
+ const rows = this.stmts.getNodesByFile.all(filePath);
400
+ return rows.map(rowToNode);
401
+ }
402
+ /**
403
+ * Find the file that holds the densest concentration of the project's
404
+ * internal call graph — the "core" file. Used by context-builder to
405
+ * boost ranking of symbols in that file's directory (so e.g. sinatra
406
+ * queries surface `lib/sinatra/base.rb`'s `route!` instead of
407
+ * `sinatra-contrib/lib/sinatra/multi_route.rb`'s `route` extension).
408
+ *
409
+ * Returns null if no file has a meaningful concentration (e.g. spread
410
+ * evenly across many files, or empty index).
411
+ *
412
+ * "Internal" = source and target are in the same file. Cross-file
413
+ * edges aren't useful here — they don't tell us which file is the
414
+ * functional center.
415
+ *
416
+ * Excludes test/spec files from candidacy via path-pattern. The agent's
417
+ * typical question is "how does X work", not "how is X tested", so
418
+ * boosting a test file's directory would be a misfire.
419
+ */
420
+ getDominantFile() {
421
+ if (!this.stmts.getDominantFile) {
422
+ // Pull top 20 candidates; we then filter out test/generated files
423
+ // in code (regex-grade matching that SQL LIKE can't express). The
424
+ // generated-file filter is critical — without it, etcd's
425
+ // `api/etcdserverpb/rpc.pb.go` (1916 in-file edges, generated
426
+ // protobuf stub) outranks the real `server/etcdserver/server.go`
427
+ // (470 edges) by 4×, and the boost would push the agent toward
428
+ // generated code.
429
+ this.stmts.getDominantFile = this.db.prepare(`
430
+ SELECT n.file_path AS file_path, COUNT(*) AS edge_count
431
+ FROM edges e
432
+ JOIN nodes n ON e.source = n.id
433
+ JOIN nodes m ON e.target = m.id
434
+ WHERE n.file_path = m.file_path
435
+ GROUP BY n.file_path
436
+ ORDER BY edge_count DESC
437
+ LIMIT 20
438
+ `);
439
+ }
440
+ const rows = this.stmts.getDominantFile.all();
441
+ const filtered = rows.filter(r => !isLowValueFile(r.file_path));
442
+ if (filtered.length === 0 || filtered[0].edge_count < 20)
443
+ return null;
444
+ return {
445
+ filePath: filtered[0].file_path,
446
+ edgeCount: filtered[0].edge_count,
447
+ nextEdgeCount: filtered[1]?.edge_count ?? 0,
448
+ };
449
+ }
450
+ /**
451
+ * Find the file that holds the densest concentration of the project's
452
+ * `route` nodes (framework-emitted: Express/Gin/Flask/Rails/Drupal/etc.).
453
+ * Used by handleContext on small repos to inline the project's routing
454
+ * config when the agent's query is about request flow — eliminating the
455
+ * "Glob + Read routes.rb" pattern that beats cgraphx on tiny realworld
456
+ * template repos.
457
+ *
458
+ * Excludes test/generated files from candidacy. Returns null if there
459
+ * are fewer than 3 non-test routes total, or if no file holds at least
460
+ * 30% of them (diffuse routing → no single answer file).
461
+ */
462
+ getTopRouteFile() {
463
+ if (!this.stmts.getTopRouteFile) {
464
+ this.stmts.getTopRouteFile = this.db.prepare(`
465
+ SELECT file_path, COUNT(*) AS cnt
466
+ FROM nodes
467
+ WHERE kind = 'route'
468
+ GROUP BY file_path
469
+ ORDER BY cnt DESC
470
+ LIMIT 20
471
+ `);
472
+ }
473
+ const rows = this.stmts.getTopRouteFile.all();
474
+ const filtered = rows.filter(r => !isLowValueFile(r.file_path));
475
+ if (filtered.length === 0)
476
+ return null;
477
+ const totalRoutes = filtered.reduce((sum, r) => sum + r.cnt, 0);
478
+ const top = filtered[0];
479
+ if (totalRoutes < 3 || top.cnt < 3)
480
+ return null;
481
+ if (top.cnt / totalRoutes < 0.30)
482
+ return null;
483
+ return { filePath: top.file_path, routeCount: top.cnt, totalRoutes };
484
+ }
485
+ /**
486
+ * Build a URL → handler manifest from the index. Each route node's
487
+ * `references` edge points at the function/method that handles the
488
+ * request. We join them in one pass; the agent gets the canonical
489
+ * routing answer ("POST /users/login → AuthController#login") without
490
+ * having to parse the framework's route DSL itself.
491
+ *
492
+ * Also returns the file with the most handler endpoints — used as the
493
+ * "top handler file" to inline source for, so the agent has both the
494
+ * mapping AND the handler implementations.
495
+ */
496
+ getRoutingManifest(limit = 40) {
497
+ if (!this.stmts.getRoutingManifest) {
498
+ // Edge kind varies across framework resolvers: Spring/Rails/
499
+ // Laravel/Drupal emit `references`, Express emits `calls`. Accept
500
+ // both — the semantic is the same (route → its handler).
501
+ this.stmts.getRoutingManifest = this.db.prepare(`
502
+ SELECT
503
+ r.name AS url,
504
+ h.name AS handler,
505
+ h.file_path AS handler_file,
506
+ h.start_line AS handler_line,
507
+ h.kind AS handler_kind
508
+ FROM nodes r
509
+ JOIN edges e ON e.source = r.id
510
+ JOIN nodes h ON e.target = h.id
511
+ WHERE r.kind = 'route'
512
+ AND e.kind IN ('references', 'calls')
513
+ AND h.kind IN ('function', 'method', 'class')
514
+ ORDER BY r.file_path, r.start_line
515
+ LIMIT ?
516
+ `);
517
+ }
518
+ const rows = this.stmts.getRoutingManifest.all(limit);
519
+ // Drop test/generated handlers — same hygiene as elsewhere.
520
+ const filtered = rows.filter(r => !isLowValueFile(r.handler_file));
521
+ if (filtered.length < 3)
522
+ return null;
523
+ // Identify the file holding the most handlers (the "primary handler file").
524
+ const fileCounts = new Map();
525
+ for (const r of filtered) {
526
+ fileCounts.set(r.handler_file, (fileCounts.get(r.handler_file) ?? 0) + 1);
527
+ }
528
+ let topHandlerFile = null;
529
+ let topHandlerFileCount = 0;
530
+ for (const [file, count] of fileCounts) {
531
+ if (count > topHandlerFileCount) {
532
+ topHandlerFile = file;
533
+ topHandlerFileCount = count;
534
+ }
535
+ }
536
+ return {
537
+ entries: filtered.map(r => ({
538
+ url: r.url,
539
+ handler: r.handler,
540
+ handlerFile: r.handler_file,
541
+ handlerLine: r.handler_line,
542
+ handlerKind: r.handler_kind,
543
+ })),
544
+ topHandlerFile,
545
+ topHandlerFileCount,
546
+ totalRoutes: filtered.length,
547
+ };
548
+ }
549
+ /**
550
+ * Get all nodes of a specific kind
551
+ */
552
+ getNodesByKind(kind) {
553
+ if (!this.stmts.getNodesByKind) {
554
+ this.stmts.getNodesByKind = this.db.prepare('SELECT * FROM nodes WHERE kind = ?');
555
+ }
556
+ const rows = this.stmts.getNodesByKind.all(kind);
557
+ return rows.map(rowToNode);
558
+ }
559
+ /**
560
+ * Stream every node of a kind one at a time (lazy) instead of materializing
561
+ * them all like {@link getNodesByKind}. For unbounded kinds (`function`,
562
+ * `method`) on a symbol-dense project the full array is gigabytes; the
563
+ * dynamic-edge synthesizers only scan-and-filter, so they iterate to keep
564
+ * memory O(1) in the node count rather than O(nodes) (#610).
565
+ */
566
+ *iterateNodesByKind(kind) {
567
+ // Fresh statement per call (not a cached one): an iterator holds an open
568
+ // cursor, so a shared statement would conflict across overlapping scans.
569
+ const stmt = this.db.prepare('SELECT * FROM nodes WHERE kind = ?');
570
+ for (const row of stmt.iterate(kind)) {
571
+ yield rowToNode(row);
572
+ }
573
+ }
574
+ /**
575
+ * Get all nodes in the database
576
+ */
577
+ getAllNodes() {
578
+ const rows = this.db.prepare('SELECT * FROM nodes').all();
579
+ return rows.map(rowToNode);
580
+ }
581
+ /**
582
+ * Get nodes by exact name match (uses idx_nodes_name index)
583
+ */
584
+ getNodesByName(name) {
585
+ if (!this.stmts.getNodesByName) {
586
+ this.stmts.getNodesByName = this.db.prepare('SELECT * FROM nodes WHERE name = ?');
587
+ }
588
+ const rows = this.stmts.getNodesByName.all(name);
589
+ return rows.map(rowToNode);
590
+ }
591
+ /**
592
+ * Get nodes by exact qualified name match (uses idx_nodes_qualified_name index)
593
+ */
594
+ getNodesByQualifiedNameExact(qualifiedName) {
595
+ if (!this.stmts.getNodesByQualifiedNameExact) {
596
+ this.stmts.getNodesByQualifiedNameExact = this.db.prepare('SELECT * FROM nodes WHERE qualified_name = ?');
597
+ }
598
+ const rows = this.stmts.getNodesByQualifiedNameExact.all(qualifiedName);
599
+ return rows.map(rowToNode);
600
+ }
601
+ /**
602
+ * Get nodes by lowercase name match (uses idx_nodes_lower_name expression index)
603
+ */
604
+ getNodesByLowerName(lowerName) {
605
+ if (!this.stmts.getNodesByLowerName) {
606
+ this.stmts.getNodesByLowerName = this.db.prepare('SELECT * FROM nodes WHERE lower(name) = ?');
607
+ }
608
+ const rows = this.stmts.getNodesByLowerName.all(lowerName);
609
+ return rows.map(rowToNode);
610
+ }
611
+ /**
612
+ * Search nodes by name using FTS with fallback to LIKE for better matching
613
+ *
614
+ * Search strategy:
615
+ * 1. Try FTS5 prefix match (query*) for word-start matching
616
+ * 2. If no results, try LIKE for substring matching (e.g., "signIn" finds "signInWithGoogle")
617
+ * 3. Score results based on match quality
618
+ */
619
+ searchNodes(query, options = {}) {
620
+ const { limit = 100, offset = 0 } = options;
621
+ // Parse field-qualified bits out of the raw query (kind:, lang:,
622
+ // path:, name:). Anything not recognised stays in `text` and goes
623
+ // to FTS unchanged. Filters compose with the SearchOptions arg —
624
+ // both are applied (intersection-style).
625
+ const parsed = (0, query_parser_1.parseQuery)(query);
626
+ const mergedKinds = parsed.kinds.length > 0
627
+ ? Array.from(new Set([...(options.kinds ?? []), ...parsed.kinds]))
628
+ : options.kinds;
629
+ const mergedLanguages = parsed.languages.length > 0
630
+ ? Array.from(new Set([...(options.languages ?? []), ...parsed.languages]))
631
+ : options.languages;
632
+ const pathFilters = parsed.pathFilters;
633
+ const nameFilters = parsed.nameFilters;
634
+ // The text portion drives FTS/LIKE; if all the user typed was
635
+ // filters (`kind:function`), we still need *some* candidate set,
636
+ // so synthesise an empty-text path that returns everything matching
637
+ // the filters.
638
+ const text = parsed.text;
639
+ const kinds = mergedKinds;
640
+ const languages = mergedLanguages;
641
+ // First try FTS5 with prefix matching
642
+ let results = text
643
+ ? this.searchNodesFTS(text, { kinds, languages, limit, offset })
644
+ // Over-fetch by 5× when running filter-only (no text). The
645
+ // post-scoring path: + name: filters can be very selective, so
646
+ // a smaller multiplier risks returning fewer than `limit`
647
+ // results despite the DB having plenty of matches.
648
+ : this.searchAllByFilters({ kinds, languages, limit: limit * 5 });
649
+ // If no FTS results, try LIKE-based substring search
650
+ if (results.length === 0 && text.length >= 2) {
651
+ results = this.searchNodesLike(text, { kinds, languages, limit, offset });
652
+ }
653
+ // Final fuzzy fallback: scan all known names and keep those within
654
+ // a tight Levenshtein distance. Only fires when both FTS and LIKE
655
+ // returned nothing AND there's a text portion long enough to be
656
+ // worth fuzzing (1-char queries would match too much).
657
+ if (results.length === 0 && text.length >= 3) {
658
+ results = this.searchNodesFuzzy(text, { kinds, languages, limit });
659
+ }
660
+ // Supplement: ensure exact name matches are always candidates.
661
+ // BM25 can bury short exact-match names (e.g. "getBean") under hundreds of
662
+ // compound names (e.g. "getBeanDescriptor") in large codebases,
663
+ // pushing them past the FTS fetch limit before post-hoc scoring can help.
664
+ // Use the max BM25 score as the base so the nameMatchBonus (exact=30 vs
665
+ // prefix=20) actually differentiates them after rescoring.
666
+ if (results.length > 0 && query) {
667
+ const existingIds = new Set(results.map(r => r.node.id));
668
+ const maxFtsScore = Math.max(...results.map(r => r.score));
669
+ const terms = query.split(/\s+/).filter(t => t.length >= 2);
670
+ for (const term of terms) {
671
+ let sql = 'SELECT * FROM nodes WHERE name = ? COLLATE NOCASE';
672
+ const params = [term];
673
+ if (kinds && kinds.length > 0) {
674
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
675
+ params.push(...kinds);
676
+ }
677
+ if (languages && languages.length > 0) {
678
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
679
+ params.push(...languages);
680
+ }
681
+ sql += ' LIMIT 20';
682
+ const rows = this.db.prepare(sql).all(...params);
683
+ for (const row of rows) {
684
+ if (!existingIds.has(row.id)) {
685
+ results.push({ node: rowToNode(row), score: maxFtsScore });
686
+ existingIds.add(row.id);
687
+ }
688
+ }
689
+ }
690
+ }
691
+ // Apply multi-signal scoring
692
+ if (results.length > 0 && (text || query)) {
693
+ const scoringQuery = text || query;
694
+ results = results.map(r => ({
695
+ ...r,
696
+ score: r.score
697
+ + (0, query_utils_1.kindBonus)(r.node.kind)
698
+ + (0, query_utils_1.scorePathRelevance)(r.node.filePath, scoringQuery, this.projectNameTokens)
699
+ + (0, query_utils_1.nameMatchBonus)(r.node.name, scoringQuery),
700
+ }));
701
+ results.sort((a, b) => b.score - a.score);
702
+ // Trim to requested limit after rescoring
703
+ if (results.length > limit) {
704
+ results = results.slice(0, limit);
705
+ }
706
+ }
707
+ // Apply path: + name: filters AFTER scoring. Scoring already uses
708
+ // path/name as a soft signal; the explicit filters here are a hard
709
+ // gate. Done last so the FTS limit fetched plenty of candidates to
710
+ // narrow from.
711
+ if (pathFilters.length > 0) {
712
+ const lowered = pathFilters.map((p) => p.toLowerCase());
713
+ results = results.filter((r) => {
714
+ const fp = r.node.filePath.toLowerCase();
715
+ return lowered.some((p) => fp.includes(p));
716
+ });
717
+ }
718
+ if (nameFilters.length > 0) {
719
+ const lowered = nameFilters.map((n) => n.toLowerCase());
720
+ results = results.filter((r) => {
721
+ const nm = r.node.name.toLowerCase();
722
+ return lowered.some((n) => nm.includes(n));
723
+ });
724
+ }
725
+ return results;
726
+ }
727
+ /**
728
+ * Match-everything path used when the user supplied only field
729
+ * filters (`kind:function lang:typescript`) with no text. Returns
730
+ * candidates ordered by name; the caller's filter pass narrows to
731
+ * what was asked for.
732
+ */
733
+ searchAllByFilters(options) {
734
+ const { kinds, languages, limit } = options;
735
+ let sql = 'SELECT * FROM nodes WHERE 1=1';
736
+ const params = [];
737
+ if (kinds && kinds.length > 0) {
738
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
739
+ params.push(...kinds);
740
+ }
741
+ if (languages && languages.length > 0) {
742
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
743
+ params.push(...languages);
744
+ }
745
+ sql += ' ORDER BY name LIMIT ?';
746
+ params.push(limit);
747
+ const rows = this.db.prepare(sql).all(...params);
748
+ return rows.map((row) => ({ node: rowToNode(row), score: 1 }));
749
+ }
750
+ /**
751
+ * Fuzzy fallback: when zero FTS/LIKE hits, try an edit-distance
752
+ * sweep over the distinct symbol-name set. Caps `maxDist` at 2 so
753
+ * `getUssr` finds `getUser` but `process` doesn't match `prosody`.
754
+ * Bounded edit distance keeps each comparison cheap; the per-query
755
+ * scan is O(distinct-name-count) which is far smaller than total
756
+ * node count on any real codebase.
757
+ */
758
+ searchNodesFuzzy(text, options) {
759
+ const { kinds, languages, limit } = options;
760
+ const lowered = text.toLowerCase();
761
+ const maxDist = lowered.length <= 4 ? 1 : 2;
762
+ // Pull the distinct name list once. The set is cached on QueryBuilder
763
+ // by getAllNodeNames(); even on a 200k-node project the distinct
764
+ // name set is typically O(10k) because most names repeat. The
765
+ // candidate-cap below bounds memory regardless.
766
+ const allNames = this.getAllNodeNames();
767
+ const candidates = [];
768
+ for (const name of allNames) {
769
+ const dist = (0, query_parser_1.boundedEditDistance)(name.toLowerCase(), lowered, maxDist);
770
+ if (dist <= maxDist)
771
+ candidates.push({ name, dist });
772
+ }
773
+ candidates.sort((a, b) => a.dist - b.dist);
774
+ // Cap the per-name follow-up queries. Each survivor triggers a
775
+ // separate `SELECT * FROM nodes WHERE name = ?`; without this cap
776
+ // a project with many similar names (`getUser1`, `getUser2`...)
777
+ // could fan out far beyond `limit` queries before the inner-loop
778
+ // limit kicks in.
779
+ const FUZZY_FOLLOWUP_CAP = Math.max(limit * 2, 50);
780
+ const cappedCandidates = candidates.slice(0, FUZZY_FOLLOWUP_CAP);
781
+ const results = [];
782
+ const seen = new Set();
783
+ for (const c of cappedCandidates) {
784
+ if (results.length >= limit)
785
+ break;
786
+ let sql = 'SELECT * FROM nodes WHERE name = ?';
787
+ const params = [c.name];
788
+ if (kinds && kinds.length > 0) {
789
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
790
+ params.push(...kinds);
791
+ }
792
+ if (languages && languages.length > 0) {
793
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
794
+ params.push(...languages);
795
+ }
796
+ sql += ' LIMIT 5';
797
+ const rows = this.db.prepare(sql).all(...params);
798
+ for (const row of rows) {
799
+ if (seen.has(row.id))
800
+ continue;
801
+ seen.add(row.id);
802
+ // Lower the score for each edit step away from the query so
803
+ // exact-match fallbacks (dist 0) outrank dist-2 typos.
804
+ results.push({ node: rowToNode(row), score: 1 / (1 + c.dist) });
805
+ if (results.length >= limit)
806
+ break;
807
+ }
808
+ }
809
+ return results;
810
+ }
811
+ /**
812
+ * FTS5 search with prefix matching
813
+ */
814
+ searchNodesFTS(query, options) {
815
+ const { kinds, languages, limit = 100, offset = 0 } = options;
816
+ // Add prefix wildcard for better matching (e.g., "auth" matches "AuthService", "authenticate")
817
+ // Escape special FTS5 characters and add prefix wildcard.
818
+ //
819
+ // `::` is a qualifier separator in Rust/C++/Ruby, not a token char,
820
+ // so treat it as whitespace before the strip step. Otherwise queries
821
+ // like `stage_apply::run` collapse to `stage_applyrun` (the colons
822
+ // are stripped without splitting) and find nothing. See #173.
823
+ const ftsQuery = query
824
+ .replace(/::/g, ' ') // Rust/C++/Ruby qualifier separator
825
+ .replace(/['"*():^]/g, '') // Remove FTS5 special chars
826
+ .split(/\s+/)
827
+ .filter(term => term.length > 0)
828
+ // Strip FTS5 boolean operators to prevent query manipulation
829
+ .filter(term => !/^(AND|OR|NOT|NEAR)$/i.test(term))
830
+ .map(term => `"${term}"*`) // Prefix match each term
831
+ .join(' OR ');
832
+ if (!ftsQuery) {
833
+ return [];
834
+ }
835
+ // BM25 column weights: id=0, name=20, qualified_name=5, docstring=1, signature=2
836
+ // Heavy name weight ensures exact/prefix name matches rank above incidental
837
+ // mentions in long docstrings or qualified names of nested symbols.
838
+ // Fetch 5x requested limit so post-hoc rescoring (kindBonus, pathRelevance,
839
+ // nameMatchBonus) can promote results that BM25 alone undervalues.
840
+ const ftsLimit = Math.max(limit * 5, 100);
841
+ let sql = `
842
+ SELECT nodes.*, bm25(nodes_fts, 0, 20, 5, 1, 2) as score
843
+ FROM nodes_fts
844
+ JOIN nodes ON nodes_fts.id = nodes.id
845
+ WHERE nodes_fts MATCH ?
846
+ `;
847
+ const params = [ftsQuery];
848
+ if (kinds && kinds.length > 0) {
849
+ sql += ` AND nodes.kind IN (${kinds.map(() => '?').join(',')})`;
850
+ params.push(...kinds);
851
+ }
852
+ if (languages && languages.length > 0) {
853
+ sql += ` AND nodes.language IN (${languages.map(() => '?').join(',')})`;
854
+ params.push(...languages);
855
+ }
856
+ sql += ' ORDER BY score LIMIT ? OFFSET ?';
857
+ params.push(ftsLimit, offset);
858
+ try {
859
+ const rows = this.db.prepare(sql).all(...params);
860
+ return rows.map((row) => ({
861
+ node: rowToNode(row),
862
+ score: Math.abs(row.score), // bm25 returns negative scores
863
+ }));
864
+ }
865
+ catch {
866
+ // FTS query failed, return empty
867
+ return [];
868
+ }
869
+ }
870
+ /**
871
+ * LIKE-based substring search for cases where FTS doesn't match
872
+ * Useful for camelCase matching (e.g., "signIn" finds "signInWithGoogle")
873
+ */
874
+ searchNodesLike(query, options) {
875
+ const { kinds, languages, limit = 100, offset = 0 } = options;
876
+ let sql = `
877
+ SELECT nodes.*,
878
+ CASE
879
+ WHEN name = ? THEN 1.0
880
+ WHEN name LIKE ? THEN 0.9
881
+ WHEN name LIKE ? THEN 0.8
882
+ WHEN qualified_name LIKE ? THEN 0.7
883
+ ELSE 0.5
884
+ END as score
885
+ FROM nodes
886
+ WHERE (
887
+ name LIKE ? OR
888
+ qualified_name LIKE ? OR
889
+ name LIKE ?
890
+ )
891
+ `;
892
+ // Pattern variants for better matching
893
+ const exactMatch = query;
894
+ const startsWith = `${query}%`;
895
+ const contains = `%${query}%`;
896
+ const params = [
897
+ exactMatch, // Exact match score
898
+ startsWith, // Starts with score
899
+ contains, // Contains score
900
+ contains, // Qualified name score
901
+ contains, // WHERE: name contains
902
+ contains, // WHERE: qualified_name contains
903
+ startsWith, // WHERE: name starts with
904
+ ];
905
+ if (kinds && kinds.length > 0) {
906
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
907
+ params.push(...kinds);
908
+ }
909
+ if (languages && languages.length > 0) {
910
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
911
+ params.push(...languages);
912
+ }
913
+ sql += ' ORDER BY score DESC, length(name) ASC LIMIT ? OFFSET ?';
914
+ params.push(limit, offset);
915
+ const rows = this.db.prepare(sql).all(...params);
916
+ return rows.map((row) => ({
917
+ node: rowToNode(row),
918
+ score: row.score,
919
+ }));
920
+ }
921
+ /**
922
+ * Find nodes by exact name match
923
+ *
924
+ * Used for hybrid search - looks up symbols by exact name or case-insensitive match.
925
+ * Returns high-confidence matches for known symbol names extracted from query.
926
+ *
927
+ * @param names - Array of symbol names to look up
928
+ * @param options - Search options (kinds, languages, limit)
929
+ * @returns SearchResult array with exact matches scored at 1.0
930
+ */
931
+ findNodesByExactName(names, options = {}) {
932
+ if (names.length === 0)
933
+ return [];
934
+ const { kinds, languages, limit = 50 } = options;
935
+ // Two-pass approach to handle common names (e.g., "run" has 40+ matches):
936
+ // Pass 1: Find which files contain distinctive (rare) symbols from the query.
937
+ // Pass 2: Query each name, boosting results that co-locate with distinctive symbols.
938
+ // Pass 1: Find files containing each queried name, identify distinctive names
939
+ const nameToFiles = new Map();
940
+ for (const name of names) {
941
+ let sql = 'SELECT DISTINCT file_path FROM nodes WHERE name COLLATE NOCASE = ?';
942
+ const params = [name];
943
+ if (kinds && kinds.length > 0) {
944
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
945
+ params.push(...kinds);
946
+ }
947
+ sql += ' LIMIT 100';
948
+ const rows = this.db.prepare(sql).all(...params);
949
+ nameToFiles.set(name.toLowerCase(), new Set(rows.map(r => r.file_path)));
950
+ }
951
+ // Distinctive names are those with fewer than 10 file matches (e.g., "scrapeLoop" = 1 file)
952
+ const distinctiveFiles = new Set();
953
+ for (const [, files] of nameToFiles) {
954
+ if (files.size > 0 && files.size < 10) {
955
+ for (const f of files)
956
+ distinctiveFiles.add(f);
957
+ }
958
+ }
959
+ // Pass 2: Query each name with per-name limit, scoring by co-location
960
+ const perNameLimit = Math.max(8, Math.ceil(limit / names.length));
961
+ const allResults = [];
962
+ const seenIds = new Set();
963
+ for (const name of names) {
964
+ let sql = `
965
+ SELECT nodes.*, 1.0 as score
966
+ FROM nodes
967
+ WHERE name COLLATE NOCASE = ?
968
+ `;
969
+ const params = [name];
970
+ if (kinds && kinds.length > 0) {
971
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
972
+ params.push(...kinds);
973
+ }
974
+ if (languages && languages.length > 0) {
975
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
976
+ params.push(...languages);
977
+ }
978
+ // Fetch enough to find co-located results among common names
979
+ sql += ' LIMIT ?';
980
+ params.push(Math.max(perNameLimit * 3, 50));
981
+ const rows = this.db.prepare(sql).all(...params);
982
+ const nameResults = [];
983
+ for (const row of rows) {
984
+ const node = rowToNode(row);
985
+ if (seenIds.has(node.id))
986
+ continue;
987
+ // Boost results in files that also contain distinctive symbols
988
+ const coLocationBoost = distinctiveFiles.has(node.filePath) ? 20 : 0;
989
+ nameResults.push({ node, score: row.score + coLocationBoost });
990
+ }
991
+ // Sort by score (co-located first), take per-name limit
992
+ nameResults.sort((a, b) => b.score - a.score);
993
+ for (const r of nameResults.slice(0, perNameLimit)) {
994
+ seenIds.add(r.node.id);
995
+ allResults.push(r);
996
+ }
997
+ }
998
+ // Sort all results by score so co-located results bubble up
999
+ allResults.sort((a, b) => b.score - a.score);
1000
+ return allResults.slice(0, limit);
1001
+ }
1002
+ /**
1003
+ * Find nodes whose name contains a substring (LIKE-based).
1004
+ * Useful for CamelCase-part matching where FTS fails because
1005
+ * e.g. "TransportSearchAction" is one FTS token, not matchable by "Search"*.
1006
+ *
1007
+ * Results are ordered by name length (shorter = more likely to be the core type).
1008
+ */
1009
+ findNodesByNameSubstring(substring, options = {}) {
1010
+ const { kinds, languages, limit = 30, excludePrefix } = options;
1011
+ let sql = `
1012
+ SELECT nodes.*, 1.0 as score
1013
+ FROM nodes
1014
+ WHERE name LIKE ?
1015
+ `;
1016
+ const params = [`%${substring}%`];
1017
+ // Exclude prefix matches (handled by FTS-based prefix search in Step 2b)
1018
+ if (excludePrefix) {
1019
+ sql += ` AND name NOT LIKE ?`;
1020
+ params.push(`${substring}%`);
1021
+ }
1022
+ if (kinds && kinds.length > 0) {
1023
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
1024
+ params.push(...kinds);
1025
+ }
1026
+ if (languages && languages.length > 0) {
1027
+ sql += ` AND language IN (${languages.map(() => '?').join(',')})`;
1028
+ params.push(...languages);
1029
+ }
1030
+ sql += ' ORDER BY length(name) ASC LIMIT ?';
1031
+ params.push(limit);
1032
+ const rows = this.db.prepare(sql).all(...params);
1033
+ return rows.map((row) => ({
1034
+ node: rowToNode(row),
1035
+ score: row.score,
1036
+ }));
1037
+ }
1038
+ // ===========================================================================
1039
+ // Edge Operations
1040
+ // ===========================================================================
1041
+ /**
1042
+ * Insert a new edge
1043
+ */
1044
+ insertEdge(edge) {
1045
+ if (!this.stmts.insertEdge) {
1046
+ this.stmts.insertEdge = this.db.prepare(`
1047
+ INSERT OR IGNORE INTO edges (source, target, kind, metadata, line, col, provenance)
1048
+ VALUES (@source, @target, @kind, @metadata, @line, @col, @provenance)
1049
+ `);
1050
+ }
1051
+ this.stmts.insertEdge.run({
1052
+ source: edge.source,
1053
+ target: edge.target,
1054
+ kind: edge.kind,
1055
+ metadata: edge.metadata ? JSON.stringify(edge.metadata) : null,
1056
+ line: edge.line ?? null,
1057
+ col: edge.column ?? null,
1058
+ provenance: edge.provenance ?? null,
1059
+ });
1060
+ }
1061
+ /**
1062
+ * Insert multiple edges in a transaction
1063
+ */
1064
+ insertEdges(edges) {
1065
+ if (edges.length === 0)
1066
+ return;
1067
+ this.db.transaction(() => {
1068
+ const endpointIds = new Set();
1069
+ for (const edge of edges) {
1070
+ endpointIds.add(edge.source);
1071
+ endpointIds.add(edge.target);
1072
+ }
1073
+ const existingNodeIds = this.getExistingNodeIds([...endpointIds]);
1074
+ for (const edge of edges) {
1075
+ if (!existingNodeIds.has(edge.source) || !existingNodeIds.has(edge.target)) {
1076
+ continue;
1077
+ }
1078
+ this.insertEdge(edge);
1079
+ }
1080
+ })();
1081
+ }
1082
+ /**
1083
+ * Delete all edges from a source node
1084
+ */
1085
+ deleteEdgesBySource(sourceId) {
1086
+ if (!this.stmts.deleteEdgesBySource) {
1087
+ this.stmts.deleteEdgesBySource = this.db.prepare('DELETE FROM edges WHERE source = ?');
1088
+ }
1089
+ this.stmts.deleteEdgesBySource.run(sourceId);
1090
+ }
1091
+ /**
1092
+ * Get outgoing edges from a node
1093
+ */
1094
+ getOutgoingEdges(sourceId, kinds, provenance) {
1095
+ if ((kinds && kinds.length > 0) || provenance) {
1096
+ let sql = 'SELECT * FROM edges WHERE source = ?';
1097
+ const params = [sourceId];
1098
+ if (kinds && kinds.length > 0) {
1099
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
1100
+ params.push(...kinds);
1101
+ }
1102
+ if (provenance) {
1103
+ sql += ' AND provenance = ?';
1104
+ params.push(provenance);
1105
+ }
1106
+ const rows = this.db.prepare(sql).all(...params);
1107
+ return rows.map(rowToEdge);
1108
+ }
1109
+ if (!this.stmts.getEdgesBySource) {
1110
+ this.stmts.getEdgesBySource = this.db.prepare('SELECT * FROM edges WHERE source = ?');
1111
+ }
1112
+ const rows = this.stmts.getEdgesBySource.all(sourceId);
1113
+ return rows.map(rowToEdge);
1114
+ }
1115
+ /**
1116
+ * Get incoming edges to a node
1117
+ */
1118
+ getIncomingEdges(targetId, kinds) {
1119
+ if (kinds && kinds.length > 0) {
1120
+ const sql = `SELECT * FROM edges WHERE target = ? AND kind IN (${kinds.map(() => '?').join(',')})`;
1121
+ const rows = this.db.prepare(sql).all(targetId, ...kinds);
1122
+ return rows.map(rowToEdge);
1123
+ }
1124
+ if (!this.stmts.getEdgesByTarget) {
1125
+ this.stmts.getEdgesByTarget = this.db.prepare('SELECT * FROM edges WHERE target = ?');
1126
+ }
1127
+ const rows = this.stmts.getEdgesByTarget.all(targetId);
1128
+ return rows.map(rowToEdge);
1129
+ }
1130
+ /**
1131
+ * Find all edges where both source and target are in the given node set.
1132
+ * Useful for recovering inter-node connectivity after BFS.
1133
+ */
1134
+ findEdgesBetweenNodes(nodeIds, kinds) {
1135
+ if (nodeIds.length === 0)
1136
+ return [];
1137
+ const idsJson = JSON.stringify(nodeIds);
1138
+ let sql = `SELECT * FROM edges WHERE source IN (SELECT value FROM json_each(?)) AND target IN (SELECT value FROM json_each(?))`;
1139
+ const params = [idsJson, idsJson];
1140
+ if (kinds && kinds.length > 0) {
1141
+ sql += ` AND kind IN (${kinds.map(() => '?').join(',')})`;
1142
+ params.push(...kinds);
1143
+ }
1144
+ const rows = this.db.prepare(sql).all(...params);
1145
+ return rows.map(rowToEdge);
1146
+ }
1147
+ /**
1148
+ * Distinct file paths that DEPEND ON `filePath`: every file containing a
1149
+ * symbol with a cross-file edge (any kind except `contains`) into a symbol
1150
+ * of this file. This is the file-level projection of the symbol dependency
1151
+ * graph and the basis for blast-radius / `affected` test selection.
1152
+ *
1153
+ * It deliberately does NOT restrict to `imports` edges. In this graph an
1154
+ * `imports` edge connects a file to its own local import declarations
1155
+ * (it is always same-file), so an imports-only lookup returns zero
1156
+ * cross-file dependents for every file. The real cross-file dependency
1157
+ * signal is the resolved call/reference graph — calls, references,
1158
+ * instantiates, extends, implements, overrides, type_of, returns,
1159
+ * decorates — exactly what {@link GraphTraverser.getImpactRadius} traverses.
1160
+ * `contains` is excluded: a parent containing a symbol does not *depend* on
1161
+ * it. One indexed query (idx_nodes_file_path + idx_edges_target_kind).
1162
+ */
1163
+ getDependentFilePaths(filePath) {
1164
+ const sql = `SELECT DISTINCT src.file_path AS fp
1165
+ FROM edges e
1166
+ JOIN nodes tgt ON tgt.id = e.target
1167
+ JOIN nodes src ON src.id = e.source
1168
+ WHERE tgt.file_path = ?
1169
+ AND e.kind != 'contains'
1170
+ AND src.file_path != ?`;
1171
+ const rows = this.db.prepare(sql).all(filePath, filePath);
1172
+ return rows.map((r) => r.fp);
1173
+ }
1174
+ /**
1175
+ * Distinct file paths that `filePath` DEPENDS ON — the inverse of
1176
+ * {@link getDependentFilePaths}: every file containing a symbol that a
1177
+ * symbol of this file has a cross-file edge into. Same edge-kind rules
1178
+ * (all kinds except `contains`); same reason imports-only is insufficient.
1179
+ */
1180
+ getDependencyFilePaths(filePath) {
1181
+ const sql = `SELECT DISTINCT tgt.file_path AS fp
1182
+ FROM edges e
1183
+ JOIN nodes src ON src.id = e.source
1184
+ JOIN nodes tgt ON tgt.id = e.target
1185
+ WHERE src.file_path = ?
1186
+ AND e.kind != 'contains'
1187
+ AND tgt.file_path != ?`;
1188
+ const rows = this.db.prepare(sql).all(filePath, filePath);
1189
+ return rows.map((r) => r.fp);
1190
+ }
1191
+ /**
1192
+ * Cross-file edges whose TARGET is a node in `filePath` and whose SOURCE is a
1193
+ * node in a *different* file, paired with the target node's (name, kind) so a
1194
+ * caller can re-resolve the edge to the re-indexed target's new ID (node IDs
1195
+ * are `sha256(filePath:kind:name:line)`, so any line shift in the callee file
1196
+ * changes target IDs and a naive re-insert by old ID silently drops them).
1197
+ * Used by `storeExtractionResult` to preserve incoming edges across a file
1198
+ * re-index (issue #899). Same edge-kind rules as
1199
+ * {@link getDependentFilePaths}: all kinds except `contains`.
1200
+ */
1201
+ getCrossFileIncomingEdgesWithTarget(filePath) {
1202
+ const sql = `SELECT e.*, tgt.name AS target_name, tgt.kind AS target_kind
1203
+ FROM edges e
1204
+ JOIN nodes tgt ON tgt.id = e.target
1205
+ JOIN nodes src ON src.id = e.source
1206
+ WHERE tgt.file_path = ?
1207
+ AND e.kind != 'contains'
1208
+ AND src.file_path != ?`;
1209
+ const rows = this.db.prepare(sql).all(filePath, filePath);
1210
+ return rows.map(row => ({
1211
+ ...rowToEdge(row),
1212
+ targetName: row.target_name,
1213
+ targetKind: row.target_kind,
1214
+ }));
1215
+ }
1216
+ // ===========================================================================
1217
+ // File Operations
1218
+ // ===========================================================================
1219
+ /**
1220
+ * Insert or update a file record
1221
+ */
1222
+ upsertFile(file) {
1223
+ if (!this.stmts.upsertFile) {
1224
+ this.stmts.upsertFile = this.db.prepare(`
1225
+ INSERT INTO files (path, content_hash, language, size, modified_at, indexed_at, node_count, errors)
1226
+ VALUES (@path, @contentHash, @language, @size, @modifiedAt, @indexedAt, @nodeCount, @errors)
1227
+ ON CONFLICT(path) DO UPDATE SET
1228
+ content_hash = @contentHash,
1229
+ language = @language,
1230
+ size = @size,
1231
+ modified_at = @modifiedAt,
1232
+ indexed_at = @indexedAt,
1233
+ node_count = @nodeCount,
1234
+ errors = @errors
1235
+ `);
1236
+ }
1237
+ this.stmts.upsertFile.run({
1238
+ path: file.path,
1239
+ contentHash: file.contentHash,
1240
+ language: file.language,
1241
+ size: file.size,
1242
+ modifiedAt: file.modifiedAt,
1243
+ indexedAt: file.indexedAt,
1244
+ nodeCount: file.nodeCount,
1245
+ errors: file.errors ? JSON.stringify(file.errors) : null,
1246
+ });
1247
+ }
1248
+ /**
1249
+ * Delete a file record and its nodes
1250
+ */
1251
+ deleteFile(filePath) {
1252
+ this.db.transaction(() => {
1253
+ this.deleteNodesByFile(filePath);
1254
+ if (!this.stmts.deleteFile) {
1255
+ this.stmts.deleteFile = this.db.prepare('DELETE FROM files WHERE path = ?');
1256
+ }
1257
+ this.stmts.deleteFile.run(filePath);
1258
+ })();
1259
+ }
1260
+ /**
1261
+ * Get a file record by path
1262
+ */
1263
+ getFileByPath(filePath) {
1264
+ if (!this.stmts.getFileByPath) {
1265
+ this.stmts.getFileByPath = this.db.prepare('SELECT * FROM files WHERE path = ?');
1266
+ }
1267
+ const row = this.stmts.getFileByPath.get(filePath);
1268
+ return row ? rowToFileRecord(row) : null;
1269
+ }
1270
+ /**
1271
+ * Get all tracked files
1272
+ */
1273
+ getAllFiles() {
1274
+ if (!this.stmts.getAllFiles) {
1275
+ this.stmts.getAllFiles = this.db.prepare('SELECT * FROM files ORDER BY path');
1276
+ }
1277
+ const rows = this.stmts.getAllFiles.all();
1278
+ return rows.map(rowToFileRecord);
1279
+ }
1280
+ /**
1281
+ * Most recent index timestamp (ms since epoch) across all tracked files, or
1282
+ * null when nothing is indexed yet. One indexed aggregate, no per-row scan. (#329)
1283
+ */
1284
+ getLastIndexedAt() {
1285
+ const row = this.db
1286
+ .prepare('SELECT MAX(indexed_at) AS last FROM files')
1287
+ .get();
1288
+ return row?.last ?? null;
1289
+ }
1290
+ /**
1291
+ * Get files that need re-indexing (hash changed)
1292
+ */
1293
+ getStaleFiles(currentHashes) {
1294
+ const files = this.getAllFiles();
1295
+ return files.filter((f) => {
1296
+ const currentHash = currentHashes.get(f.path);
1297
+ return currentHash && currentHash !== f.contentHash;
1298
+ });
1299
+ }
1300
+ // ===========================================================================
1301
+ // Unresolved References
1302
+ // ===========================================================================
1303
+ /**
1304
+ * Insert an unresolved reference
1305
+ */
1306
+ insertUnresolvedRef(ref) {
1307
+ if (!this.stmts.insertUnresolved) {
1308
+ this.stmts.insertUnresolved = this.db.prepare(`
1309
+ INSERT INTO unresolved_refs (from_node_id, reference_name, reference_kind, line, col, candidates, file_path, language)
1310
+ VALUES (@fromNodeId, @referenceName, @referenceKind, @line, @col, @candidates, @filePath, @language)
1311
+ `);
1312
+ }
1313
+ this.stmts.insertUnresolved.run({
1314
+ fromNodeId: ref.fromNodeId,
1315
+ referenceName: ref.referenceName,
1316
+ referenceKind: ref.referenceKind,
1317
+ line: ref.line,
1318
+ col: ref.column,
1319
+ candidates: ref.candidates ? JSON.stringify(ref.candidates) : null,
1320
+ filePath: ref.filePath ?? '',
1321
+ language: ref.language ?? 'unknown',
1322
+ });
1323
+ }
1324
+ /**
1325
+ * Insert multiple unresolved references in a transaction
1326
+ */
1327
+ insertUnresolvedRefsBatch(refs) {
1328
+ if (refs.length === 0)
1329
+ return;
1330
+ const insert = this.db.transaction(() => {
1331
+ for (const ref of refs) {
1332
+ this.insertUnresolvedRef(ref);
1333
+ }
1334
+ });
1335
+ insert();
1336
+ }
1337
+ /**
1338
+ * Delete unresolved references from a node
1339
+ */
1340
+ deleteUnresolvedByNode(nodeId) {
1341
+ if (!this.stmts.deleteUnresolvedByNode) {
1342
+ this.stmts.deleteUnresolvedByNode = this.db.prepare('DELETE FROM unresolved_refs WHERE from_node_id = ?');
1343
+ }
1344
+ this.stmts.deleteUnresolvedByNode.run(nodeId);
1345
+ }
1346
+ /**
1347
+ * Get unresolved references by name (for resolution)
1348
+ */
1349
+ getUnresolvedByName(name) {
1350
+ if (!this.stmts.getUnresolvedByName) {
1351
+ this.stmts.getUnresolvedByName = this.db.prepare('SELECT * FROM unresolved_refs WHERE reference_name = ?');
1352
+ }
1353
+ const rows = this.stmts.getUnresolvedByName.all(name);
1354
+ return rows.map((row) => ({
1355
+ fromNodeId: row.from_node_id,
1356
+ referenceName: row.reference_name,
1357
+ referenceKind: row.reference_kind,
1358
+ line: row.line,
1359
+ column: row.col,
1360
+ candidates: row.candidates ? (0, utils_1.safeJsonParse)(row.candidates, undefined) : undefined,
1361
+ filePath: row.file_path,
1362
+ language: row.language,
1363
+ }));
1364
+ }
1365
+ /**
1366
+ * Get all unresolved references
1367
+ */
1368
+ getUnresolvedReferences() {
1369
+ const rows = this.db.prepare('SELECT * FROM unresolved_refs').all();
1370
+ return rows.map((row) => ({
1371
+ fromNodeId: row.from_node_id,
1372
+ referenceName: row.reference_name,
1373
+ referenceKind: row.reference_kind,
1374
+ line: row.line,
1375
+ column: row.col,
1376
+ candidates: row.candidates ? (0, utils_1.safeJsonParse)(row.candidates, undefined) : undefined,
1377
+ filePath: row.file_path,
1378
+ language: row.language,
1379
+ }));
1380
+ }
1381
+ /**
1382
+ * Get the count of unresolved references without loading them into memory
1383
+ */
1384
+ getUnresolvedReferencesCount() {
1385
+ if (!this.stmts.getUnresolvedCount) {
1386
+ this.stmts.getUnresolvedCount = this.db.prepare('SELECT COUNT(*) as count FROM unresolved_refs');
1387
+ }
1388
+ const row = this.stmts.getUnresolvedCount.get();
1389
+ return row.count;
1390
+ }
1391
+ /**
1392
+ * Get a batch of unresolved references using LIMIT/OFFSET pagination.
1393
+ * Used to process references in bounded memory chunks.
1394
+ */
1395
+ getUnresolvedReferencesBatch(offset, limit) {
1396
+ if (!this.stmts.getUnresolvedBatch) {
1397
+ this.stmts.getUnresolvedBatch = this.db.prepare('SELECT * FROM unresolved_refs LIMIT ? OFFSET ?');
1398
+ }
1399
+ const rows = this.stmts.getUnresolvedBatch.all(limit, offset);
1400
+ return rows.map((row) => ({
1401
+ fromNodeId: row.from_node_id,
1402
+ referenceName: row.reference_name,
1403
+ referenceKind: row.reference_kind,
1404
+ line: row.line,
1405
+ column: row.col,
1406
+ candidates: row.candidates ? (0, utils_1.safeJsonParse)(row.candidates, undefined) : undefined,
1407
+ filePath: row.file_path,
1408
+ language: row.language,
1409
+ }));
1410
+ }
1411
+ /**
1412
+ * Get all tracked file paths (lightweight — no full FileRecord objects)
1413
+ */
1414
+ getAllFilePaths() {
1415
+ if (!this.stmts.getAllFilePaths) {
1416
+ this.stmts.getAllFilePaths = this.db.prepare('SELECT path FROM files ORDER BY path');
1417
+ }
1418
+ const rows = this.stmts.getAllFilePaths.all();
1419
+ return rows.map((r) => r.path);
1420
+ }
1421
+ /**
1422
+ * Get all distinct node names (lightweight — just name strings for pre-filtering)
1423
+ */
1424
+ getAllNodeNames() {
1425
+ if (!this.stmts.getAllNodeNames) {
1426
+ this.stmts.getAllNodeNames = this.db.prepare('SELECT DISTINCT name FROM nodes');
1427
+ }
1428
+ const rows = this.stmts.getAllNodeNames.all();
1429
+ return rows.map((r) => r.name);
1430
+ }
1431
+ /**
1432
+ * Get unresolved references scoped to specific file paths.
1433
+ * Uses the idx_unresolved_file_path index for efficient lookup.
1434
+ */
1435
+ getUnresolvedReferencesByFiles(filePaths) {
1436
+ if (filePaths.length === 0)
1437
+ return [];
1438
+ // Chunk under SQLite's parameter limit: the first sync of a very large repo
1439
+ // passes every changed file here, which an unbounded `IN (...)` would bind
1440
+ // as one parameter each — exceeding MAX_VARIABLE_NUMBER and aborting with
1441
+ // "too many SQL variables". (#540)
1442
+ const rows = [];
1443
+ for (let i = 0; i < filePaths.length; i += SQLITE_PARAM_CHUNK_SIZE) {
1444
+ const chunk = filePaths.slice(i, i + SQLITE_PARAM_CHUNK_SIZE);
1445
+ const placeholders = chunk.map(() => '?').join(',');
1446
+ const chunkRows = this.db
1447
+ .prepare(`SELECT * FROM unresolved_refs WHERE file_path IN (${placeholders})`)
1448
+ .all(...chunk);
1449
+ rows.push(...chunkRows);
1450
+ }
1451
+ return rows.map((row) => ({
1452
+ fromNodeId: row.from_node_id,
1453
+ referenceName: row.reference_name,
1454
+ referenceKind: row.reference_kind,
1455
+ line: row.line,
1456
+ column: row.col,
1457
+ candidates: row.candidates ? (0, utils_1.safeJsonParse)(row.candidates, undefined) : undefined,
1458
+ filePath: row.file_path,
1459
+ language: row.language,
1460
+ }));
1461
+ }
1462
+ /**
1463
+ * Delete all unresolved references (after resolution)
1464
+ */
1465
+ clearUnresolvedReferences() {
1466
+ this.db.exec('DELETE FROM unresolved_refs');
1467
+ }
1468
+ /**
1469
+ * Delete resolved references by their IDs
1470
+ */
1471
+ deleteResolvedReferences(fromNodeIds) {
1472
+ if (fromNodeIds.length === 0)
1473
+ return;
1474
+ const placeholders = fromNodeIds.map(() => '?').join(',');
1475
+ this.db.prepare(`DELETE FROM unresolved_refs WHERE from_node_id IN (${placeholders})`).run(...fromNodeIds);
1476
+ }
1477
+ /**
1478
+ * Delete specific resolved references by (fromNodeId, referenceName, referenceKind) tuples.
1479
+ * More precise than deleteResolvedReferences — only removes refs that were actually resolved.
1480
+ */
1481
+ deleteSpecificResolvedReferences(refs) {
1482
+ if (refs.length === 0)
1483
+ return;
1484
+ const stmt = this.db.prepare('DELETE FROM unresolved_refs WHERE from_node_id = ? AND reference_name = ? AND reference_kind = ?');
1485
+ const deleteMany = this.db.transaction((items) => {
1486
+ for (const ref of items) {
1487
+ stmt.run(ref.fromNodeId, ref.referenceName, ref.referenceKind);
1488
+ }
1489
+ });
1490
+ deleteMany(refs);
1491
+ }
1492
+ // ===========================================================================
1493
+ // Statistics
1494
+ // ===========================================================================
1495
+ /**
1496
+ * Lightweight (nodes, edges) count snapshot. Used around an index/sync
1497
+ * run to compute true additions across extraction + resolution +
1498
+ * synthesis — the per-phase counter in the orchestrator only sees
1499
+ * extraction's contribution, which is why the CLI summary under-reported
1500
+ * the edge count (resolution + synthesizer edges were invisible).
1501
+ */
1502
+ getNodeAndEdgeCount() {
1503
+ return this.db
1504
+ .prepare('SELECT (SELECT COUNT(*) FROM nodes) AS nodes, (SELECT COUNT(*) FROM edges) AS edges')
1505
+ .get();
1506
+ }
1507
+ /**
1508
+ * Get graph statistics
1509
+ */
1510
+ getStats() {
1511
+ // Single query for all three aggregate counts
1512
+ const counts = this.db.prepare(`
1513
+ SELECT
1514
+ (SELECT COUNT(*) FROM nodes) AS node_count,
1515
+ (SELECT COUNT(*) FROM edges) AS edge_count,
1516
+ (SELECT COUNT(*) FROM files) AS file_count
1517
+ `).get();
1518
+ const nodesByKind = {};
1519
+ const nodeKindRows = this.db
1520
+ .prepare('SELECT kind, COUNT(*) as count FROM nodes GROUP BY kind')
1521
+ .all();
1522
+ for (const row of nodeKindRows) {
1523
+ nodesByKind[row.kind] = row.count;
1524
+ }
1525
+ const edgesByKind = {};
1526
+ const edgeKindRows = this.db
1527
+ .prepare('SELECT kind, COUNT(*) as count FROM edges GROUP BY kind')
1528
+ .all();
1529
+ for (const row of edgeKindRows) {
1530
+ edgesByKind[row.kind] = row.count;
1531
+ }
1532
+ const filesByLanguage = {};
1533
+ const languageRows = this.db
1534
+ .prepare('SELECT language, COUNT(*) as count FROM files GROUP BY language')
1535
+ .all();
1536
+ for (const row of languageRows) {
1537
+ filesByLanguage[row.language] = row.count;
1538
+ }
1539
+ return {
1540
+ nodeCount: counts.node_count,
1541
+ edgeCount: counts.edge_count,
1542
+ fileCount: counts.file_count,
1543
+ nodesByKind,
1544
+ edgesByKind,
1545
+ filesByLanguage,
1546
+ dbSizeBytes: 0, // Set by caller using DatabaseConnection.getSize()
1547
+ lastUpdated: Date.now(),
1548
+ };
1549
+ }
1550
+ // ===========================================================================
1551
+ // Project Metadata
1552
+ // ===========================================================================
1553
+ /**
1554
+ * Get a metadata value by key
1555
+ */
1556
+ getMetadata(key) {
1557
+ const row = this.db.prepare('SELECT value FROM project_metadata WHERE key = ?').get(key);
1558
+ return row?.value ?? null;
1559
+ }
1560
+ /**
1561
+ * Set a metadata key-value pair (upsert)
1562
+ */
1563
+ setMetadata(key, value) {
1564
+ this.db.prepare('INSERT INTO project_metadata (key, value, updated_at) VALUES (?, ?, ?) ON CONFLICT(key) DO UPDATE SET value = excluded.value, updated_at = excluded.updated_at').run(key, value, Date.now());
1565
+ }
1566
+ /**
1567
+ * Get all metadata as a key-value record
1568
+ */
1569
+ getAllMetadata() {
1570
+ const rows = this.db.prepare('SELECT key, value FROM project_metadata').all();
1571
+ const result = {};
1572
+ for (const row of rows) {
1573
+ result[row.key] = row.value;
1574
+ }
1575
+ return result;
1576
+ }
1577
+ /**
1578
+ * Clear all data from the database
1579
+ */
1580
+ clear() {
1581
+ this.nodeCache.clear();
1582
+ this.db.transaction(() => {
1583
+ this.db.exec('DELETE FROM unresolved_refs');
1584
+ this.db.exec('DELETE FROM edges');
1585
+ this.db.exec('DELETE FROM nodes');
1586
+ this.db.exec('DELETE FROM files');
1587
+ })();
1588
+ }
1589
+ }
1590
+ exports.QueryBuilder = QueryBuilder;
1591
+ //# sourceMappingURL=queries.js.map