@optave/codegraph 3.4.0 → 3.5.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 (453) hide show
  1. package/README.md +23 -22
  2. package/dist/ast-analysis/engine.d.ts.map +1 -1
  3. package/dist/ast-analysis/engine.js +3 -9
  4. package/dist/ast-analysis/engine.js.map +1 -1
  5. package/dist/ast-analysis/rules/javascript.d.ts.map +1 -1
  6. package/dist/ast-analysis/rules/javascript.js +1 -0
  7. package/dist/ast-analysis/rules/javascript.js.map +1 -1
  8. package/dist/ast-analysis/shared.d.ts.map +1 -1
  9. package/dist/ast-analysis/shared.js +0 -1
  10. package/dist/ast-analysis/shared.js.map +1 -1
  11. package/dist/ast-analysis/visitors/ast-store-visitor.d.ts.map +1 -1
  12. package/dist/ast-analysis/visitors/ast-store-visitor.js +103 -35
  13. package/dist/ast-analysis/visitors/ast-store-visitor.js.map +1 -1
  14. package/dist/ast-analysis/visitors/cfg-conditionals.d.ts +5 -0
  15. package/dist/ast-analysis/visitors/cfg-conditionals.d.ts.map +1 -0
  16. package/dist/ast-analysis/visitors/cfg-conditionals.js +166 -0
  17. package/dist/ast-analysis/visitors/cfg-conditionals.js.map +1 -0
  18. package/dist/ast-analysis/visitors/cfg-loops.d.ts +7 -0
  19. package/dist/ast-analysis/visitors/cfg-loops.d.ts.map +1 -0
  20. package/dist/ast-analysis/visitors/cfg-loops.js +73 -0
  21. package/dist/ast-analysis/visitors/cfg-loops.js.map +1 -0
  22. package/dist/ast-analysis/visitors/cfg-shared.d.ts +56 -0
  23. package/dist/ast-analysis/visitors/cfg-shared.d.ts.map +1 -0
  24. package/dist/ast-analysis/visitors/cfg-shared.js +107 -0
  25. package/dist/ast-analysis/visitors/cfg-shared.js.map +1 -0
  26. package/dist/ast-analysis/visitors/cfg-try-catch.d.ts +4 -0
  27. package/dist/ast-analysis/visitors/cfg-try-catch.d.ts.map +1 -0
  28. package/dist/ast-analysis/visitors/cfg-try-catch.js +100 -0
  29. package/dist/ast-analysis/visitors/cfg-try-catch.js.map +1 -0
  30. package/dist/ast-analysis/visitors/cfg-visitor.d.ts +2 -2
  31. package/dist/ast-analysis/visitors/cfg-visitor.d.ts.map +1 -1
  32. package/dist/ast-analysis/visitors/cfg-visitor.js +11 -445
  33. package/dist/ast-analysis/visitors/cfg-visitor.js.map +1 -1
  34. package/dist/ast-analysis/visitors/complexity-visitor.d.ts.map +1 -1
  35. package/dist/ast-analysis/visitors/complexity-visitor.js.map +1 -1
  36. package/dist/ast-analysis/visitors/dataflow-visitor.d.ts.map +1 -1
  37. package/dist/ast-analysis/visitors/dataflow-visitor.js.map +1 -1
  38. package/dist/cli/commands/batch.d.ts.map +1 -1
  39. package/dist/cli/commands/batch.js +4 -3
  40. package/dist/cli/commands/batch.js.map +1 -1
  41. package/dist/cli/commands/branch-compare.js +1 -1
  42. package/dist/cli/commands/branch-compare.js.map +1 -1
  43. package/dist/cli/commands/build.js +1 -1
  44. package/dist/cli/commands/build.js.map +1 -1
  45. package/dist/cli/commands/info.d.ts.map +1 -1
  46. package/dist/cli/commands/info.js +1 -2
  47. package/dist/cli/commands/info.js.map +1 -1
  48. package/dist/cli/commands/path.d.ts.map +1 -1
  49. package/dist/cli/commands/path.js +7 -2
  50. package/dist/cli/commands/path.js.map +1 -1
  51. package/dist/cli/commands/plot.d.ts.map +1 -1
  52. package/dist/cli/commands/plot.js +2 -2
  53. package/dist/cli/commands/plot.js.map +1 -1
  54. package/dist/cli/commands/watch.js +1 -1
  55. package/dist/cli/commands/watch.js.map +1 -1
  56. package/dist/cli/index.js +2 -2
  57. package/dist/cli/index.js.map +1 -1
  58. package/dist/cli/shared/open-graph.d.ts +2 -2
  59. package/dist/cli/shared/open-graph.d.ts.map +1 -1
  60. package/dist/cli/shared/open-graph.js.map +1 -1
  61. package/dist/cli/types.d.ts +1 -1
  62. package/dist/cli/types.d.ts.map +1 -1
  63. package/dist/cli.js +2 -3
  64. package/dist/cli.js.map +1 -1
  65. package/dist/db/better-sqlite3.d.ts +3 -0
  66. package/dist/db/better-sqlite3.d.ts.map +1 -0
  67. package/dist/db/better-sqlite3.js +19 -0
  68. package/dist/db/better-sqlite3.js.map +1 -0
  69. package/dist/db/connection.d.ts +30 -2
  70. package/dist/db/connection.d.ts.map +1 -1
  71. package/dist/db/connection.js +167 -4
  72. package/dist/db/connection.js.map +1 -1
  73. package/dist/db/index.d.ts +2 -2
  74. package/dist/db/index.d.ts.map +1 -1
  75. package/dist/db/index.js +1 -1
  76. package/dist/db/index.js.map +1 -1
  77. package/dist/db/migrations.d.ts.map +1 -1
  78. package/dist/db/migrations.js +9 -0
  79. package/dist/db/migrations.js.map +1 -1
  80. package/dist/db/query-builder.d.ts +5 -5
  81. package/dist/db/query-builder.d.ts.map +1 -1
  82. package/dist/db/query-builder.js +20 -4
  83. package/dist/db/query-builder.js.map +1 -1
  84. package/dist/db/repository/index.d.ts +1 -0
  85. package/dist/db/repository/index.d.ts.map +1 -1
  86. package/dist/db/repository/index.js +1 -0
  87. package/dist/db/repository/index.js.map +1 -1
  88. package/dist/db/repository/native-repository.d.ts +58 -0
  89. package/dist/db/repository/native-repository.d.ts.map +1 -0
  90. package/dist/db/repository/native-repository.js +261 -0
  91. package/dist/db/repository/native-repository.js.map +1 -0
  92. package/dist/db/repository/nodes.d.ts +4 -4
  93. package/dist/db/repository/nodes.d.ts.map +1 -1
  94. package/dist/db/repository/nodes.js +6 -6
  95. package/dist/db/repository/nodes.js.map +1 -1
  96. package/dist/domain/analysis/brief.d.ts.map +1 -1
  97. package/dist/domain/analysis/brief.js +1 -3
  98. package/dist/domain/analysis/brief.js.map +1 -1
  99. package/dist/domain/analysis/context.d.ts.map +1 -1
  100. package/dist/domain/analysis/context.js +2 -4
  101. package/dist/domain/analysis/context.js.map +1 -1
  102. package/dist/domain/analysis/dependencies.d.ts +49 -0
  103. package/dist/domain/analysis/dependencies.d.ts.map +1 -1
  104. package/dist/domain/analysis/dependencies.js +145 -0
  105. package/dist/domain/analysis/dependencies.js.map +1 -1
  106. package/dist/domain/analysis/diff-impact.d.ts +76 -0
  107. package/dist/domain/analysis/diff-impact.d.ts.map +1 -0
  108. package/dist/domain/analysis/diff-impact.js +282 -0
  109. package/dist/domain/analysis/diff-impact.js.map +1 -0
  110. package/dist/domain/analysis/exports.d.ts.map +1 -1
  111. package/dist/domain/analysis/exports.js +0 -1
  112. package/dist/domain/analysis/exports.js.map +1 -1
  113. package/dist/domain/analysis/fn-impact.d.ts +66 -0
  114. package/dist/domain/analysis/fn-impact.d.ts.map +1 -0
  115. package/dist/domain/analysis/fn-impact.js +189 -0
  116. package/dist/domain/analysis/fn-impact.js.map +1 -0
  117. package/dist/domain/analysis/impact.d.ts +8 -148
  118. package/dist/domain/analysis/impact.d.ts.map +1 -1
  119. package/dist/domain/analysis/impact.js +8 -568
  120. package/dist/domain/analysis/impact.js.map +1 -1
  121. package/dist/domain/analysis/module-map.d.ts.map +1 -1
  122. package/dist/domain/analysis/module-map.js +1 -3
  123. package/dist/domain/analysis/module-map.js.map +1 -1
  124. package/dist/domain/graph/builder/context.d.ts +3 -3
  125. package/dist/domain/graph/builder/context.d.ts.map +1 -1
  126. package/dist/domain/graph/builder/context.js +1 -0
  127. package/dist/domain/graph/builder/context.js.map +1 -1
  128. package/dist/domain/graph/builder/helpers.d.ts +4 -5
  129. package/dist/domain/graph/builder/helpers.d.ts.map +1 -1
  130. package/dist/domain/graph/builder/helpers.js +1 -2
  131. package/dist/domain/graph/builder/helpers.js.map +1 -1
  132. package/dist/domain/graph/builder/incremental.d.ts +2 -3
  133. package/dist/domain/graph/builder/incremental.d.ts.map +1 -1
  134. package/dist/domain/graph/builder/incremental.js.map +1 -1
  135. package/dist/domain/graph/builder/pipeline.d.ts.map +1 -1
  136. package/dist/domain/graph/builder/pipeline.js +34 -6
  137. package/dist/domain/graph/builder/pipeline.js.map +1 -1
  138. package/dist/domain/graph/builder/stages/build-edges.d.ts.map +1 -1
  139. package/dist/domain/graph/builder/stages/build-edges.js +113 -15
  140. package/dist/domain/graph/builder/stages/build-edges.js.map +1 -1
  141. package/dist/domain/graph/builder/stages/build-structure.d.ts.map +1 -1
  142. package/dist/domain/graph/builder/stages/build-structure.js +186 -62
  143. package/dist/domain/graph/builder/stages/build-structure.js.map +1 -1
  144. package/dist/domain/graph/builder/stages/collect-files.d.ts.map +1 -1
  145. package/dist/domain/graph/builder/stages/collect-files.js +71 -7
  146. package/dist/domain/graph/builder/stages/collect-files.js.map +1 -1
  147. package/dist/domain/graph/builder/stages/detect-changes.d.ts.map +1 -1
  148. package/dist/domain/graph/builder/stages/detect-changes.js +42 -20
  149. package/dist/domain/graph/builder/stages/detect-changes.js.map +1 -1
  150. package/dist/domain/graph/builder/stages/finalize.d.ts.map +1 -1
  151. package/dist/domain/graph/builder/stages/finalize.js +111 -64
  152. package/dist/domain/graph/builder/stages/finalize.js.map +1 -1
  153. package/dist/domain/graph/builder/stages/insert-nodes.d.ts.map +1 -1
  154. package/dist/domain/graph/builder/stages/insert-nodes.js +104 -9
  155. package/dist/domain/graph/builder/stages/insert-nodes.js.map +1 -1
  156. package/dist/domain/graph/builder/stages/resolve-imports.d.ts.map +1 -1
  157. package/dist/domain/graph/builder/stages/resolve-imports.js +58 -11
  158. package/dist/domain/graph/builder/stages/resolve-imports.js.map +1 -1
  159. package/dist/domain/graph/cycles.js +2 -2
  160. package/dist/domain/graph/cycles.js.map +1 -1
  161. package/dist/domain/graph/resolve.d.ts.map +1 -1
  162. package/dist/domain/graph/resolve.js +10 -8
  163. package/dist/domain/graph/resolve.js.map +1 -1
  164. package/dist/domain/graph/watcher.d.ts.map +1 -1
  165. package/dist/domain/graph/watcher.js +1 -3
  166. package/dist/domain/graph/watcher.js.map +1 -1
  167. package/dist/domain/parser.d.ts.map +1 -1
  168. package/dist/domain/parser.js +12 -12
  169. package/dist/domain/parser.js.map +1 -1
  170. package/dist/domain/queries.d.ts +3 -2
  171. package/dist/domain/queries.d.ts.map +1 -1
  172. package/dist/domain/queries.js +3 -2
  173. package/dist/domain/queries.js.map +1 -1
  174. package/dist/domain/search/generator.d.ts.map +1 -1
  175. package/dist/domain/search/generator.js.map +1 -1
  176. package/dist/extractors/csharp.js +2 -2
  177. package/dist/extractors/csharp.js.map +1 -1
  178. package/dist/extractors/go.js +2 -2
  179. package/dist/extractors/go.js.map +1 -1
  180. package/dist/extractors/helpers.d.ts +5 -0
  181. package/dist/extractors/helpers.d.ts.map +1 -1
  182. package/dist/extractors/helpers.js +5 -0
  183. package/dist/extractors/helpers.js.map +1 -1
  184. package/dist/extractors/javascript.js +111 -98
  185. package/dist/extractors/javascript.js.map +1 -1
  186. package/dist/extractors/php.js +2 -2
  187. package/dist/extractors/php.js.map +1 -1
  188. package/dist/extractors/python.js +2 -2
  189. package/dist/extractors/python.js.map +1 -1
  190. package/dist/extractors/rust.js +4 -3
  191. package/dist/extractors/rust.js.map +1 -1
  192. package/dist/features/ast.d.ts +14 -1
  193. package/dist/features/ast.d.ts.map +1 -1
  194. package/dist/features/ast.js +38 -1
  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 +1 -2
  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 +5 -4
  201. package/dist/features/branch-compare.js.map +1 -1
  202. package/dist/features/cfg.d.ts.map +1 -1
  203. package/dist/features/cfg.js +2 -4
  204. package/dist/features/cfg.js.map +1 -1
  205. package/dist/features/cochange.js +4 -4
  206. package/dist/features/cochange.js.map +1 -1
  207. package/dist/features/communities.js +4 -4
  208. package/dist/features/communities.js.map +1 -1
  209. package/dist/features/complexity-query.d.ts +37 -0
  210. package/dist/features/complexity-query.d.ts.map +1 -0
  211. package/dist/features/complexity-query.js +263 -0
  212. package/dist/features/complexity-query.js.map +1 -0
  213. package/dist/features/complexity.d.ts +2 -30
  214. package/dist/features/complexity.d.ts.map +1 -1
  215. package/dist/features/complexity.js +7 -261
  216. package/dist/features/complexity.js.map +1 -1
  217. package/dist/features/dataflow.d.ts.map +1 -1
  218. package/dist/features/dataflow.js +8 -24
  219. package/dist/features/dataflow.js.map +1 -1
  220. package/dist/features/export.d.ts +7 -8
  221. package/dist/features/export.d.ts.map +1 -1
  222. package/dist/features/export.js.map +1 -1
  223. package/dist/features/flow.d.ts.map +1 -1
  224. package/dist/features/flow.js.map +1 -1
  225. package/dist/features/graph-enrichment.d.ts.map +1 -1
  226. package/dist/features/graph-enrichment.js +1 -3
  227. package/dist/features/graph-enrichment.js.map +1 -1
  228. package/dist/features/manifesto.js +8 -8
  229. package/dist/features/manifesto.js.map +1 -1
  230. package/dist/features/snapshot.js +2 -2
  231. package/dist/features/snapshot.js.map +1 -1
  232. package/dist/features/structure-query.d.ts +76 -0
  233. package/dist/features/structure-query.d.ts.map +1 -0
  234. package/dist/features/structure-query.js +245 -0
  235. package/dist/features/structure-query.js.map +1 -0
  236. package/dist/features/structure.d.ts +12 -67
  237. package/dist/features/structure.d.ts.map +1 -1
  238. package/dist/features/structure.js +188 -244
  239. package/dist/features/structure.js.map +1 -1
  240. package/dist/features/triage.js +2 -2
  241. package/dist/features/triage.js.map +1 -1
  242. package/dist/graph/algorithms/leiden/adapter.d.ts.map +1 -1
  243. package/dist/graph/algorithms/leiden/adapter.js +2 -9
  244. package/dist/graph/algorithms/leiden/adapter.js.map +1 -1
  245. package/dist/graph/classifiers/roles.d.ts +5 -1
  246. package/dist/graph/classifiers/roles.d.ts.map +1 -1
  247. package/dist/graph/classifiers/roles.js +20 -12
  248. package/dist/graph/classifiers/roles.js.map +1 -1
  249. package/dist/index.d.ts +1 -0
  250. package/dist/index.d.ts.map +1 -1
  251. package/dist/index.js.map +1 -1
  252. package/dist/infrastructure/config.d.ts.map +1 -1
  253. package/dist/infrastructure/config.js +12 -11
  254. package/dist/infrastructure/config.js.map +1 -1
  255. package/dist/infrastructure/native.d.ts.map +1 -1
  256. package/dist/infrastructure/native.js +7 -3
  257. package/dist/infrastructure/native.js.map +1 -1
  258. package/dist/infrastructure/registry.d.ts.map +1 -1
  259. package/dist/infrastructure/registry.js +1 -1
  260. package/dist/infrastructure/registry.js.map +1 -1
  261. package/dist/infrastructure/update-check.js +3 -3
  262. package/dist/infrastructure/update-check.js.map +1 -1
  263. package/dist/mcp/server.d.ts.map +1 -1
  264. package/dist/mcp/server.js +4 -17
  265. package/dist/mcp/server.js.map +1 -1
  266. package/dist/mcp/tool-registry.d.ts.map +1 -1
  267. package/dist/mcp/tool-registry.js +9 -4
  268. package/dist/mcp/tool-registry.js.map +1 -1
  269. package/dist/mcp/tools/audit.js +1 -1
  270. package/dist/mcp/tools/audit.js.map +1 -1
  271. package/dist/mcp/tools/cfg.js +1 -1
  272. package/dist/mcp/tools/cfg.js.map +1 -1
  273. package/dist/mcp/tools/check.js +2 -2
  274. package/dist/mcp/tools/check.js.map +1 -1
  275. package/dist/mcp/tools/dataflow.js +2 -2
  276. package/dist/mcp/tools/dataflow.js.map +1 -1
  277. package/dist/mcp/tools/export-graph.js +1 -1
  278. package/dist/mcp/tools/export-graph.js.map +1 -1
  279. package/dist/mcp/tools/index.d.ts.map +1 -1
  280. package/dist/mcp/tools/index.js.map +1 -1
  281. package/dist/mcp/tools/path.d.ts +1 -0
  282. package/dist/mcp/tools/path.d.ts.map +1 -1
  283. package/dist/mcp/tools/path.js +9 -0
  284. package/dist/mcp/tools/path.js.map +1 -1
  285. package/dist/mcp/tools/query.js +1 -1
  286. package/dist/mcp/tools/query.js.map +1 -1
  287. package/dist/mcp/tools/semantic-search.js +1 -1
  288. package/dist/mcp/tools/semantic-search.js.map +1 -1
  289. package/dist/mcp/tools/sequence.js +1 -1
  290. package/dist/mcp/tools/sequence.js.map +1 -1
  291. package/dist/mcp/tools/symbol-children.js +1 -1
  292. package/dist/mcp/tools/symbol-children.js.map +1 -1
  293. package/dist/mcp/tools/triage.js +1 -1
  294. package/dist/mcp/tools/triage.js.map +1 -1
  295. package/dist/presentation/audit.d.ts.map +1 -1
  296. package/dist/presentation/audit.js +0 -1
  297. package/dist/presentation/audit.js.map +1 -1
  298. package/dist/presentation/diff-impact-mermaid.d.ts +11 -0
  299. package/dist/presentation/diff-impact-mermaid.d.ts.map +1 -0
  300. package/dist/presentation/diff-impact-mermaid.js +105 -0
  301. package/dist/presentation/diff-impact-mermaid.js.map +1 -0
  302. package/dist/presentation/flow.d.ts.map +1 -1
  303. package/dist/presentation/flow.js +0 -2
  304. package/dist/presentation/flow.js.map +1 -1
  305. package/dist/presentation/manifesto.d.ts.map +1 -1
  306. package/dist/presentation/manifesto.js +0 -1
  307. package/dist/presentation/manifesto.js.map +1 -1
  308. package/dist/presentation/queries-cli/inspect.d.ts.map +1 -1
  309. package/dist/presentation/queries-cli/inspect.js.map +1 -1
  310. package/dist/presentation/queries-cli/path.d.ts.map +1 -1
  311. package/dist/presentation/queries-cli/path.js +45 -1
  312. package/dist/presentation/queries-cli/path.js.map +1 -1
  313. package/dist/presentation/result-formatter.d.ts.map +1 -1
  314. package/dist/presentation/result-formatter.js +1 -3
  315. package/dist/presentation/result-formatter.js.map +1 -1
  316. package/dist/presentation/sequence.d.ts.map +1 -1
  317. package/dist/presentation/sequence.js +0 -1
  318. package/dist/presentation/sequence.js.map +1 -1
  319. package/dist/presentation/structure.d.ts.map +1 -1
  320. package/dist/presentation/structure.js.map +1 -1
  321. package/dist/presentation/triage.d.ts.map +1 -1
  322. package/dist/presentation/triage.js +0 -1
  323. package/dist/presentation/triage.js.map +1 -1
  324. package/dist/shared/constants.d.ts +9 -3
  325. package/dist/shared/constants.d.ts.map +1 -1
  326. package/dist/shared/constants.js +6 -3
  327. package/dist/shared/constants.js.map +1 -1
  328. package/dist/shared/errors.d.ts +2 -0
  329. package/dist/shared/errors.d.ts.map +1 -1
  330. package/dist/shared/errors.js +4 -0
  331. package/dist/shared/errors.js.map +1 -1
  332. package/dist/shared/version.d.ts +2 -0
  333. package/dist/shared/version.d.ts.map +1 -0
  334. package/dist/shared/version.js +5 -0
  335. package/dist/shared/version.js.map +1 -0
  336. package/dist/types.d.ts +230 -2
  337. package/dist/types.d.ts.map +1 -1
  338. package/package.json +62 -11
  339. package/src/ast-analysis/engine.ts +3 -9
  340. package/src/ast-analysis/rules/javascript.ts +1 -0
  341. package/src/ast-analysis/shared.ts +0 -1
  342. package/src/ast-analysis/visitors/ast-store-visitor.ts +102 -33
  343. package/src/ast-analysis/visitors/cfg-conditionals.ts +227 -0
  344. package/src/ast-analysis/visitors/cfg-loops.ts +136 -0
  345. package/src/ast-analysis/visitors/cfg-shared.ts +196 -0
  346. package/src/ast-analysis/visitors/cfg-try-catch.ts +142 -0
  347. package/src/ast-analysis/visitors/cfg-visitor.ts +34 -655
  348. package/src/ast-analysis/visitors/complexity-visitor.ts +0 -1
  349. package/src/ast-analysis/visitors/dataflow-visitor.ts +0 -1
  350. package/src/cli/commands/batch.ts +4 -3
  351. package/src/cli/commands/branch-compare.ts +1 -1
  352. package/src/cli/commands/build.ts +1 -1
  353. package/src/cli/commands/info.ts +1 -2
  354. package/src/cli/commands/path.ts +7 -2
  355. package/src/cli/commands/plot.ts +2 -2
  356. package/src/cli/commands/watch.ts +1 -1
  357. package/src/cli/index.ts +2 -2
  358. package/src/cli/shared/open-graph.ts +2 -2
  359. package/src/cli/types.ts +1 -1
  360. package/src/cli.ts +2 -3
  361. package/src/db/better-sqlite3.ts +20 -0
  362. package/src/db/connection.ts +191 -16
  363. package/src/db/index.ts +5 -1
  364. package/src/db/migrations.ts +9 -0
  365. package/src/db/query-builder.ts +30 -5
  366. package/src/db/repository/index.ts +1 -0
  367. package/src/db/repository/native-repository.ts +361 -0
  368. package/src/db/repository/nodes.ts +7 -3
  369. package/src/domain/analysis/brief.ts +0 -1
  370. package/src/domain/analysis/context.ts +2 -6
  371. package/src/domain/analysis/dependencies.ts +165 -0
  372. package/src/domain/analysis/diff-impact.ts +354 -0
  373. package/src/domain/analysis/exports.ts +0 -2
  374. package/src/domain/analysis/fn-impact.ts +241 -0
  375. package/src/domain/analysis/impact.ts +8 -718
  376. package/src/domain/analysis/module-map.ts +1 -5
  377. package/src/domain/graph/builder/context.ts +4 -2
  378. package/src/domain/graph/builder/helpers.ts +14 -11
  379. package/src/domain/graph/builder/incremental.ts +33 -28
  380. package/src/domain/graph/builder/pipeline.ts +37 -5
  381. package/src/domain/graph/builder/stages/build-edges.ts +131 -20
  382. package/src/domain/graph/builder/stages/build-structure.ts +245 -80
  383. package/src/domain/graph/builder/stages/collect-files.ts +84 -7
  384. package/src/domain/graph/builder/stages/detect-changes.ts +49 -32
  385. package/src/domain/graph/builder/stages/finalize.ts +132 -84
  386. package/src/domain/graph/builder/stages/insert-nodes.ts +141 -18
  387. package/src/domain/graph/builder/stages/resolve-imports.ts +75 -10
  388. package/src/domain/graph/cycles.ts +2 -2
  389. package/src/domain/graph/resolve.ts +14 -8
  390. package/src/domain/graph/watcher.ts +2 -4
  391. package/src/domain/parser.ts +12 -13
  392. package/src/domain/queries.ts +2 -2
  393. package/src/domain/search/generator.ts +3 -4
  394. package/src/extractors/csharp.ts +2 -2
  395. package/src/extractors/go.ts +2 -2
  396. package/src/extractors/helpers.ts +6 -0
  397. package/src/extractors/javascript.ts +112 -97
  398. package/src/extractors/php.ts +2 -2
  399. package/src/extractors/python.ts +2 -2
  400. package/src/extractors/rust.ts +4 -3
  401. package/src/features/ast.ts +66 -1
  402. package/src/features/audit.ts +1 -2
  403. package/src/features/branch-compare.ts +6 -10
  404. package/src/features/cfg.ts +2 -4
  405. package/src/features/cochange.ts +4 -4
  406. package/src/features/communities.ts +4 -4
  407. package/src/features/complexity-query.ts +370 -0
  408. package/src/features/complexity.ts +6 -365
  409. package/src/features/dataflow.ts +48 -70
  410. package/src/features/export.ts +12 -16
  411. package/src/features/flow.ts +0 -1
  412. package/src/features/graph-enrichment.ts +1 -3
  413. package/src/features/manifesto.ts +8 -8
  414. package/src/features/snapshot.ts +3 -3
  415. package/src/features/structure-query.ts +387 -0
  416. package/src/features/structure.ts +231 -376
  417. package/src/features/triage.ts +2 -2
  418. package/src/graph/algorithms/leiden/adapter.ts +2 -9
  419. package/src/graph/classifiers/roles.ts +22 -13
  420. package/src/index.ts +1 -0
  421. package/src/infrastructure/config.ts +12 -13
  422. package/src/infrastructure/native.ts +7 -3
  423. package/src/infrastructure/registry.ts +1 -1
  424. package/src/infrastructure/update-check.ts +3 -3
  425. package/src/mcp/server.ts +4 -20
  426. package/src/mcp/tool-registry.ts +11 -4
  427. package/src/mcp/tools/audit.ts +1 -1
  428. package/src/mcp/tools/cfg.ts +1 -1
  429. package/src/mcp/tools/check.ts +2 -2
  430. package/src/mcp/tools/dataflow.ts +2 -2
  431. package/src/mcp/tools/export-graph.ts +1 -1
  432. package/src/mcp/tools/index.ts +0 -1
  433. package/src/mcp/tools/path.ts +10 -0
  434. package/src/mcp/tools/query.ts +1 -1
  435. package/src/mcp/tools/semantic-search.ts +1 -1
  436. package/src/mcp/tools/sequence.ts +1 -1
  437. package/src/mcp/tools/symbol-children.ts +1 -1
  438. package/src/mcp/tools/triage.ts +1 -1
  439. package/src/presentation/audit.ts +0 -1
  440. package/src/presentation/diff-impact-mermaid.ts +127 -0
  441. package/src/presentation/flow.ts +0 -2
  442. package/src/presentation/manifesto.ts +0 -1
  443. package/src/presentation/queries-cli/inspect.ts +0 -1
  444. package/src/presentation/queries-cli/path.ts +71 -1
  445. package/src/presentation/result-formatter.ts +0 -1
  446. package/src/presentation/sequence.ts +0 -1
  447. package/src/presentation/structure.ts +0 -12
  448. package/src/presentation/triage.ts +0 -1
  449. package/src/shared/constants.ts +33 -19
  450. package/src/shared/errors.ts +5 -0
  451. package/src/shared/version.ts +10 -0
  452. package/src/types.ts +277 -10
  453. package/src/vendor.d.ts +0 -39
