@optave/codegraph 3.7.0 → 3.8.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (401) hide show
  1. package/README.md +28 -16
  2. package/dist/ast-analysis/engine.d.ts.map +1 -1
  3. package/dist/ast-analysis/engine.js +195 -30
  4. package/dist/ast-analysis/engine.js.map +1 -1
  5. package/dist/ast-analysis/metrics.d.ts +0 -3
  6. package/dist/ast-analysis/metrics.d.ts.map +1 -1
  7. package/dist/ast-analysis/metrics.js +30 -13
  8. package/dist/ast-analysis/metrics.js.map +1 -1
  9. package/dist/ast-analysis/rules/javascript.d.ts.map +1 -1
  10. package/dist/ast-analysis/rules/javascript.js +0 -1
  11. package/dist/ast-analysis/rules/javascript.js.map +1 -1
  12. package/dist/ast-analysis/shared.d.ts.map +1 -1
  13. package/dist/ast-analysis/shared.js +24 -19
  14. package/dist/ast-analysis/shared.js.map +1 -1
  15. package/dist/ast-analysis/visitor-utils.d.ts.map +1 -1
  16. package/dist/ast-analysis/visitor-utils.js +55 -39
  17. package/dist/ast-analysis/visitor-utils.js.map +1 -1
  18. package/dist/ast-analysis/visitor.d.ts.map +1 -1
  19. package/dist/ast-analysis/visitor.js +91 -70
  20. package/dist/ast-analysis/visitor.js.map +1 -1
  21. package/dist/ast-analysis/visitors/ast-store-visitor.d.ts.map +1 -1
  22. package/dist/ast-analysis/visitors/ast-store-visitor.js +52 -129
  23. package/dist/ast-analysis/visitors/ast-store-visitor.js.map +1 -1
  24. package/dist/ast-analysis/visitors/complexity-visitor.d.ts.map +1 -1
  25. package/dist/ast-analysis/visitors/complexity-visitor.js +32 -39
  26. package/dist/ast-analysis/visitors/complexity-visitor.js.map +1 -1
  27. package/dist/ast-analysis/visitors/dataflow-visitor.d.ts.map +1 -1
  28. package/dist/ast-analysis/visitors/dataflow-visitor.js +57 -38
  29. package/dist/ast-analysis/visitors/dataflow-visitor.js.map +1 -1
  30. package/dist/cli/commands/ast.js +2 -2
  31. package/dist/cli/commands/ast.js.map +1 -1
  32. package/dist/cli/commands/watch.d.ts.map +1 -1
  33. package/dist/cli/commands/watch.js +16 -2
  34. package/dist/cli/commands/watch.js.map +1 -1
  35. package/dist/db/connection.d.ts.map +1 -1
  36. package/dist/db/connection.js +29 -26
  37. package/dist/db/connection.js.map +1 -1
  38. package/dist/db/query-builder.d.ts.map +1 -1
  39. package/dist/db/query-builder.js +16 -5
  40. package/dist/db/query-builder.js.map +1 -1
  41. package/dist/db/repository/base.d.ts +10 -0
  42. package/dist/db/repository/base.d.ts.map +1 -1
  43. package/dist/db/repository/base.js +17 -0
  44. package/dist/db/repository/base.js.map +1 -1
  45. package/dist/db/repository/native-repository.d.ts +6 -1
  46. package/dist/db/repository/native-repository.d.ts.map +1 -1
  47. package/dist/db/repository/native-repository.js +77 -1
  48. package/dist/db/repository/native-repository.js.map +1 -1
  49. package/dist/db/repository/nodes.d.ts.map +1 -1
  50. package/dist/db/repository/nodes.js +8 -4
  51. package/dist/db/repository/nodes.js.map +1 -1
  52. package/dist/db/repository/sqlite-repository.d.ts +3 -0
  53. package/dist/db/repository/sqlite-repository.d.ts.map +1 -1
  54. package/dist/db/repository/sqlite-repository.js +26 -0
  55. package/dist/db/repository/sqlite-repository.js.map +1 -1
  56. package/dist/domain/analysis/brief.d.ts.map +1 -1
  57. package/dist/domain/analysis/brief.js +13 -17
  58. package/dist/domain/analysis/brief.js.map +1 -1
  59. package/dist/domain/analysis/context.d.ts.map +1 -1
  60. package/dist/domain/analysis/context.js +14 -11
  61. package/dist/domain/analysis/context.js.map +1 -1
  62. package/dist/domain/analysis/dependencies.d.ts.map +1 -1
  63. package/dist/domain/analysis/dependencies.js +53 -52
  64. package/dist/domain/analysis/dependencies.js.map +1 -1
  65. package/dist/domain/analysis/fn-impact.d.ts +2 -7
  66. package/dist/domain/analysis/fn-impact.d.ts.map +1 -1
  67. package/dist/domain/analysis/fn-impact.js +33 -31
  68. package/dist/domain/analysis/fn-impact.js.map +1 -1
  69. package/dist/domain/analysis/implementations.d.ts.map +1 -1
  70. package/dist/domain/analysis/implementations.js +11 -19
  71. package/dist/domain/analysis/implementations.js.map +1 -1
  72. package/dist/domain/analysis/module-map.d.ts.map +1 -1
  73. package/dist/domain/analysis/module-map.js +55 -76
  74. package/dist/domain/analysis/module-map.js.map +1 -1
  75. package/dist/domain/analysis/query-helpers.d.ts +7 -0
  76. package/dist/domain/analysis/query-helpers.d.ts.map +1 -1
  77. package/dist/domain/analysis/query-helpers.js +15 -1
  78. package/dist/domain/analysis/query-helpers.js.map +1 -1
  79. package/dist/domain/graph/builder/pipeline.d.ts.map +1 -1
  80. package/dist/domain/graph/builder/pipeline.js +315 -43
  81. package/dist/domain/graph/builder/pipeline.js.map +1 -1
  82. package/dist/domain/graph/builder/stages/build-edges.d.ts.map +1 -1
  83. package/dist/domain/graph/builder/stages/build-edges.js +106 -1
  84. package/dist/domain/graph/builder/stages/build-edges.js.map +1 -1
  85. package/dist/domain/graph/builder/stages/collect-files.d.ts.map +1 -1
  86. package/dist/domain/graph/builder/stages/collect-files.js +17 -5
  87. package/dist/domain/graph/builder/stages/collect-files.js.map +1 -1
  88. package/dist/domain/graph/builder/stages/detect-changes.d.ts.map +1 -1
  89. package/dist/domain/graph/builder/stages/detect-changes.js +99 -51
  90. package/dist/domain/graph/builder/stages/detect-changes.js.map +1 -1
  91. package/dist/domain/graph/builder/stages/finalize.d.ts.map +1 -1
  92. package/dist/domain/graph/builder/stages/finalize.js +34 -7
  93. package/dist/domain/graph/builder/stages/finalize.js.map +1 -1
  94. package/dist/domain/graph/builder/stages/insert-nodes.d.ts.map +1 -1
  95. package/dist/domain/graph/builder/stages/insert-nodes.js +50 -26
  96. package/dist/domain/graph/builder/stages/insert-nodes.js.map +1 -1
  97. package/dist/domain/graph/builder/stages/resolve-imports.d.ts.map +1 -1
  98. package/dist/domain/graph/builder/stages/resolve-imports.js +95 -84
  99. package/dist/domain/graph/builder/stages/resolve-imports.js.map +1 -1
  100. package/dist/domain/graph/cycles.d.ts +6 -0
  101. package/dist/domain/graph/cycles.d.ts.map +1 -1
  102. package/dist/domain/graph/cycles.js +114 -22
  103. package/dist/domain/graph/cycles.js.map +1 -1
  104. package/dist/domain/graph/resolve.js +1 -1
  105. package/dist/domain/graph/resolve.js.map +1 -1
  106. package/dist/domain/graph/watcher.d.ts +2 -0
  107. package/dist/domain/graph/watcher.d.ts.map +1 -1
  108. package/dist/domain/graph/watcher.js +170 -75
  109. package/dist/domain/graph/watcher.js.map +1 -1
  110. package/dist/domain/parser.d.ts +1 -6
  111. package/dist/domain/parser.d.ts.map +1 -1
  112. package/dist/domain/parser.js +101 -32
  113. package/dist/domain/parser.js.map +1 -1
  114. package/dist/domain/search/generator.js +1 -1
  115. package/dist/domain/search/generator.js.map +1 -1
  116. package/dist/domain/search/models.d.ts +4 -3
  117. package/dist/domain/search/models.d.ts.map +1 -1
  118. package/dist/domain/search/models.js +18 -5
  119. package/dist/domain/search/models.js.map +1 -1
  120. package/dist/domain/search/search/hybrid.d.ts.map +1 -1
  121. package/dist/domain/search/search/hybrid.js +29 -18
  122. package/dist/domain/search/search/hybrid.js.map +1 -1
  123. package/dist/extractors/clojure.d.ts +12 -0
  124. package/dist/extractors/clojure.d.ts.map +1 -0
  125. package/dist/extractors/clojure.js +245 -0
  126. package/dist/extractors/clojure.js.map +1 -0
  127. package/dist/extractors/cuda.d.ts +11 -0
  128. package/dist/extractors/cuda.d.ts.map +1 -0
  129. package/dist/extractors/cuda.js +302 -0
  130. package/dist/extractors/cuda.js.map +1 -0
  131. package/dist/extractors/erlang.d.ts +14 -0
  132. package/dist/extractors/erlang.d.ts.map +1 -0
  133. package/dist/extractors/erlang.js +239 -0
  134. package/dist/extractors/erlang.js.map +1 -0
  135. package/dist/extractors/fsharp.d.ts +13 -0
  136. package/dist/extractors/fsharp.d.ts.map +1 -0
  137. package/dist/extractors/fsharp.js +218 -0
  138. package/dist/extractors/fsharp.js.map +1 -0
  139. package/dist/extractors/gleam.d.ts +14 -0
  140. package/dist/extractors/gleam.d.ts.map +1 -0
  141. package/dist/extractors/gleam.js +229 -0
  142. package/dist/extractors/gleam.js.map +1 -0
  143. package/dist/extractors/go.js +36 -33
  144. package/dist/extractors/go.js.map +1 -1
  145. package/dist/extractors/groovy.d.ts +10 -0
  146. package/dist/extractors/groovy.d.ts.map +1 -0
  147. package/dist/extractors/groovy.js +304 -0
  148. package/dist/extractors/groovy.js.map +1 -0
  149. package/dist/extractors/helpers.d.ts.map +1 -1
  150. package/dist/extractors/helpers.js +40 -29
  151. package/dist/extractors/helpers.js.map +1 -1
  152. package/dist/extractors/index.d.ts +11 -0
  153. package/dist/extractors/index.d.ts.map +1 -1
  154. package/dist/extractors/index.js +11 -0
  155. package/dist/extractors/index.js.map +1 -1
  156. package/dist/extractors/java.js +58 -46
  157. package/dist/extractors/java.js.map +1 -1
  158. package/dist/extractors/javascript.js +46 -45
  159. package/dist/extractors/javascript.js.map +1 -1
  160. package/dist/extractors/julia.d.ts +16 -0
  161. package/dist/extractors/julia.d.ts.map +1 -0
  162. package/dist/extractors/julia.js +287 -0
  163. package/dist/extractors/julia.js.map +1 -0
  164. package/dist/extractors/kotlin.js +84 -78
  165. package/dist/extractors/kotlin.js.map +1 -1
  166. package/dist/extractors/objc.d.ts +9 -0
  167. package/dist/extractors/objc.d.ts.map +1 -0
  168. package/dist/extractors/objc.js +406 -0
  169. package/dist/extractors/objc.js.map +1 -0
  170. package/dist/extractors/ocaml.js +74 -0
  171. package/dist/extractors/ocaml.js.map +1 -1
  172. package/dist/extractors/python.js +29 -24
  173. package/dist/extractors/python.js.map +1 -1
  174. package/dist/extractors/r.d.ts +13 -0
  175. package/dist/extractors/r.d.ts.map +1 -0
  176. package/dist/extractors/r.js +251 -0
  177. package/dist/extractors/r.js.map +1 -0
  178. package/dist/extractors/rust.js +41 -32
  179. package/dist/extractors/rust.js.map +1 -1
  180. package/dist/extractors/solidity.d.ts +9 -0
  181. package/dist/extractors/solidity.d.ts.map +1 -0
  182. package/dist/extractors/solidity.js +365 -0
  183. package/dist/extractors/solidity.js.map +1 -0
  184. package/dist/extractors/swift.js +83 -81
  185. package/dist/extractors/swift.js.map +1 -1
  186. package/dist/extractors/verilog.d.ts +9 -0
  187. package/dist/extractors/verilog.d.ts.map +1 -0
  188. package/dist/extractors/verilog.js +286 -0
  189. package/dist/extractors/verilog.js.map +1 -0
  190. package/dist/extractors/zig.js +58 -60
  191. package/dist/extractors/zig.js.map +1 -1
  192. package/dist/features/ast.d.ts +16 -14
  193. package/dist/features/ast.d.ts.map +1 -1
  194. package/dist/features/ast.js +84 -83
  195. package/dist/features/ast.js.map +1 -1
  196. package/dist/features/audit.d.ts.map +1 -1
  197. package/dist/features/audit.js +8 -6
  198. package/dist/features/audit.js.map +1 -1
  199. package/dist/features/branch-compare.d.ts.map +1 -1
  200. package/dist/features/branch-compare.js +69 -72
  201. package/dist/features/branch-compare.js.map +1 -1
  202. package/dist/features/communities.d.ts.map +1 -1
  203. package/dist/features/communities.js +19 -7
  204. package/dist/features/communities.js.map +1 -1
  205. package/dist/features/complexity.d.ts.map +1 -1
  206. package/dist/features/complexity.js +120 -125
  207. package/dist/features/complexity.js.map +1 -1
  208. package/dist/features/dataflow.d.ts.map +1 -1
  209. package/dist/features/dataflow.js +136 -137
  210. package/dist/features/dataflow.js.map +1 -1
  211. package/dist/features/flow.d.ts.map +1 -1
  212. package/dist/features/flow.js +84 -79
  213. package/dist/features/flow.js.map +1 -1
  214. package/dist/features/structure-query.d.ts.map +1 -1
  215. package/dist/features/structure-query.js +69 -65
  216. package/dist/features/structure-query.js.map +1 -1
  217. package/dist/graph/algorithms/bfs.d.ts +2 -0
  218. package/dist/graph/algorithms/bfs.d.ts.map +1 -1
  219. package/dist/graph/algorithms/bfs.js +27 -0
  220. package/dist/graph/algorithms/bfs.js.map +1 -1
  221. package/dist/graph/algorithms/centrality.d.ts +2 -0
  222. package/dist/graph/algorithms/centrality.d.ts.map +1 -1
  223. package/dist/graph/algorithms/centrality.js +28 -0
  224. package/dist/graph/algorithms/centrality.js.map +1 -1
  225. package/dist/graph/algorithms/leiden/optimiser.d.ts.map +1 -1
  226. package/dist/graph/algorithms/leiden/optimiser.js +70 -55
  227. package/dist/graph/algorithms/leiden/optimiser.js.map +1 -1
  228. package/dist/graph/algorithms/leiden/partition.d.ts.map +1 -1
  229. package/dist/graph/algorithms/leiden/partition.js +288 -266
  230. package/dist/graph/algorithms/leiden/partition.js.map +1 -1
  231. package/dist/graph/algorithms/louvain.d.ts +3 -4
  232. package/dist/graph/algorithms/louvain.d.ts.map +1 -1
  233. package/dist/graph/algorithms/louvain.js +29 -0
  234. package/dist/graph/algorithms/louvain.js.map +1 -1
  235. package/dist/graph/algorithms/shortest-path.d.ts +2 -0
  236. package/dist/graph/algorithms/shortest-path.d.ts.map +1 -1
  237. package/dist/graph/algorithms/shortest-path.js +18 -1
  238. package/dist/graph/algorithms/shortest-path.js.map +1 -1
  239. package/dist/graph/model.d.ts.map +1 -1
  240. package/dist/graph/model.js +5 -1
  241. package/dist/graph/model.js.map +1 -1
  242. package/dist/infrastructure/config.d.ts.map +1 -1
  243. package/dist/infrastructure/config.js +6 -4
  244. package/dist/infrastructure/config.js.map +1 -1
  245. package/dist/infrastructure/suppress.d.ts +25 -0
  246. package/dist/infrastructure/suppress.d.ts.map +1 -0
  247. package/dist/infrastructure/suppress.js +43 -0
  248. package/dist/infrastructure/suppress.js.map +1 -0
  249. package/dist/mcp/server.d.ts.map +1 -1
  250. package/dist/mcp/server.js +29 -24
  251. package/dist/mcp/server.js.map +1 -1
  252. package/dist/presentation/dataflow.d.ts.map +1 -1
  253. package/dist/presentation/dataflow.js +47 -38
  254. package/dist/presentation/dataflow.js.map +1 -1
  255. package/dist/presentation/diff-impact-mermaid.d.ts.map +1 -1
  256. package/dist/presentation/diff-impact-mermaid.js +60 -51
  257. package/dist/presentation/diff-impact-mermaid.js.map +1 -1
  258. package/dist/presentation/queries-cli/exports.d.ts.map +1 -1
  259. package/dist/presentation/queries-cli/exports.js +20 -14
  260. package/dist/presentation/queries-cli/exports.js.map +1 -1
  261. package/dist/presentation/queries-cli/impact.d.ts.map +1 -1
  262. package/dist/presentation/queries-cli/impact.js +15 -13
  263. package/dist/presentation/queries-cli/impact.js.map +1 -1
  264. package/dist/presentation/queries-cli/inspect.d.ts.map +1 -1
  265. package/dist/presentation/queries-cli/inspect.js +101 -79
  266. package/dist/presentation/queries-cli/inspect.js.map +1 -1
  267. package/dist/presentation/queries-cli/overview.d.ts.map +1 -1
  268. package/dist/presentation/queries-cli/overview.js +25 -16
  269. package/dist/presentation/queries-cli/overview.js.map +1 -1
  270. package/dist/presentation/queries-cli/path.js +26 -20
  271. package/dist/presentation/queries-cli/path.js.map +1 -1
  272. package/dist/presentation/result-formatter.d.ts +10 -0
  273. package/dist/presentation/result-formatter.d.ts.map +1 -1
  274. package/dist/presentation/result-formatter.js +16 -1
  275. package/dist/presentation/result-formatter.js.map +1 -1
  276. package/dist/presentation/viewer.d.ts.map +1 -1
  277. package/dist/presentation/viewer.js +18 -12
  278. package/dist/presentation/viewer.js.map +1 -1
  279. package/dist/shared/errors.d.ts +5 -0
  280. package/dist/shared/errors.d.ts.map +1 -1
  281. package/dist/shared/errors.js +5 -0
  282. package/dist/shared/errors.js.map +1 -1
  283. package/dist/shared/hierarchy.d.ts +8 -2
  284. package/dist/shared/hierarchy.d.ts.map +1 -1
  285. package/dist/shared/hierarchy.js +42 -1
  286. package/dist/shared/hierarchy.js.map +1 -1
  287. package/dist/shared/normalize.d.ts +6 -1
  288. package/dist/shared/normalize.d.ts.map +1 -1
  289. package/dist/shared/normalize.js +20 -12
  290. package/dist/shared/normalize.js.map +1 -1
  291. package/dist/shared/paginate.d.ts +0 -9
  292. package/dist/shared/paginate.d.ts.map +1 -1
  293. package/dist/shared/paginate.js +0 -15
  294. package/dist/shared/paginate.js.map +1 -1
  295. package/dist/types.d.ts +129 -3
  296. package/dist/types.d.ts.map +1 -1
  297. package/grammars/tree-sitter-clojure.wasm +0 -0
  298. package/grammars/tree-sitter-cuda.wasm +0 -0
  299. package/grammars/tree-sitter-erlang.wasm +0 -0
  300. package/grammars/tree-sitter-fsharp.wasm +0 -0
  301. package/grammars/tree-sitter-gleam.wasm +0 -0
  302. package/grammars/tree-sitter-groovy.wasm +0 -0
  303. package/grammars/tree-sitter-julia.wasm +0 -0
  304. package/grammars/tree-sitter-objc.wasm +0 -0
  305. package/grammars/tree-sitter-ocaml_interface.wasm +0 -0
  306. package/grammars/tree-sitter-r.wasm +0 -0
  307. package/grammars/tree-sitter-solidity.wasm +0 -0
  308. package/grammars/tree-sitter-verilog.wasm +0 -0
  309. package/package.json +18 -7
  310. package/src/ast-analysis/engine.ts +245 -42
  311. package/src/ast-analysis/metrics.ts +33 -11
  312. package/src/ast-analysis/rules/javascript.ts +0 -1
  313. package/src/ast-analysis/shared.ts +33 -24
  314. package/src/ast-analysis/visitor-utils.ts +52 -32
  315. package/src/ast-analysis/visitor.ts +132 -71
  316. package/src/ast-analysis/visitors/ast-store-visitor.ts +49 -119
  317. package/src/ast-analysis/visitors/complexity-visitor.ts +35 -40
  318. package/src/ast-analysis/visitors/dataflow-visitor.ts +87 -43
  319. package/src/cli/commands/ast.ts +2 -2
  320. package/src/cli/commands/watch.ts +16 -2
  321. package/src/db/connection.ts +29 -28
  322. package/src/db/query-builder.ts +15 -3
  323. package/src/db/repository/base.ts +20 -0
  324. package/src/db/repository/native-repository.ts +79 -1
  325. package/src/db/repository/nodes.ts +13 -8
  326. package/src/db/repository/sqlite-repository.ts +29 -0
  327. package/src/domain/analysis/brief.ts +15 -25
  328. package/src/domain/analysis/context.ts +17 -10
  329. package/src/domain/analysis/dependencies.ts +67 -76
  330. package/src/domain/analysis/fn-impact.ts +36 -43
  331. package/src/domain/analysis/implementations.ts +11 -17
  332. package/src/domain/analysis/module-map.ts +58 -92
  333. package/src/domain/analysis/query-helpers.ts +18 -1
  334. package/src/domain/graph/builder/pipeline.ts +366 -41
  335. package/src/domain/graph/builder/stages/build-edges.ts +162 -1
  336. package/src/domain/graph/builder/stages/collect-files.ts +18 -7
  337. package/src/domain/graph/builder/stages/detect-changes.ts +110 -56
  338. package/src/domain/graph/builder/stages/finalize.ts +41 -11
  339. package/src/domain/graph/builder/stages/insert-nodes.ts +75 -39
  340. package/src/domain/graph/builder/stages/resolve-imports.ts +122 -100
  341. package/src/domain/graph/cycles.ts +110 -23
  342. package/src/domain/graph/resolve.ts +1 -1
  343. package/src/domain/graph/watcher.ts +202 -96
  344. package/src/domain/parser.ts +122 -28
  345. package/src/domain/search/generator.ts +1 -1
  346. package/src/domain/search/models.ts +17 -4
  347. package/src/domain/search/search/hybrid.ts +69 -51
  348. package/src/extractors/clojure.ts +273 -0
  349. package/src/extractors/cuda.ts +316 -0
  350. package/src/extractors/erlang.ts +252 -0
  351. package/src/extractors/fsharp.ts +253 -0
  352. package/src/extractors/gleam.ts +246 -0
  353. package/src/extractors/go.ts +43 -33
  354. package/src/extractors/groovy.ts +332 -0
  355. package/src/extractors/helpers.ts +37 -23
  356. package/src/extractors/index.ts +11 -0
  357. package/src/extractors/java.ts +66 -47
  358. package/src/extractors/javascript.ts +45 -46
  359. package/src/extractors/julia.ts +318 -0
  360. package/src/extractors/kotlin.ts +84 -77
  361. package/src/extractors/objc.ts +431 -0
  362. package/src/extractors/ocaml.ts +78 -0
  363. package/src/extractors/python.ts +31 -25
  364. package/src/extractors/r.ts +253 -0
  365. package/src/extractors/rust.ts +37 -29
  366. package/src/extractors/solidity.ts +394 -0
  367. package/src/extractors/swift.ts +81 -80
  368. package/src/extractors/verilog.ts +315 -0
  369. package/src/extractors/zig.ts +58 -61
  370. package/src/features/ast.ts +131 -112
  371. package/src/features/audit.ts +8 -6
  372. package/src/features/branch-compare.ts +105 -79
  373. package/src/features/communities.ts +25 -10
  374. package/src/features/complexity.ts +171 -134
  375. package/src/features/dataflow.ts +165 -175
  376. package/src/features/flow.ts +129 -92
  377. package/src/features/structure-query.ts +79 -64
  378. package/src/graph/algorithms/bfs.ts +34 -0
  379. package/src/graph/algorithms/centrality.ts +30 -0
  380. package/src/graph/algorithms/leiden/optimiser.ts +99 -55
  381. package/src/graph/algorithms/leiden/partition.ts +359 -294
  382. package/src/graph/algorithms/louvain.ts +31 -4
  383. package/src/graph/algorithms/shortest-path.ts +20 -1
  384. package/src/graph/model.ts +6 -1
  385. package/src/infrastructure/config.ts +6 -4
  386. package/src/infrastructure/suppress.ts +47 -0
  387. package/src/mcp/server.ts +53 -37
  388. package/src/presentation/dataflow.ts +50 -44
  389. package/src/presentation/diff-impact-mermaid.ts +104 -62
  390. package/src/presentation/queries-cli/exports.ts +21 -13
  391. package/src/presentation/queries-cli/impact.ts +15 -13
  392. package/src/presentation/queries-cli/inspect.ts +100 -81
  393. package/src/presentation/queries-cli/overview.ts +26 -16
  394. package/src/presentation/queries-cli/path.ts +33 -25
  395. package/src/presentation/result-formatter.ts +19 -1
  396. package/src/presentation/viewer.ts +42 -14
  397. package/src/shared/errors.ts +6 -0
  398. package/src/shared/hierarchy.ts +50 -2
  399. package/src/shared/normalize.ts +31 -12
  400. package/src/shared/paginate.ts +0 -17
  401. package/src/types.ts +138 -3