@@ -237,6 +237,7 @@ export const dataflow: DataflowRulesConfig = makeDataflowRules({
237
237
  // ─── AST Node Types ───────────────────────────────────────────────────────
238
238
 
239
239
  export const astTypes: Record<string, string> | null = {
240
+ call_expression: 'call',
240
241
  new_expression: 'new',
241
242
  throw_statement: 'throw',
242
243
  await_expression: 'await',
@@ -176,7 +176,6 @@ export function findFunctionNode(
176
176
  }
177
177
 
178
178
  for (let i = 0; i < node.childCount; i++) {
179
- // biome-ignore lint/style/noNonNullAssertion: tree-sitter child(i) within childCount is always non-null
180
179
  search(node.child(i)!);
181
180
  }
182
181
  }
@@ -44,6 +44,14 @@ function extractExpressionText(node: TreeSitterNode): string | null {
44
44
  return truncate(node.text);
45
45
  }
46
46
 
47
+ function extractCallName(node: TreeSitterNode): string {
48
+ for (const field of ['function', 'method', 'name']) {
49
+ const fn = node.childForFieldName(field);
50
+ if (fn) return fn.text;
51
+ }
52
+ return node.text?.split('(')[0] || '?';
53
+ }
54
+
47
55
  function extractName(kind: string, node: TreeSitterNode): string | null {
48
56
  if (kind === 'throw') {
49
57
  for (let i = 0; i < node.childCount; i++) {
@@ -102,6 +110,93 @@ export function createAstStoreVisitor(
102
110
  return nodeIdMap.get(`${parentDef.name}|${parentDef.kind}|${parentDef.line}`) || null;
103
111
  }
104
112
 
113
+ /** Recursively walk a subtree collecting AST nodes — used for arguments-only traversal. */
114
+ function walkSubtree(node: TreeSitterNode | null): void {
115
+ if (!node) return;
116
+ if (matched.has(node.id)) return;
117
+
118
+ const kind = astTypeMap[node.type];
119
+ if (kind === 'call') {
120
+ // Capture this call and recurse only into its arguments
121
+ collectNode(node, kind);
122
+ walkCallArguments(node);
123
+ return;
124
+ }
125
+ if (kind) {
126
+ collectNode(node, kind);
127
+ if (kind !== 'string' && kind !== 'regex') return; // skipChildren for non-leaf kinds
128
+ }
129
+ for (let i = 0; i < node.childCount; i++) {
130
+ walkSubtree(node.child(i));
131
+ }
132
+ }
133
+
134
+ /**
135
+ * Recurse into only the arguments of a call node — mirrors the native engine's
136
+ * strategy that prevents double-counting nested calls in the function field
137
+ * (e.g. chained calls like `a().b()`).
138
+ */
139
+ function walkCallArguments(callNode: TreeSitterNode): void {
140
+ // Try field-based lookup first, fall back to kind-based matching
141
+ const argsNode =
142
+ callNode.childForFieldName('arguments') ??
143
+ findChildByKind(callNode, ['arguments', 'argument_list', 'method_arguments']);
144
+ if (!argsNode) return;
145
+ for (let i = 0; i < argsNode.childCount; i++) {
146
+ walkSubtree(argsNode.child(i));
147
+ }
148
+ }
149
+
150
+ function findChildByKind(node: TreeSitterNode, kinds: string[]): TreeSitterNode | null {
151
+ for (let i = 0; i < node.childCount; i++) {
152
+ const child = node.child(i);
153
+ if (child && kinds.includes(child.type)) return child;
154
+ }
155
+ return null;
156
+ }
157
+
158
+ function collectNode(node: TreeSitterNode, kind: string): void {
159
+ if (matched.has(node.id)) return;
160
+
161
+ const line = node.startPosition.row + 1;
162
+ let name: string | null | undefined;
163
+ let text: string | null = null;
164
+
165
+ if (kind === 'call') {
166
+ name = extractCallName(node);
167
+ text = truncate(node.text);
168
+ } else if (kind === 'new') {
169
+ name = extractNewName(node);
170
+ text = truncate(node.text);
171
+ } else if (kind === 'throw') {
172
+ name = extractName('throw', node);
173
+ text = extractExpressionText(node);
174
+ } else if (kind === 'await') {
175
+ name = extractName('await', node);
176
+ text = extractExpressionText(node);
177
+ } else if (kind === 'string') {
178
+ const content = node.text?.replace(/^['"`]|['"`]$/g, '') || '';
179
+ if (content.length < 2) return;
180
+ name = truncate(content, 100);
181
+ text = truncate(node.text);
182
+ } else if (kind === 'regex') {
183
+ name = node.text || '?';
184
+ text = truncate(node.text);
185
+ }
186
+
187
+ rows.push({
188
+ file: relPath,
189
+ line,
190
+ kind,
191
+ name,
192
+ text,
193
+ receiver: null,
194
+ parentNodeId: resolveParentNodeId(line),
195
+ });
196
+
197
+ matched.add(node.id);
198
+ }
199
+
105
200
  return {
106
201
  name: 'ast-store',
107
202
 
@@ -111,40 +206,14 @@ export function createAstStoreVisitor(
111
206
  const kind = astTypeMap[node.type];
112
207
  if (!kind) return;
113
208
 
114
- const line = node.startPosition.row + 1;
115
- let name: string | null | undefined;
116
- let text: string | null = null;
117
-
118
- if (kind === 'new') {
119
- name = extractNewName(node);
120
- text = truncate(node.text);
121
- } else if (kind === 'throw') {
122
- name = extractName('throw', node);
123
- text = extractExpressionText(node);
124
- } else if (kind === 'await') {
125
- name = extractName('await', node);
126
- text = extractExpressionText(node);
127
- } else if (kind === 'string') {
128
- const content = node.text?.replace(/^['"`]|['"`]$/g, '') || '';
129
- if (content.length < 2) return;
130
- name = truncate(content, 100);
131
- text = truncate(node.text);
132
- } else if (kind === 'regex') {
133
- name = node.text || '?';
134
- text = truncate(node.text);
135
- }
209
+ collectNode(node, kind);
136
210
 
137
- rows.push({
138
- file: relPath,
139
- line,
140
- kind,
141
- name,
142
- text,
143
- receiver: null,
144
- parentNodeId: resolveParentNodeId(line),
145
- });
146
-
147
- matched.add(node.id);
211
+ if (kind === 'call') {
212
+ // Mirror native: skip full subtree, recurse only into arguments.
213
+ // Prevents double-counting chained calls like service.getUser().getName().
214
+ walkCallArguments(node);
215
+ return { skipChildren: true };
216
+ }
148
217
 
149
218
  if (kind !== 'string' && kind !== 'regex') {
150
219
  return { skipChildren: true };
@@ -0,0 +1,227 @@
1
+ import type { TreeSitterNode } from '../../types.js';
2
+ import type {
3
+ AnyRules,
4
+ CfgBlockInternal,
5
+ FuncState,
6
+ LoopCtx,
7
+ ProcessStatementsFn,
8
+ } from './cfg-shared.js';
9
+ import { getBodyStatements, isCaseNode, isIfNode, nn } from './cfg-shared.js';
10
+
11
+ export function processIf(
12
+ ifStmt: TreeSitterNode,
13
+ currentBlock: CfgBlockInternal,
14
+ S: FuncState,
15
+ cfgRules: AnyRules,
16
+ processStatements: ProcessStatementsFn,
17
+ ): CfgBlockInternal {
18
+ currentBlock.endLine = ifStmt.startPosition.row + 1;
19
+
20
+ const condBlock = S.makeBlock(
21
+ 'condition',
22
+ ifStmt.startPosition.row + 1,
23
+ ifStmt.startPosition.row + 1,
24
+ 'if',
25
+ );
26
+ S.addEdge(currentBlock, condBlock, 'fallthrough');
27
+
28
+ const joinBlock = S.makeBlock('body');
29
+
30
+ const consequentField = cfgRules.ifConsequentField || 'consequence';
31
+ const consequent = ifStmt.childForFieldName(consequentField);
32
+ const trueBlock = S.makeBlock('branch_true', null, null, 'then');
33
+ S.addEdge(condBlock, trueBlock, 'branch_true');
34
+ const trueStmts = getBodyStatements(consequent, cfgRules);
35
+ const trueEnd = processStatements(trueStmts, trueBlock, S, cfgRules);
36
+ if (trueEnd) {
37
+ S.addEdge(trueEnd, joinBlock, 'fallthrough');
38
+ }
39
+
40
+ if (cfgRules.elifNode) {
41
+ processElifSiblings(ifStmt, condBlock, joinBlock, S, cfgRules, processStatements);
42
+ } else {
43
+ processAlternative(ifStmt, condBlock, joinBlock, S, cfgRules, processStatements);
44
+ }
45
+
46
+ return joinBlock;
47
+ }
48
+
49
+ function processAlternative(
50
+ ifStmt: TreeSitterNode,
51
+ condBlock: CfgBlockInternal,
52
+ joinBlock: CfgBlockInternal,
53
+ S: FuncState,
54
+ cfgRules: AnyRules,
55
+ processStatements: ProcessStatementsFn,
56
+ ): void {
57
+ const alternative = ifStmt.childForFieldName('alternative');
58
+ if (!alternative) {
59
+ S.addEdge(condBlock, joinBlock, 'branch_false');
60
+ return;
61
+ }
62
+
63
+ if (cfgRules.elseViaAlternative && alternative.type !== cfgRules.elseClause) {
64
+ if (isIfNode(alternative.type, cfgRules)) {
65
+ const falseBlock = S.makeBlock('branch_false', null, null, 'else-if');
66
+ S.addEdge(condBlock, falseBlock, 'branch_false');
67
+ const elseIfEnd = processIf(alternative, falseBlock, S, cfgRules, processStatements);
68
+ if (elseIfEnd) S.addEdge(elseIfEnd, joinBlock, 'fallthrough');
69
+ } else {
70
+ const falseBlock = S.makeBlock('branch_false', null, null, 'else');
71
+ S.addEdge(condBlock, falseBlock, 'branch_false');
72
+ const falseStmts = getBodyStatements(alternative, cfgRules);
73
+ const falseEnd = processStatements(falseStmts, falseBlock, S, cfgRules);
74
+ if (falseEnd) S.addEdge(falseEnd, joinBlock, 'fallthrough');
75
+ }
76
+ } else if (alternative.type === cfgRules.elseClause) {
77
+ const elseChildren: TreeSitterNode[] = [];
78
+ for (let i = 0; i < alternative.namedChildCount; i++) {
79
+ elseChildren.push(nn(alternative.namedChild(i)));
80
+ }
81
+ const firstChild = elseChildren[0];
82
+ if (elseChildren.length === 1 && firstChild && isIfNode(firstChild.type, cfgRules)) {
83
+ const falseBlock = S.makeBlock('branch_false', null, null, 'else-if');
84
+ S.addEdge(condBlock, falseBlock, 'branch_false');
85
+ const elseIfEnd = processIf(firstChild, falseBlock, S, cfgRules, processStatements);
86
+ if (elseIfEnd) S.addEdge(elseIfEnd, joinBlock, 'fallthrough');
87
+ } else {
88
+ const falseBlock = S.makeBlock('branch_false', null, null, 'else');
89
+ S.addEdge(condBlock, falseBlock, 'branch_false');
90
+ const falseEnd = processStatements(elseChildren, falseBlock, S, cfgRules);
91
+ if (falseEnd) S.addEdge(falseEnd, joinBlock, 'fallthrough');
92
+ }
93
+ }
94
+ }
95
+
96
+ function processElifSiblings(
97
+ ifStmt: TreeSitterNode,
98
+ firstCondBlock: CfgBlockInternal,
99
+ joinBlock: CfgBlockInternal,
100
+ S: FuncState,
101
+ cfgRules: AnyRules,
102
+ processStatements: ProcessStatementsFn,
103
+ ): void {
104
+ let lastCondBlock = firstCondBlock;
105
+ let foundElse = false;
106
+
107
+ for (let i = 0; i < ifStmt.namedChildCount; i++) {
108
+ const child = nn(ifStmt.namedChild(i));
109
+
110
+ if (child.type === cfgRules.elifNode) {
111
+ const elifCondBlock = S.makeBlock(
112
+ 'condition',
113
+ child.startPosition.row + 1,
114
+ child.startPosition.row + 1,
115
+ 'else-if',
116
+ );
117
+ S.addEdge(lastCondBlock, elifCondBlock, 'branch_false');
118
+
119
+ const elifConsequentField = cfgRules.ifConsequentField || 'consequence';
120
+ const elifConsequent = child.childForFieldName(elifConsequentField);
121
+ const elifTrueBlock = S.makeBlock('branch_true', null, null, 'then');
122
+ S.addEdge(elifCondBlock, elifTrueBlock, 'branch_true');
123
+ const elifTrueStmts = getBodyStatements(elifConsequent, cfgRules);
124
+ const elifTrueEnd = processStatements(elifTrueStmts, elifTrueBlock, S, cfgRules);
125
+ if (elifTrueEnd) S.addEdge(elifTrueEnd, joinBlock, 'fallthrough');
126
+
127
+ lastCondBlock = elifCondBlock;
128
+ } else if (child.type === cfgRules.elseClause) {
129
+ const elseBlock = S.makeBlock('branch_false', null, null, 'else');
130
+ S.addEdge(lastCondBlock, elseBlock, 'branch_false');
131
+
132
+ const elseBody = child.childForFieldName('body');
133
+ let elseStmts: TreeSitterNode[];
134
+ if (elseBody) {
135
+ elseStmts = getBodyStatements(elseBody, cfgRules);
136
+ } else {
137
+ elseStmts = [];
138
+ for (let j = 0; j < child.namedChildCount; j++) {
139
+ elseStmts.push(nn(child.namedChild(j)));
140
+ }
141
+ }
142
+ const elseEnd = processStatements(elseStmts, elseBlock, S, cfgRules);
143
+ if (elseEnd) S.addEdge(elseEnd, joinBlock, 'fallthrough');
144
+
145
+ foundElse = true;
146
+ }
147
+ }
148
+
149
+ if (!foundElse) {
150
+ S.addEdge(lastCondBlock, joinBlock, 'branch_false');
151
+ }
152
+ }
153
+
154
+ export function processSwitch(
155
+ switchStmt: TreeSitterNode,
156
+ currentBlock: CfgBlockInternal,
157
+ S: FuncState,
158
+ cfgRules: AnyRules,
159
+ processStatements: ProcessStatementsFn,
160
+ ): CfgBlockInternal {
161
+ currentBlock.endLine = switchStmt.startPosition.row + 1;
162
+
163
+ const switchHeader = S.makeBlock(
164
+ 'condition',
165
+ switchStmt.startPosition.row + 1,
166
+ switchStmt.startPosition.row + 1,
167
+ 'switch',
168
+ );
169
+ S.addEdge(currentBlock, switchHeader, 'fallthrough');
170
+
171
+ const joinBlock = S.makeBlock('body');
172
+ const switchCtx: LoopCtx = { headerBlock: switchHeader, exitBlock: joinBlock };
173
+ S.loopStack.push(switchCtx);
174
+
175
+ const switchBody = switchStmt.childForFieldName('body');
176
+ const container = switchBody || switchStmt;
177
+
178
+ let hasDefault = false;
179
+ for (let i = 0; i < container.namedChildCount; i++) {
180
+ const caseClause = nn(container.namedChild(i));
181
+
182
+ const isDefault = caseClause.type === cfgRules.defaultNode;
183
+ const isCase = isDefault || isCaseNode(caseClause.type, cfgRules);
184
+ if (!isCase) continue;
185
+
186
+ const caseLabel = isDefault ? 'default' : 'case';
187
+ const caseBlock = S.makeBlock('case', caseClause.startPosition.row + 1, null, caseLabel);
188
+ S.addEdge(switchHeader, caseBlock, isDefault ? 'branch_false' : 'branch_true');
189
+ if (isDefault) hasDefault = true;
190
+
191
+ const caseStmts = extractCaseBody(caseClause, cfgRules);
192
+ const caseEnd = processStatements(caseStmts, caseBlock, S, cfgRules);
193
+ if (caseEnd) S.addEdge(caseEnd, joinBlock, 'fallthrough');
194
+ }
195
+
196
+ if (!hasDefault) {
197
+ S.addEdge(switchHeader, joinBlock, 'branch_false');
198
+ }
199
+
200
+ S.loopStack.pop();
201
+ return joinBlock;
202
+ }
203
+
204
+ function extractCaseBody(caseClause: TreeSitterNode, cfgRules: AnyRules): TreeSitterNode[] {
205
+ const caseBodyNode =
206
+ caseClause.childForFieldName('body') || caseClause.childForFieldName('consequence');
207
+ if (caseBodyNode) {
208
+ return getBodyStatements(caseBodyNode, cfgRules);
209
+ }
210
+
211
+ const stmts: TreeSitterNode[] = [];
212
+ const valueNode = caseClause.childForFieldName('value');
213
+ const patternNode = caseClause.childForFieldName('pattern');
214
+ for (let j = 0; j < caseClause.namedChildCount; j++) {
215
+ const child = nn(caseClause.namedChild(j));
216
+ if (child !== valueNode && child !== patternNode && child.type !== 'switch_label') {
217
+ if (child.type === 'statement_list') {
218
+ for (let k = 0; k < child.namedChildCount; k++) {
219
+ stmts.push(nn(child.namedChild(k)));
220
+ }
221
+ } else {
222
+ stmts.push(child);
223
+ }
224
+ }
225
+ }
226
+ return stmts;
227
+ }
@@ -0,0 +1,136 @@
1
+ import type { TreeSitterNode } from '../../types.js';
2
+ import type {
3
+ AnyRules,
4
+ CfgBlockInternal,
5
+ FuncState,
6
+ LoopCtx,
7
+ ProcessStatementsFn,
8
+ } from './cfg-shared.js';
9
+ import { getBodyStatements, registerLabelCtx } from './cfg-shared.js';
10
+
11
+ export function processForLoop(
12
+ forStmt: TreeSitterNode,
13
+ currentBlock: CfgBlockInternal,
14
+ S: FuncState,
15
+ cfgRules: AnyRules,
16
+ processStatements: ProcessStatementsFn,
17
+ ): CfgBlockInternal {
18
+ const headerBlock = S.makeBlock(
19
+ 'loop_header',
20
+ forStmt.startPosition.row + 1,
21
+ forStmt.startPosition.row + 1,
22
+ 'for',
23
+ );
24
+ S.addEdge(currentBlock, headerBlock, 'fallthrough');
25
+
26
+ const loopExitBlock = S.makeBlock('body');
27
+ const loopCtx: LoopCtx = { headerBlock, exitBlock: loopExitBlock };
28
+ S.loopStack.push(loopCtx);
29
+ registerLabelCtx(S, headerBlock, loopExitBlock);
30
+
31
+ const body = forStmt.childForFieldName('body');
32
+ const bodyBlock = S.makeBlock('loop_body');
33
+ S.addEdge(headerBlock, bodyBlock, 'branch_true');
34
+
35
+ const bodyStmts = getBodyStatements(body, cfgRules);
36
+ const bodyEnd = processStatements(bodyStmts, bodyBlock, S, cfgRules);
37
+ if (bodyEnd) S.addEdge(bodyEnd, headerBlock, 'loop_back');
38
+
39
+ S.addEdge(headerBlock, loopExitBlock, 'loop_exit');
40
+ S.loopStack.pop();
41
+ return loopExitBlock;
42
+ }
43
+
44
+ export function processWhileLoop(
45
+ whileStmt: TreeSitterNode,
46
+ currentBlock: CfgBlockInternal,
47
+ S: FuncState,
48
+ cfgRules: AnyRules,
49
+ processStatements: ProcessStatementsFn,
50
+ ): CfgBlockInternal {
51
+ const headerBlock = S.makeBlock(
52
+ 'loop_header',
53
+ whileStmt.startPosition.row + 1,
54
+ whileStmt.startPosition.row + 1,
55
+ 'while',
56
+ );
57
+ S.addEdge(currentBlock, headerBlock, 'fallthrough');
58
+
59
+ const loopExitBlock = S.makeBlock('body');
60
+ const loopCtx: LoopCtx = { headerBlock, exitBlock: loopExitBlock };
61
+ S.loopStack.push(loopCtx);
62
+ registerLabelCtx(S, headerBlock, loopExitBlock);
63
+
64
+ const body = whileStmt.childForFieldName('body');
65
+ const bodyBlock = S.makeBlock('loop_body');
66
+ S.addEdge(headerBlock, bodyBlock, 'branch_true');
67
+
68
+ const bodyStmts = getBodyStatements(body, cfgRules);
69
+ const bodyEnd = processStatements(bodyStmts, bodyBlock, S, cfgRules);
70
+ if (bodyEnd) S.addEdge(bodyEnd, headerBlock, 'loop_back');
71
+
72
+ S.addEdge(headerBlock, loopExitBlock, 'loop_exit');
73
+ S.loopStack.pop();
74
+ return loopExitBlock;
75
+ }
76
+
77
+ export function processDoWhileLoop(
78
+ doStmt: TreeSitterNode,
79
+ currentBlock: CfgBlockInternal,
80
+ S: FuncState,
81
+ cfgRules: AnyRules,
82
+ processStatements: ProcessStatementsFn,
83
+ ): CfgBlockInternal {
84
+ const bodyBlock = S.makeBlock('loop_body', doStmt.startPosition.row + 1, null, 'do');
85
+ S.addEdge(currentBlock, bodyBlock, 'fallthrough');
86
+
87
+ const condBlock = S.makeBlock('loop_header', null, null, 'do-while');
88
+ const loopExitBlock = S.makeBlock('body');
89
+
90
+ const loopCtx: LoopCtx = { headerBlock: condBlock, exitBlock: loopExitBlock };
91
+ S.loopStack.push(loopCtx);
92
+ registerLabelCtx(S, condBlock, loopExitBlock);
93
+
94
+ const body = doStmt.childForFieldName('body');
95
+ const bodyStmts = getBodyStatements(body, cfgRules);
96
+ const bodyEnd = processStatements(bodyStmts, bodyBlock, S, cfgRules);
97
+ if (bodyEnd) S.addEdge(bodyEnd, condBlock, 'fallthrough');
98
+
99
+ S.addEdge(condBlock, bodyBlock, 'loop_back');
100
+ S.addEdge(condBlock, loopExitBlock, 'loop_exit');
101
+
102
+ S.loopStack.pop();
103
+ return loopExitBlock;
104
+ }
105
+
106
+ export function processInfiniteLoop(
107
+ loopStmt: TreeSitterNode,
108
+ currentBlock: CfgBlockInternal,
109
+ S: FuncState,
110
+ cfgRules: AnyRules,
111
+ processStatements: ProcessStatementsFn,
112
+ ): CfgBlockInternal {
113
+ const headerBlock = S.makeBlock(
114
+ 'loop_header',
115
+ loopStmt.startPosition.row + 1,
116
+ loopStmt.startPosition.row + 1,
117
+ 'loop',
118
+ );
119
+ S.addEdge(currentBlock, headerBlock, 'fallthrough');
120
+
121
+ const loopExitBlock = S.makeBlock('body');
122
+ const loopCtx: LoopCtx = { headerBlock, exitBlock: loopExitBlock };
123
+ S.loopStack.push(loopCtx);
124
+ registerLabelCtx(S, headerBlock, loopExitBlock);
125
+
126
+ const body = loopStmt.childForFieldName('body');
127
+ const bodyBlock = S.makeBlock('loop_body');
128
+ S.addEdge(headerBlock, bodyBlock, 'branch_true');
129
+
130
+ const bodyStmts = getBodyStatements(body, cfgRules);
131
+ const bodyEnd = processStatements(bodyStmts, bodyBlock, S, cfgRules);
132
+ if (bodyEnd) S.addEdge(bodyEnd, headerBlock, 'loop_back');
133
+
134
+ S.loopStack.pop();
135
+ return loopExitBlock;
136
+ }