@@ -0,0 +1,316 @@
1
+ import type {
2
+ Call,
3
+ ExtractorOutput,
4
+ SubDeclaration,
5
+ TreeSitterNode,
6
+ TreeSitterTree,
7
+ } from '../types.js';
8
+ import { extractModifierVisibility, findChild, nodeEndLine } from './helpers.js';
9
+
10
+ /**
11
+ * Extract symbols from CUDA files.
12
+ *
13
+ * CUDA is a C++ superset. The tree-sitter-cuda grammar extends C++ with
14
+ * __global__, __device__, __host__, __shared__ qualifiers and kernel
15
+ * launch syntax (<<<...>>>). We reuse C++ handler patterns and add
16
+ * CUDA-specific qualifier detection.
17
+ */
18
+ export function extractCudaSymbols(tree: TreeSitterTree, _filePath: string): ExtractorOutput {
19
+ const ctx: ExtractorOutput = {
20
+ definitions: [],
21
+ calls: [],
22
+ imports: [],
23
+ classes: [],
24
+ exports: [],
25
+ typeMap: new Map(),
26
+ };
27
+
28
+ walkCudaNode(tree.rootNode, ctx);
29
+ return ctx;
30
+ }
31
+
32
+ const CUDA_QUALIFIERS = new Set([
33
+ '__global__',
34
+ '__device__',
35
+ '__host__',
36
+ '__shared__',
37
+ '__constant__',
38
+ ]);
39
+
40
+ function walkCudaNode(node: TreeSitterNode, ctx: ExtractorOutput): void {
41
+ switch (node.type) {
42
+ case 'function_definition':
43
+ handleCudaFunctionDef(node, ctx);
44
+ break;
45
+ case 'class_specifier':
46
+ handleCudaClassSpecifier(node, ctx);
47
+ break;
48
+ case 'struct_specifier':
49
+ handleCudaStructSpecifier(node, ctx);
50
+ break;
51
+ case 'enum_specifier':
52
+ handleCudaEnumSpecifier(node, ctx);
53
+ break;
54
+ case 'namespace_definition':
55
+ handleCudaNamespaceDef(node, ctx);
56
+ break;
57
+ case 'type_definition':
58
+ handleCudaTypedef(node, ctx);
59
+ break;
60
+ case 'preproc_include':
61
+ handleCudaInclude(node, ctx);
62
+ break;
63
+ case 'call_expression':
64
+ handleCudaCallExpression(node, ctx);
65
+ break;
66
+ }
67
+
68
+ for (let i = 0; i < node.childCount; i++) {
69
+ const child = node.child(i);
70
+ if (child) walkCudaNode(child, ctx);
71
+ }
72
+ }
73
+
74
+ // ── Handlers ───────────────────────────────────────────────────────────────
75
+
76
+ function handleCudaFunctionDef(node: TreeSitterNode, ctx: ExtractorOutput): void {
77
+ const declarator = node.childForFieldName('declarator');
78
+ if (!declarator) return;
79
+ const funcDeclarator =
80
+ declarator.type === 'function_declarator'
81
+ ? declarator
82
+ : findChild(declarator, 'function_declarator');
83
+ if (!funcDeclarator) return;
84
+ const nameNode = funcDeclarator.childForFieldName('declarator');
85
+ if (!nameNode) return;
86
+ const name = nameNode.text;
87
+
88
+ const parentClass = findCudaParentClass(node);
89
+ const fullName = parentClass ? `${parentClass}.${name}` : name;
90
+ const kind = parentClass ? 'method' : 'function';
91
+
92
+ const params = extractCudaParameters(funcDeclarator.childForFieldName('parameters'));
93
+ const decorators = extractCudaQualifiers(node);
94
+
95
+ ctx.definitions.push({
96
+ name: fullName,
97
+ kind,
98
+ line: node.startPosition.row + 1,
99
+ endLine: nodeEndLine(node),
100
+ children: params.length > 0 ? params : undefined,
101
+ visibility: parentClass ? extractModifierVisibility(node) : undefined,
102
+ decorators: decorators.length > 0 ? decorators : undefined,
103
+ });
104
+ }
105
+
106
+ function handleCudaClassSpecifier(node: TreeSitterNode, ctx: ExtractorOutput): void {
107
+ const nameNode = node.childForFieldName('name');
108
+ if (!nameNode) return;
109
+ const children = extractCudaClassFields(node);
110
+ ctx.definitions.push({
111
+ name: nameNode.text,
112
+ kind: 'class',
113
+ line: node.startPosition.row + 1,
114
+ endLine: nodeEndLine(node),
115
+ children: children.length > 0 ? children : undefined,
116
+ });
117
+
118
+ const baseClause = findChild(node, 'base_class_clause');
119
+ if (baseClause) {
120
+ for (let i = 0; i < baseClause.childCount; i++) {
121
+ const child = baseClause.child(i);
122
+ if (child && (child.type === 'type_identifier' || child.type === 'qualified_identifier')) {
123
+ ctx.classes.push({
124
+ name: nameNode.text,
125
+ extends: child.text,
126
+ line: node.startPosition.row + 1,
127
+ });
128
+ }
129
+ }
130
+ }
131
+ }
132
+
133
+ function handleCudaStructSpecifier(node: TreeSitterNode, ctx: ExtractorOutput): void {
134
+ const nameNode = node.childForFieldName('name');
135
+ if (!nameNode) return;
136
+ const children = extractCudaClassFields(node);
137
+ ctx.definitions.push({
138
+ name: nameNode.text,
139
+ kind: 'struct',
140
+ line: node.startPosition.row + 1,
141
+ endLine: nodeEndLine(node),
142
+ children: children.length > 0 ? children : undefined,
143
+ });
144
+ }
145
+
146
+ function handleCudaEnumSpecifier(node: TreeSitterNode, ctx: ExtractorOutput): void {
147
+ const nameNode = node.childForFieldName('name');
148
+ if (!nameNode) return;
149
+ const children = extractCudaEnumEntries(node);
150
+ ctx.definitions.push({
151
+ name: nameNode.text,
152
+ kind: 'enum',
153
+ line: node.startPosition.row + 1,
154
+ endLine: nodeEndLine(node),
155
+ children: children.length > 0 ? children : undefined,
156
+ });
157
+ }
158
+
159
+ function handleCudaNamespaceDef(node: TreeSitterNode, ctx: ExtractorOutput): void {
160
+ const nameNode = node.childForFieldName('name');
161
+ if (!nameNode) return;
162
+ ctx.definitions.push({
163
+ name: nameNode.text,
164
+ kind: 'namespace',
165
+ line: node.startPosition.row + 1,
166
+ endLine: nodeEndLine(node),
167
+ });
168
+ }
169
+
170
+ function handleCudaTypedef(node: TreeSitterNode, ctx: ExtractorOutput): void {
171
+ let name: string | undefined;
172
+ for (let i = node.childCount - 1; i >= 0; i--) {
173
+ const child = node.child(i);
174
+ if (
175
+ child &&
176
+ (child.type === 'type_identifier' ||
177
+ child.type === 'identifier' ||
178
+ child.type === 'primitive_type')
179
+ ) {
180
+ name = child.text;
181
+ break;
182
+ }
183
+ }
184
+ if (!name) return;
185
+ ctx.definitions.push({
186
+ name,
187
+ kind: 'type',
188
+ line: node.startPosition.row + 1,
189
+ endLine: nodeEndLine(node),
190
+ });
191
+ }
192
+
193
+ function handleCudaInclude(node: TreeSitterNode, ctx: ExtractorOutput): void {
194
+ const pathNode = node.childForFieldName('path');
195
+ if (!pathNode) return;
196
+ const raw = pathNode.text;
197
+ const source = raw.replace(/^["<]|[">]$/g, '');
198
+ const lastName = source.split('/').pop() ?? source;
199
+ ctx.imports.push({
200
+ source,
201
+ names: [lastName],
202
+ line: node.startPosition.row + 1,
203
+ cInclude: true,
204
+ });
205
+ }
206
+
207
+ function handleCudaCallExpression(node: TreeSitterNode, ctx: ExtractorOutput): void {
208
+ const funcNode = node.childForFieldName('function');
209
+ if (!funcNode) return;
210
+ const call: Call = { name: '', line: node.startPosition.row + 1 };
211
+ if (funcNode.type === 'field_expression') {
212
+ const field = funcNode.childForFieldName('field');
213
+ const argument = funcNode.childForFieldName('argument');
214
+ if (field) call.name = field.text;
215
+ if (argument) call.receiver = argument.text;
216
+ } else {
217
+ call.name = funcNode.text;
218
+ }
219
+ if (call.name) ctx.calls.push(call);
220
+ }
221
+
222
+ // ── Helpers ────────────────────────────────────────────────────────────────
223
+
224
+ function extractCudaQualifiers(node: TreeSitterNode): string[] {
225
+ const qualifiers: string[] = [];
226
+ for (let i = 0; i < node.childCount; i++) {
227
+ const child = node.child(i);
228
+ if (!child) continue;
229
+ // Check direct text match for bare qualifier tokens, or look inside
230
+ // storage_class_specifier / attribute_specifier wrapper nodes.
231
+ // Use `else if` to avoid pushing the same qualifier twice when
232
+ // wrapper-node text also matches CUDA_QUALIFIERS directly.
233
+ if (child.type === 'storage_class_specifier' || child.type === 'attribute_specifier') {
234
+ if (CUDA_QUALIFIERS.has(child.text)) qualifiers.push(child.text);
235
+ } else if (CUDA_QUALIFIERS.has(child.text)) {
236
+ qualifiers.push(child.text);
237
+ }
238
+ }
239
+ return qualifiers;
240
+ }
241
+
242
+ function findCudaParentClass(node: TreeSitterNode): string | null {
243
+ let current = node.parent;
244
+ while (current) {
245
+ if (current.type === 'field_declaration_list') {
246
+ const classNode = current.parent;
247
+ if (
248
+ classNode &&
249
+ (classNode.type === 'class_specifier' || classNode.type === 'struct_specifier')
250
+ ) {
251
+ const nameNode = classNode.childForFieldName('name');
252
+ return nameNode ? nameNode.text : null;
253
+ }
254
+ }
255
+ current = current.parent;
256
+ }
257
+ return null;
258
+ }
259
+
260
+ function extractCudaParameters(paramListNode: TreeSitterNode | null): SubDeclaration[] {
261
+ const params: SubDeclaration[] = [];
262
+ if (!paramListNode) return params;
263
+ for (let i = 0; i < paramListNode.childCount; i++) {
264
+ const param = paramListNode.child(i);
265
+ if (!param || param.type !== 'parameter_declaration') continue;
266
+ const nameNode = param.childForFieldName('declarator');
267
+ if (nameNode) {
268
+ const name =
269
+ nameNode.type === 'identifier'
270
+ ? nameNode.text
271
+ : (findChild(nameNode, 'identifier')?.text ?? nameNode.text);
272
+ params.push({ name, kind: 'parameter', line: param.startPosition.row + 1 });
273
+ }
274
+ }
275
+ return params;
276
+ }
277
+
278
+ function extractCudaClassFields(classNode: TreeSitterNode): SubDeclaration[] {
279
+ const fields: SubDeclaration[] = [];
280
+ const body =
281
+ classNode.childForFieldName('body') || findChild(classNode, 'field_declaration_list');
282
+ if (!body) return fields;
283
+ for (let i = 0; i < body.childCount; i++) {
284
+ const member = body.child(i);
285
+ if (!member || member.type !== 'field_declaration') continue;
286
+ const nameNode = member.childForFieldName('declarator');
287
+ if (nameNode) {
288
+ const name =
289
+ nameNode.type === 'identifier'
290
+ ? nameNode.text
291
+ : (findChild(nameNode, 'identifier')?.text ?? nameNode.text);
292
+ fields.push({
293
+ name,
294
+ kind: 'property',
295
+ line: member.startPosition.row + 1,
296
+ visibility: extractModifierVisibility(member),
297
+ });
298
+ }
299
+ }
300
+ return fields;
301
+ }
302
+
303
+ function extractCudaEnumEntries(enumNode: TreeSitterNode): SubDeclaration[] {
304
+ const entries: SubDeclaration[] = [];
305
+ const body = findChild(enumNode, 'enumerator_list');
306
+ if (!body) return entries;
307
+ for (let i = 0; i < body.childCount; i++) {
308
+ const member = body.child(i);
309
+ if (!member || member.type !== 'enumerator') continue;
310
+ const nameNode = member.childForFieldName('name');
311
+ if (nameNode) {
312
+ entries.push({ name: nameNode.text, kind: 'constant', line: member.startPosition.row + 1 });
313
+ }
314
+ }
315
+ return entries;
316
+ }
@@ -0,0 +1,252 @@
1
+ import type { ExtractorOutput, SubDeclaration, TreeSitterNode, TreeSitterTree } from '../types.js';
2
+ import { findChild, nodeEndLine } from './helpers.js';
3
+
4
+ /**
5
+ * Extract symbols from Erlang files.
6
+ *
7
+ * tree-sitter-erlang (WhatsApp) grammar notes:
8
+ * - module_attribute: -module(name).
9
+ * - record_decl: -record(name, {fields}).
10
+ * - fun_decl: contains function_clause children
11
+ * - function_clause: atom expr_args clause_body
12
+ * - call: function calls, with remote child for module:func
13
+ * - expr_args: parenthesized argument lists
14
+ */
15
+ export function extractErlangSymbols(tree: TreeSitterTree, _filePath: string): ExtractorOutput {
16
+ const ctx: ExtractorOutput = {
17
+ definitions: [],
18
+ calls: [],
19
+ imports: [],
20
+ classes: [],
21
+ exports: [],
22
+ typeMap: new Map(),
23
+ };
24
+
25
+ walkErlangNode(tree.rootNode, ctx);
26
+ return ctx;
27
+ }
28
+
29
+ function walkErlangNode(node: TreeSitterNode, ctx: ExtractorOutput): void {
30
+ switch (node.type) {
31
+ case 'module_attribute':
32
+ handleModuleAttr(node, ctx);
33
+ break;
34
+ case 'record_decl':
35
+ handleRecordDecl(node, ctx);
36
+ break;
37
+ case 'type_alias':
38
+ case 'opaque':
39
+ handleTypeAlias(node, ctx);
40
+ break;
41
+ case 'fun_decl':
42
+ handleFunDecl(node, ctx);
43
+ break;
44
+ case 'function_clause':
45
+ // Only handle if not inside fun_decl (fun_decl handles its own clauses)
46
+ if (node.parent?.type !== 'fun_decl') {
47
+ handleFunctionClause(node, ctx);
48
+ }
49
+ break;
50
+ case 'pp_define':
51
+ handleDefine(node, ctx);
52
+ break;
53
+ case 'pp_include':
54
+ case 'pp_include_lib':
55
+ handleInclude(node, ctx);
56
+ break;
57
+ case 'import_attribute':
58
+ handleImportAttr(node, ctx);
59
+ break;
60
+ case 'call':
61
+ handleCall(node, ctx);
62
+ break;
63
+ }
64
+
65
+ for (let i = 0; i < node.childCount; i++) {
66
+ const child = node.child(i);
67
+ if (child) walkErlangNode(child, ctx);
68
+ }
69
+ }
70
+
71
+ function handleModuleAttr(node: TreeSitterNode, ctx: ExtractorOutput): void {
72
+ // module_attribute: - module ( atom ) .
73
+ const nameNode = findChild(node, 'atom');
74
+ if (!nameNode) return;
75
+
76
+ ctx.definitions.push({
77
+ name: nameNode.text,
78
+ kind: 'module',
79
+ line: node.startPosition.row + 1,
80
+ endLine: nodeEndLine(node),
81
+ });
82
+ }
83
+
84
+ function handleRecordDecl(node: TreeSitterNode, ctx: ExtractorOutput): void {
85
+ // record_decl: - record ( atom , { record_field, ... } ) .
86
+ const nameNode = findChild(node, 'atom');
87
+ if (!nameNode) return;
88
+
89
+ const children: SubDeclaration[] = [];
90
+ for (let i = 0; i < node.childCount; i++) {
91
+ const child = node.child(i);
92
+ if (!child) continue;
93
+ if (child.type === 'record_field' || child.type === 'typed_record_field') {
94
+ const fieldName = findChild(child, 'atom');
95
+ if (fieldName) {
96
+ children.push({
97
+ name: fieldName.text,
98
+ kind: 'property',
99
+ line: child.startPosition.row + 1,
100
+ });
101
+ }
102
+ }
103
+ }
104
+
105
+ ctx.definitions.push({
106
+ name: nameNode.text,
107
+ kind: 'record',
108
+ line: node.startPosition.row + 1,
109
+ endLine: nodeEndLine(node),
110
+ children: children.length > 0 ? children : undefined,
111
+ });
112
+ }
113
+
114
+ function handleTypeAlias(node: TreeSitterNode, ctx: ExtractorOutput): void {
115
+ const nameNode = findChild(node, 'atom');
116
+ if (!nameNode) return;
117
+
118
+ ctx.definitions.push({
119
+ name: nameNode.text,
120
+ kind: 'type',
121
+ line: node.startPosition.row + 1,
122
+ endLine: nodeEndLine(node),
123
+ });
124
+ }
125
+
126
+ function handleFunDecl(node: TreeSitterNode, ctx: ExtractorOutput): void {
127
+ // fun_decl contains one or more function_clause children + dots
128
+ // Extract from the first function_clause
129
+ const clause = findChild(node, 'function_clause');
130
+ if (!clause) return;
131
+
132
+ handleFunctionClause(clause, ctx);
133
+ }
134
+
135
+ function handleFunctionClause(node: TreeSitterNode, ctx: ExtractorOutput): void {
136
+ // function_clause: atom expr_args clause_body
137
+ const nameNode = findChild(node, 'atom');
138
+ if (!nameNode) return;
139
+
140
+ // Don't duplicate if we already have this function
141
+ if (ctx.definitions.some((d) => d.name === nameNode.text && d.kind === 'function')) return;
142
+
143
+ const params = extractErlangParams(node);
144
+
145
+ ctx.definitions.push({
146
+ name: nameNode.text,
147
+ kind: 'function',
148
+ line: node.startPosition.row + 1,
149
+ endLine: nodeEndLine(node.parent?.type === 'fun_decl' ? node.parent : node),
150
+ children: params.length > 0 ? params : undefined,
151
+ visibility: 'public',
152
+ });
153
+ }
154
+
155
+ function extractErlangParams(clauseNode: TreeSitterNode): SubDeclaration[] {
156
+ const params: SubDeclaration[] = [];
157
+ const argsNode = findChild(clauseNode, 'expr_args');
158
+ if (!argsNode) return params;
159
+
160
+ for (let i = 0; i < argsNode.childCount; i++) {
161
+ const child = argsNode.child(i);
162
+ if (!child) continue;
163
+ if (child.type === 'var') {
164
+ params.push({ name: child.text, kind: 'parameter', line: child.startPosition.row + 1 });
165
+ }
166
+ if (child.type === 'atom') {
167
+ params.push({ name: child.text, kind: 'parameter', line: child.startPosition.row + 1 });
168
+ }
169
+ }
170
+ return params;
171
+ }
172
+
173
+ function handleDefine(node: TreeSitterNode, ctx: ExtractorOutput): void {
174
+ // pp_define: -define(NAME, value).
175
+ const nameNode =
176
+ findChild(node, 'var') || findChild(node, 'atom') || findChild(node, 'macro_lhs');
177
+ if (!nameNode) return;
178
+
179
+ const name =
180
+ nameNode.type === 'macro_lhs'
181
+ ? (findChild(nameNode, 'var')?.text ?? nameNode.text)
182
+ : nameNode.text;
183
+
184
+ ctx.definitions.push({
185
+ name,
186
+ kind: 'variable',
187
+ line: node.startPosition.row + 1,
188
+ endLine: nodeEndLine(node),
189
+ });
190
+ }
191
+
192
+ function handleInclude(node: TreeSitterNode, ctx: ExtractorOutput): void {
193
+ const strNode = findChild(node, 'string');
194
+ if (!strNode) return;
195
+
196
+ const source = strNode.text.replace(/^"|"$/g, '');
197
+ ctx.imports.push({
198
+ source,
199
+ names: ['include'],
200
+ line: node.startPosition.row + 1,
201
+ });
202
+ }
203
+
204
+ function handleImportAttr(node: TreeSitterNode, ctx: ExtractorOutput): void {
205
+ const moduleNode = findChild(node, 'atom');
206
+ if (!moduleNode) return;
207
+
208
+ const names: string[] = [];
209
+ // Find exported function names
210
+ for (let i = 0; i < node.childCount; i++) {
211
+ const child = node.child(i);
212
+ if (!child) continue;
213
+ if (child.type === 'fa') {
214
+ const fnName = findChild(child, 'atom');
215
+ if (fnName) names.push(fnName.text);
216
+ }
217
+ }
218
+
219
+ ctx.imports.push({
220
+ source: moduleNode.text,
221
+ names: names.length > 0 ? names : [moduleNode.text],
222
+ line: node.startPosition.row + 1,
223
+ });
224
+ }
225
+
226
+ function handleCall(node: TreeSitterNode, ctx: ExtractorOutput): void {
227
+ // call: first child is function ref (atom or remote), then expr_args
228
+ const funcNode = node.child(0);
229
+ if (!funcNode) return;
230
+
231
+ if (funcNode.type === 'atom' || funcNode.type === 'identifier') {
232
+ ctx.calls.push({ name: funcNode.text, line: node.startPosition.row + 1 });
233
+ } else if (funcNode.type === 'remote') {
234
+ // module:function — remote has atom : atom children
235
+ const atoms: string[] = [];
236
+ for (let i = 0; i < funcNode.childCount; i++) {
237
+ const child = funcNode.child(i);
238
+ if (child && (child.type === 'atom' || child.type === 'var')) {
239
+ atoms.push(child.text);
240
+ }
241
+ }
242
+ if (atoms.length >= 2) {
243
+ ctx.calls.push({
244
+ name: atoms[atoms.length - 1]!,
245
+ receiver: atoms.slice(0, -1).join(':'),
246
+ line: node.startPosition.row + 1,
247
+ });
248
+ } else if (atoms.length === 1) {
249
+ ctx.calls.push({ name: atoms[0]!, line: node.startPosition.row + 1 });
250
+ }
251
+ }
252
+ }