@unerr-ai/unerr 0.1.0 → 0.1.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 (553) hide show
  1. package/dist/__tests__/architecture-guard.test.js +122 -0
  2. package/dist/__tests__/arg-validator.test.js +205 -0
  3. package/dist/__tests__/ast-extractor.test.js +203 -0
  4. package/dist/__tests__/auto-bootstrap.test.js +280 -0
  5. package/dist/__tests__/background-indexer.test.js +228 -0
  6. package/dist/__tests__/blast-radius-engine.test.js +200 -0
  7. package/dist/__tests__/bridge-isolation.test.js +37 -0
  8. package/dist/__tests__/budget-enforcer.test.js +53 -0
  9. package/dist/__tests__/cfg-test-detection-perf.test.js +82 -0
  10. package/dist/__tests__/change-narrative.test.js +190 -0
  11. package/dist/__tests__/check-commit.test.js +258 -0
  12. package/dist/__tests__/checksum.test.js +34 -0
  13. package/dist/__tests__/commit-watcher.test.js +154 -0
  14. package/dist/__tests__/community-detection.test.js +179 -0
  15. package/dist/__tests__/community-tools.test.js +299 -0
  16. package/dist/__tests__/components.test.js +449 -0
  17. package/dist/__tests__/compression-log.test.js +174 -0
  18. package/dist/__tests__/compression-quality-monitor.test.js +40 -0
  19. package/dist/__tests__/config-healer.test.js +165 -0
  20. package/dist/__tests__/context-ledger.test.js +58 -0
  21. package/dist/__tests__/convention-detector.test.js +99 -0
  22. package/dist/__tests__/convention-learner.test.js +86 -0
  23. package/dist/__tests__/correction-detector.test.js +330 -0
  24. package/dist/__tests__/daemon-autostart-install.test.js +283 -0
  25. package/dist/__tests__/daemon-bridge.test.js +222 -0
  26. package/dist/__tests__/daemon-dashboard.test.js +202 -0
  27. package/dist/__tests__/daemon-registry.test.js +240 -0
  28. package/dist/__tests__/daemon-supervisor.test.js +318 -0
  29. package/dist/__tests__/daemon-version-check.test.js +275 -0
  30. package/dist/__tests__/decision-point-detector.test.js +98 -0
  31. package/dist/__tests__/deep-link.test.js +143 -0
  32. package/dist/__tests__/disallowed-tools.test.js +115 -0
  33. package/dist/__tests__/drift-tracker.test.js +582 -0
  34. package/dist/__tests__/durability-scorer.test.js +152 -0
  35. package/dist/__tests__/efficiency-tracker.test.js +65 -0
  36. package/dist/__tests__/enrich.test.js +144 -0
  37. package/dist/__tests__/entity-rewind.test.js +248 -0
  38. package/dist/__tests__/ephemeral.test.js +111 -0
  39. package/dist/__tests__/exploration-cost.test.js +93 -0
  40. package/dist/__tests__/fact-generator.test.js +197 -0
  41. package/dist/__tests__/file-l0-graph.test.js +244 -0
  42. package/dist/__tests__/file-logger.test.js +82 -0
  43. package/dist/__tests__/file-outline.test.js +141 -0
  44. package/dist/__tests__/file-read-protocol.test.js +188 -0
  45. package/dist/__tests__/format-encoder.test.js +233 -0
  46. package/dist/__tests__/git-attribution.test.js +259 -0
  47. package/dist/__tests__/graph-temporal-joiner.test.js +219 -0
  48. package/dist/__tests__/health-grade-enhanced.test.js +138 -0
  49. package/dist/__tests__/health-map-data.test.js +173 -0
  50. package/dist/__tests__/helpers/mcp-harness.js +45 -0
  51. package/dist/__tests__/helpers/mcp-harness.test.js +68 -0
  52. package/dist/__tests__/hook-dedup.test.js +112 -0
  53. package/dist/__tests__/hook-runner.test.js +253 -0
  54. package/dist/__tests__/indexer-cfg.test.js +185 -0
  55. package/dist/__tests__/indexer-cross-file.test.js +172 -0
  56. package/dist/__tests__/indexer-extraction.test.js +245 -0
  57. package/dist/__tests__/indexer-incremental.test.js +232 -0
  58. package/dist/__tests__/indexer-language-expansion.test.js +165 -0
  59. package/dist/__tests__/init-push.test.js +131 -0
  60. package/dist/__tests__/instruction-writer.test.js +179 -0
  61. package/dist/__tests__/intelligence-integration.test.js +217 -0
  62. package/dist/__tests__/intent-correlator.test.js +175 -0
  63. package/dist/__tests__/intent-detector.test.js +235 -0
  64. package/dist/__tests__/intent-encoder.test.js +167 -0
  65. package/dist/__tests__/java-build-tool-detection.test.js +174 -0
  66. package/dist/__tests__/layer3-sprint-q.test.js +160 -0
  67. package/dist/__tests__/layer3-sprint-r.test.js +91 -0
  68. package/dist/__tests__/layer3-sprint-s.test.js +183 -0
  69. package/dist/__tests__/layer3-sprint-t.test.js +201 -0
  70. package/dist/__tests__/layer3-sprint-u.test.js +174 -0
  71. package/dist/__tests__/layer4-sprint-ba2.test.js +354 -0
  72. package/dist/__tests__/layer4-sprint-ba4.test.js +84 -0
  73. package/dist/__tests__/layer4-sprint-vs.test.js +105 -0
  74. package/dist/__tests__/ledger-chains.test.js +162 -0
  75. package/dist/__tests__/lifecycle-machine.test.js +226 -0
  76. package/dist/__tests__/local-chat-provider.test.js +170 -0
  77. package/dist/__tests__/local-convention-detector.test.js +308 -0
  78. package/dist/__tests__/local-embeddings.test.js +422 -0
  79. package/dist/__tests__/local-graph.test.js +540 -0
  80. package/dist/__tests__/local-indexer.test.js +228 -0
  81. package/dist/__tests__/local-intelligence-l3.test.js +332 -0
  82. package/dist/__tests__/local-llm.test.js +253 -0
  83. package/dist/__tests__/local-mode-offline.test.js +187 -0
  84. package/dist/__tests__/local-mode-stats.test.js +273 -0
  85. package/dist/__tests__/local-mode-tui.test.js +343 -0
  86. package/dist/__tests__/local-parse.test.js +199 -0
  87. package/dist/__tests__/log-tailer.test.js +208 -0
  88. package/dist/__tests__/loop-breaker.test.js +276 -0
  89. package/dist/__tests__/loop-miner.test.js +226 -0
  90. package/dist/__tests__/mcp-config.test.js +126 -0
  91. package/dist/__tests__/mcp-content-json.test.js +10 -0
  92. package/dist/__tests__/mcp-envelope.test.js +124 -0
  93. package/dist/__tests__/metrics-store.test.js +223 -0
  94. package/dist/__tests__/native-watcher.test.js +191 -0
  95. package/dist/__tests__/navigation-hooks-agent-aware.test.js +145 -0
  96. package/dist/__tests__/negative-knowledge.test.js +116 -0
  97. package/dist/__tests__/network-boundary.test.js +190 -0
  98. package/dist/__tests__/network-firewall.test.js +112 -0
  99. package/dist/__tests__/nudge-invariants.test.js +160 -0
  100. package/dist/__tests__/nudge-v2.test.js +225 -0
  101. package/dist/__tests__/offline-rewind.test.js +251 -0
  102. package/dist/__tests__/open-threads.test.js +89 -0
  103. package/dist/__tests__/output-compressor.test.js +93 -0
  104. package/dist/__tests__/pending-violations.test.js +112 -0
  105. package/dist/__tests__/persistence-effectiveness.test.js +143 -0
  106. package/dist/__tests__/provider-factory.test.js +42 -0
  107. package/dist/__tests__/providers.test.js +24 -0
  108. package/dist/__tests__/proxy.test.js +314 -0
  109. package/dist/__tests__/query-router.test.js +1018 -0
  110. package/dist/__tests__/reasoning-quality-route.test.js +138 -0
  111. package/dist/__tests__/redactor.test.js +120 -0
  112. package/dist/__tests__/resource-monitor.test.js +57 -0
  113. package/dist/__tests__/response-envelope.test.js +100 -0
  114. package/dist/__tests__/risk-classifier.test.js +101 -0
  115. package/dist/__tests__/risk-signal-scope.test.js +75 -0
  116. package/dist/__tests__/rule-evaluator.test.js +280 -0
  117. package/dist/__tests__/scip-decoder.test.js +49 -0
  118. package/dist/__tests__/scip-downloader.test.js +201 -0
  119. package/dist/__tests__/scip-merger.test.js +103 -0
  120. package/dist/__tests__/search-index.test.js +422 -0
  121. package/dist/__tests__/semantic-enrichment.test.js +360 -0
  122. package/dist/__tests__/session-brief-builder.test.js +187 -0
  123. package/dist/__tests__/session-context.test.js +221 -0
  124. package/dist/__tests__/session-continuity.test.js +144 -0
  125. package/dist/__tests__/session-dedup.test.js +74 -0
  126. package/dist/__tests__/session-event-wiring.test.js +206 -0
  127. package/dist/__tests__/session-events.test.js +149 -0
  128. package/dist/__tests__/session-legend.test.js +20 -0
  129. package/dist/__tests__/session-persistence.test.js +131 -0
  130. package/dist/__tests__/session-resume-block.test.js +107 -0
  131. package/dist/__tests__/session-resume.test.js +97 -0
  132. package/dist/__tests__/session-summary-writer.test.js +134 -0
  133. package/dist/__tests__/shadow-ledger.test.js +203 -0
  134. package/dist/__tests__/shell-classifier.test.js +151 -0
  135. package/dist/__tests__/shell-compression-floor.test.js +189 -0
  136. package/dist/__tests__/shell-compression-v2.test.js +339 -0
  137. package/dist/__tests__/shell-compressor.test.js +35 -0
  138. package/dist/__tests__/shell-hooks.test.js +128 -0
  139. package/dist/__tests__/shell-strategies.test.js +644 -0
  140. package/dist/__tests__/shell-tee.test.js +133 -0
  141. package/dist/__tests__/signal-dedup.test.js +158 -0
  142. package/dist/__tests__/signal-reinforcer.test.js +77 -0
  143. package/dist/__tests__/signal-scorer.test.js +251 -0
  144. package/dist/__tests__/signal-show-store.test.js +108 -0
  145. package/dist/__tests__/smart-truncate.test.js +215 -0
  146. package/dist/__tests__/snapshot-v2.test.js +113 -0
  147. package/dist/__tests__/sprint-l1-local-mode.test.js +130 -0
  148. package/dist/__tests__/sprint-l10-boot.test.js +220 -0
  149. package/dist/__tests__/sprint-l9-offline-commands.test.js +189 -0
  150. package/dist/__tests__/sprint-q-persistent-context.test.js +198 -0
  151. package/dist/__tests__/sprint-s1-wiring.test.js +215 -0
  152. package/dist/__tests__/sprint-s2-wiring.test.js +256 -0
  153. package/dist/__tests__/sprint-s3-wiring.test.js +195 -0
  154. package/dist/__tests__/sprint-s4-wiring.test.js +213 -0
  155. package/dist/__tests__/sprint-s6-hooks.test.js +222 -0
  156. package/dist/__tests__/sprint-s7-persistent.test.js +263 -0
  157. package/dist/__tests__/sprint-s8-value.test.js +167 -0
  158. package/dist/__tests__/sprint-s9-behavioral.test.js +179 -0
  159. package/dist/__tests__/sprint3-intelligence.test.js +297 -0
  160. package/dist/__tests__/sprint5-mcp-server.test.js +136 -0
  161. package/dist/__tests__/startup-display.test.js +302 -0
  162. package/dist/__tests__/startup-log-file.test.js +97 -0
  163. package/dist/__tests__/stash-manager.test.js +229 -0
  164. package/dist/__tests__/state-detector.test.js +92 -0
  165. package/dist/__tests__/status-dashboard.test.js +142 -0
  166. package/dist/__tests__/temporal-facts.test.js +292 -0
  167. package/dist/__tests__/temporal-routes.test.js +142 -0
  168. package/dist/__tests__/test-detector.test.js +174 -0
  169. package/dist/__tests__/theme.test.js +72 -0
  170. package/dist/__tests__/timeline-agents.test.js +122 -0
  171. package/dist/__tests__/timeline-bootstrap.test.js +176 -0
  172. package/dist/__tests__/timeline-filters.test.js +193 -0
  173. package/dist/__tests__/timeline-markers.test.js +151 -0
  174. package/dist/__tests__/timeline-routes.test.js +156 -0
  175. package/dist/__tests__/timeline-store.test.js +171 -0
  176. package/dist/__tests__/token-counter.test.js +86 -0
  177. package/dist/__tests__/token-estimator.test.js +96 -0
  178. package/dist/__tests__/token-flow-api.test.js +239 -0
  179. package/dist/__tests__/token-flow-instrumentation.test.js +437 -0
  180. package/dist/__tests__/token-flow-persistence.test.js +356 -0
  181. package/dist/__tests__/token-flow-routes.test.js +199 -0
  182. package/dist/__tests__/token-flow.test.js +695 -0
  183. package/dist/__tests__/tool-clusters.test.js +177 -0
  184. package/dist/__tests__/transport-mux.test.js +283 -0
  185. package/dist/__tests__/turn-segmenter.test.js +166 -0
  186. package/dist/__tests__/uninstall.test.js +141 -0
  187. package/dist/__tests__/warm-start-policy.test.js +271 -0
  188. package/dist/__tests__/wire-cap-nudge.test.js +77 -0
  189. package/dist/__tests__/worker-pool.test.js +101 -0
  190. package/dist/behaviors/agent-llm-bridge.js +166 -0
  191. package/dist/behaviors/architecture-guard.js +256 -0
  192. package/dist/behaviors/auto-doc.js +247 -0
  193. package/dist/behaviors/cascade-guard.js +289 -0
  194. package/dist/behaviors/change-narrative.js +270 -0
  195. package/dist/behaviors/convention-drift.js +290 -0
  196. package/dist/behaviors/framework.js +235 -0
  197. package/dist/behaviors/guard-formatter.js +44 -0
  198. package/dist/behaviors/incomplete-work.js +270 -0
  199. package/dist/behaviors/loop-breaker.js +300 -0
  200. package/dist/behaviors/session-continuity.js +208 -0
  201. package/dist/cli.js +992 -706
  202. package/dist/commands/branches.js +97 -0
  203. package/dist/commands/check-commit.js +225 -0
  204. package/dist/commands/compress-output.js +64 -0
  205. package/dist/commands/config-verify.js +243 -0
  206. package/dist/commands/daemon.js +905 -0
  207. package/dist/commands/dashboard.js +52 -0
  208. package/dist/commands/debug.js +200 -0
  209. package/dist/commands/enrich.js +184 -0
  210. package/dist/commands/exec.js +233 -0
  211. package/dist/commands/gain.js +156 -0
  212. package/dist/commands/hook.js +88 -0
  213. package/dist/commands/index.js +88 -0
  214. package/dist/commands/init.js +74 -0
  215. package/dist/commands/install.js +505 -0
  216. package/dist/commands/learn.js +116 -0
  217. package/dist/commands/manifest.js +193 -0
  218. package/dist/commands/rewind.js +103 -0
  219. package/dist/commands/serve.js +19 -0
  220. package/dist/commands/setup-wizard.js +414 -0
  221. package/dist/commands/skills.js +64 -0
  222. package/dist/commands/stats.js +20 -0
  223. package/dist/commands/status.js +654 -0
  224. package/dist/commands/timeline.js +139 -0
  225. package/dist/commands/uninstall.js +230 -0
  226. package/dist/components/App.js +109 -0
  227. package/dist/components/Banner.js +12 -0
  228. package/dist/components/ConfirmPrompt.js +25 -0
  229. package/dist/components/DriftSummary.js +23 -0
  230. package/dist/components/GradeBadge.js +15 -0
  231. package/dist/components/HealthCard.js +18 -0
  232. package/dist/components/InkSpinner.js +22 -0
  233. package/dist/components/InputBox.js +17 -0
  234. package/dist/components/KeyValue.js +13 -0
  235. package/dist/components/MessageList.js +14 -0
  236. package/dist/components/ProgressBar.js +26 -0
  237. package/dist/components/Section.js +16 -0
  238. package/dist/components/SessionSummaryCard.js +73 -0
  239. package/dist/components/StartupDisplay.js +24 -0
  240. package/dist/components/StatusDashboard.js +57 -0
  241. package/dist/components/StatusLine.js +8 -0
  242. package/dist/components/StepLine.js +22 -0
  243. package/dist/components/Theme.js +20 -0
  244. package/dist/components/ToolProgress.js +8 -0
  245. package/dist/components/ViolationList.js +21 -0
  246. package/dist/components/render.js +13 -0
  247. package/dist/config/agent-registry.js +237 -0
  248. package/dist/config/claude-settings-hooks.js +304 -0
  249. package/dist/config/hook-installer.js +65 -0
  250. package/dist/config/instruction-writer.js +388 -0
  251. package/dist/config/mcp-config-writer.js +266 -0
  252. package/dist/config/settings.js +174 -0
  253. package/dist/config/tool-detector.js +42 -0
  254. package/dist/config/value-surfacing.js +119 -0
  255. package/dist/core/context-assembly.js +108 -0
  256. package/dist/core/conversation.js +33 -0
  257. package/dist/core/local-chat-provider.js +475 -0
  258. package/dist/core/provider-factory.js +55 -0
  259. package/dist/core/providers.js +90 -0
  260. package/dist/core/query-engine.js +174 -0
  261. package/dist/daemon/api.js +312 -0
  262. package/dist/daemon/autostart.js +119 -0
  263. package/dist/daemon/bootstrap.js +39 -0
  264. package/dist/daemon/client.js +164 -0
  265. package/dist/daemon/detect-ci.js +81 -0
  266. package/dist/daemon/platform-linux.js +146 -0
  267. package/dist/daemon/platform-macos.js +134 -0
  268. package/dist/daemon/platform-windows.js +116 -0
  269. package/dist/daemon/process-manager.js +299 -0
  270. package/dist/daemon/protocol.js +23 -0
  271. package/dist/daemon/registry.js +270 -0
  272. package/dist/daemon/settings-schema.js +72 -0
  273. package/dist/daemon/system-health.js +134 -0
  274. package/dist/daemon/version-checker.js +262 -0
  275. package/dist/daemon/warm-start.js +223 -0
  276. package/dist/entrypoints/cli.js +1043 -0
  277. package/dist/entrypoints/daemon.js +380 -0
  278. package/dist/entrypoints/repl.js +147 -0
  279. package/dist/hooks/adapters/claude-code.js +90 -0
  280. package/dist/hooks/adapters/cline.js +100 -0
  281. package/dist/hooks/adapters/cursor.js +98 -0
  282. package/dist/hooks/hook-dedup.js +79 -0
  283. package/dist/hooks/hook-runner.js +113 -0
  284. package/dist/hooks/navigation-hooks.js +175 -0
  285. package/dist/hooks/prompt-hooks.js +63 -0
  286. package/dist/hooks/shell-hooks.js +47 -0
  287. package/dist/ignore.js +111 -0
  288. package/dist/intelligence/approach-suggester.js +61 -0
  289. package/dist/intelligence/ast-extractor.js +2615 -0
  290. package/dist/intelligence/ast-worker.js +34 -0
  291. package/dist/intelligence/background-indexer.js +121 -0
  292. package/dist/intelligence/blast-radius.js +200 -0
  293. package/dist/intelligence/community-detection.js +691 -0
  294. package/dist/intelligence/community-detector.js +184 -0
  295. package/dist/intelligence/computation-scheduler.js +75 -0
  296. package/dist/intelligence/confidence-propagation.js +47 -0
  297. package/dist/intelligence/convention-detector.js +242 -0
  298. package/dist/intelligence/convention-learner.js +205 -0
  299. package/dist/intelligence/convention-matcher.js +205 -0
  300. package/dist/intelligence/cozo-schema.js +376 -0
  301. package/dist/intelligence/decision-point-detector.js +90 -0
  302. package/dist/intelligence/deep-dive-tools.js +586 -0
  303. package/dist/intelligence/durability-scorer.js +84 -0
  304. package/dist/intelligence/exploration-cost.js +204 -0
  305. package/dist/intelligence/exploration-pattern-tracker.js +61 -0
  306. package/dist/intelligence/fact-generator.js +322 -0
  307. package/dist/intelligence/facts-schema.js +90 -0
  308. package/dist/intelligence/file-intelligence.js +59 -0
  309. package/dist/intelligence/graph-holder.js +220 -0
  310. package/dist/intelligence/graph-temporal-joiner.js +238 -0
  311. package/dist/intelligence/health-grade.js +423 -0
  312. package/dist/intelligence/health-grader.js +200 -0
  313. package/dist/intelligence/health-map-data.js +259 -0
  314. package/dist/intelligence/import-symbols.js +136 -0
  315. package/dist/intelligence/incremental-indexer.js +658 -0
  316. package/dist/intelligence/indexer/centrality.js +62 -0
  317. package/dist/intelligence/indexer/cfg-context.js +95 -0
  318. package/dist/intelligence/indexer/confidence.js +34 -0
  319. package/dist/intelligence/indexer/cross-file-resolver.js +104 -0
  320. package/dist/intelligence/indexer/edge-repair.js +89 -0
  321. package/dist/intelligence/indexer/entity-key.js +17 -0
  322. package/dist/intelligence/indexer/export-map.js +132 -0
  323. package/dist/intelligence/indexer/git-cochange.js +128 -0
  324. package/dist/intelligence/indexer/graph-patch.js +147 -0
  325. package/dist/intelligence/indexer/incremental.js +78 -0
  326. package/dist/intelligence/indexer/ingest.js +160 -0
  327. package/dist/intelligence/indexer/language-detect.js +226 -0
  328. package/dist/intelligence/indexer/metadata.js +63 -0
  329. package/dist/intelligence/indexer/mutation-tracker.js +79 -0
  330. package/dist/intelligence/indexer/orchestrator.js +155 -0
  331. package/dist/intelligence/indexer/plugin-interface.js +31 -0
  332. package/dist/intelligence/indexer/plugins/csharp.js +440 -0
  333. package/dist/intelligence/indexer/plugins/go.js +335 -0
  334. package/dist/intelligence/indexer/plugins/java.js +370 -0
  335. package/dist/intelligence/indexer/plugins/python.js +358 -0
  336. package/dist/intelligence/indexer/plugins/regex-fallback.js +82 -0
  337. package/dist/intelligence/indexer/plugins/ruby.js +290 -0
  338. package/dist/intelligence/indexer/plugins/rust.js +484 -0
  339. package/dist/intelligence/indexer/plugins/tier2-generic.js +310 -0
  340. package/dist/intelligence/indexer/plugins/typescript.js +456 -0
  341. package/dist/intelligence/indexer/resource-monitor.js +93 -0
  342. package/dist/intelligence/indexer/scip/decoder.js +253 -0
  343. package/dist/intelligence/indexer/scip/detector.js +232 -0
  344. package/dist/intelligence/indexer/scip/downloader.js +427 -0
  345. package/dist/intelligence/indexer/scip/fallback.js +34 -0
  346. package/dist/intelligence/indexer/scip/merger.js +109 -0
  347. package/dist/intelligence/indexer/scip/orchestrator.js +433 -0
  348. package/dist/intelligence/indexer/scip/runner.js +98 -0
  349. package/dist/intelligence/indexer/snapshot.js +66 -0
  350. package/dist/intelligence/indexer/test-detector.js +196 -0
  351. package/dist/intelligence/indexer/watch-integration.js +61 -0
  352. package/dist/intelligence/indexer/worker.js +85 -0
  353. package/dist/intelligence/local-convention-detector.js +437 -0
  354. package/dist/intelligence/local-embeddings.js +190 -0
  355. package/dist/intelligence/local-graph.js +1946 -0
  356. package/dist/intelligence/local-indexer.js +1575 -0
  357. package/dist/intelligence/local-llm.js +163 -0
  358. package/dist/intelligence/local-rule-generator.js +154 -0
  359. package/dist/intelligence/local-snapshot.js +213 -0
  360. package/dist/intelligence/negative-knowledge.js +103 -0
  361. package/dist/intelligence/persistent-db.js +85 -0
  362. package/dist/intelligence/query-router.js +2556 -0
  363. package/dist/intelligence/risk-classifier.js +116 -0
  364. package/dist/intelligence/rule-evaluator.js +380 -0
  365. package/dist/intelligence/rule-generator.js +49 -0
  366. package/dist/intelligence/search-index.js +173 -0
  367. package/dist/intelligence/semantic/docstring-extractor.js +67 -0
  368. package/dist/intelligence/semantic/embedding-store.js +52 -0
  369. package/dist/intelligence/semantic/enrichment-orchestrator.js +48 -0
  370. package/dist/intelligence/semantic/git-message-miner.js +114 -0
  371. package/dist/intelligence/semantic/identifier-tokenizer.js +51 -0
  372. package/dist/intelligence/semantic/node2vec-embeddings.js +71 -0
  373. package/dist/intelligence/semantic/node2vec-walks.js +103 -0
  374. package/dist/intelligence/semantic/path-domain-inference.js +112 -0
  375. package/dist/intelligence/semantic/similarity-engine.js +60 -0
  376. package/dist/intelligence/semantic/tfidf-vectors.js +88 -0
  377. package/dist/intelligence/session-brief-builder.js +159 -0
  378. package/dist/intelligence/session-context.js +221 -0
  379. package/dist/intelligence/session-health-monitor.js +211 -0
  380. package/dist/intelligence/session-narrative.js +197 -0
  381. package/dist/intelligence/session-pattern-analyzer.js +218 -0
  382. package/dist/intelligence/signal-scorer.js +390 -0
  383. package/dist/intelligence/signal-show-store.js +182 -0
  384. package/dist/intelligence/smart-truncate.js +158 -0
  385. package/dist/intelligence/subgraph-cache.js +88 -0
  386. package/dist/intelligence/temporal-facts.js +494 -0
  387. package/dist/intelligence/token-estimator.js +100 -0
  388. package/dist/intelligence/tool-injector.js +87 -0
  389. package/dist/intelligence/tree-sitter-loader.js +71 -0
  390. package/dist/intelligence/worker-pool.js +116 -0
  391. package/dist/proxy/arg-validator.js +79 -0
  392. package/dist/proxy/auto-bootstrap.js +167 -0
  393. package/dist/proxy/bridge.js +147 -0
  394. package/dist/proxy/budget-enforcer.js +70 -0
  395. package/dist/proxy/compression-quality-monitor.js +160 -0
  396. package/dist/proxy/compression-stats.js +51 -0
  397. package/dist/proxy/context-rot-detector.js +137 -0
  398. package/dist/proxy/drift-detector.js +139 -0
  399. package/dist/proxy/efficiency-tracker.js +79 -0
  400. package/dist/proxy/fact-ranking.js +154 -0
  401. package/dist/proxy/format-encoder.js +266 -0
  402. package/dist/proxy/http-transport.js +90 -0
  403. package/dist/proxy/lifecycle-actor.js +55 -0
  404. package/dist/proxy/lifecycle-machine.js +187 -0
  405. package/dist/proxy/log-tailer.js +265 -0
  406. package/dist/proxy/model-pricing.js +98 -0
  407. package/dist/proxy/network-firewall.js +141 -0
  408. package/dist/proxy/nudge-state.js +93 -0
  409. package/dist/proxy/output-compressor.js +185 -0
  410. package/dist/proxy/pid-lock.js +291 -0
  411. package/dist/proxy/proxy-context.js +11 -0
  412. package/dist/proxy/proxy.js +2633 -0
  413. package/dist/proxy/response-enrichment.js +32 -0
  414. package/dist/proxy/response-envelope.js +313 -0
  415. package/dist/proxy/session-dedup.js +82 -0
  416. package/dist/proxy/session-legend.js +30 -0
  417. package/dist/proxy/session-persistence.js +210 -0
  418. package/dist/proxy/session-resume.js +94 -0
  419. package/dist/proxy/session-stats.js +513 -0
  420. package/dist/proxy/shell-classifier.js +1346 -0
  421. package/dist/proxy/shell-compression-log.js +93 -0
  422. package/dist/proxy/shell-compressor.js +390 -0
  423. package/dist/proxy/shell-graph-boost.js +202 -0
  424. package/dist/proxy/shell-monitor-map.js +18 -0
  425. package/dist/proxy/shell-stats.js +54 -0
  426. package/dist/proxy/shell-strategies/cloud.js +215 -0
  427. package/dist/proxy/shell-strategies/diff.js +159 -0
  428. package/dist/proxy/shell-strategies/error-diagnostic.js +796 -0
  429. package/dist/proxy/shell-strategies/filter-dsl.js +358 -0
  430. package/dist/proxy/shell-strategies/git-status.js +177 -0
  431. package/dist/proxy/shell-strategies/key-value.js +193 -0
  432. package/dist/proxy/shell-strategies/log-text.js +154 -0
  433. package/dist/proxy/shell-strategies/omni.js +188 -0
  434. package/dist/proxy/shell-strategies/progress.js +55 -0
  435. package/dist/proxy/shell-strategies/redact.js +76 -0
  436. package/dist/proxy/shell-strategies/structured.js +241 -0
  437. package/dist/proxy/shell-strategies/tabular.js +243 -0
  438. package/dist/proxy/shell-strategies/test-results-types.js +13 -0
  439. package/dist/proxy/shell-strategies/test-results.js +784 -0
  440. package/dist/proxy/shell-strategies/tree-paths.js +144 -0
  441. package/dist/proxy/shell-strategies/yaml.js +182 -0
  442. package/dist/proxy/shell-tee.js +111 -0
  443. package/dist/proxy/signal-dedup.js +171 -0
  444. package/dist/proxy/startup-renderer.js +158 -0
  445. package/dist/proxy/task-token-display.js +38 -0
  446. package/dist/proxy/token-counter.js +61 -0
  447. package/dist/proxy/tool-clusters.js +273 -0
  448. package/dist/proxy/tool-definitions.js +525 -0
  449. package/dist/proxy/transport-mux.js +229 -0
  450. package/dist/proxy/wire-cap.js +268 -0
  451. package/dist/schemas/api/skills.js +19 -0
  452. package/dist/schemas/common/errors.js +7 -0
  453. package/dist/schemas/common/headers.js +5 -0
  454. package/dist/schemas/entities/edge.js +25 -0
  455. package/dist/schemas/entities/entity.js +22 -0
  456. package/dist/schemas/entities/rule.js +18 -0
  457. package/dist/schemas/index.js +14 -0
  458. package/dist/server/event-bus.js +59 -0
  459. package/dist/server/http.js +156 -0
  460. package/dist/server/middleware.js +70 -0
  461. package/dist/server/routes/drift.js +97 -0
  462. package/dist/server/routes/intelligence.js +1217 -0
  463. package/dist/server/routes/reasoning-quality.js +444 -0
  464. package/dist/server/routes/session.js +86 -0
  465. package/dist/server/routes/stream.js +120 -0
  466. package/dist/server/routes/system.js +73 -0
  467. package/dist/server/routes/temporal.js +170 -0
  468. package/dist/server/routes/timeline.js +232 -0
  469. package/dist/server/routes/token-flow.js +403 -0
  470. package/dist/skills/effectiveness-tracker.js +93 -0
  471. package/dist/skills/local-pack.js +380 -0
  472. package/dist/skills/resolver.js +495 -0
  473. package/dist/state-detector.js +83 -0
  474. package/dist/timeline/intent-detector.js +263 -0
  475. package/dist/timeline/loop-miner.js +140 -0
  476. package/dist/timeline/open-threads.js +49 -0
  477. package/dist/timeline/signal-reinforcer.js +62 -0
  478. package/dist/timeline/timeline-bootstrap.js +151 -0
  479. package/dist/timeline/timeline-store.js +618 -0
  480. package/dist/tools/coding/bash.js +49 -0
  481. package/dist/tools/coding/file-edit.js +72 -0
  482. package/dist/tools/coding/file-outline.js +227 -0
  483. package/dist/tools/coding/file-read-protocol.js +425 -0
  484. package/dist/tools/coding/file-read.js +35 -0
  485. package/dist/tools/coding/file-write.js +43 -0
  486. package/dist/tools/coding/glob-tool.js +109 -0
  487. package/dist/tools/coding/grep.js +162 -0
  488. package/dist/tools/coding/index.js +27 -0
  489. package/dist/tools/intelligence/index.js +269 -0
  490. package/dist/tools/intelligence/record-fact.js +48 -0
  491. package/dist/tools/intelligence/timeline-markers.js +130 -0
  492. package/dist/tools/registry.js +47 -0
  493. package/dist/tools/types.js +8 -0
  494. package/dist/tracking/auto-snapshot-triggers.js +246 -0
  495. package/dist/tracking/branch-context.js +115 -0
  496. package/dist/tracking/branch-snapshot.js +217 -0
  497. package/dist/tracking/causal-bridge.js +317 -0
  498. package/dist/tracking/circuit-breaker.js +147 -0
  499. package/dist/tracking/commit-watcher.js +114 -0
  500. package/dist/tracking/context-ledger.js +119 -0
  501. package/dist/tracking/correction-detector.js +324 -0
  502. package/dist/tracking/drift-tracker.js +874 -0
  503. package/dist/tracking/durability-tracker.js +94 -0
  504. package/dist/tracking/entity-rewind.js +200 -0
  505. package/dist/tracking/file-hash-state.js +114 -0
  506. package/dist/tracking/git-attribution.js +132 -0
  507. package/dist/tracking/git-trailers.js +171 -0
  508. package/dist/tracking/intelligence-counter.js +46 -0
  509. package/dist/tracking/intent-correlator.js +202 -0
  510. package/dist/tracking/intent-encoder.js +52 -0
  511. package/dist/tracking/intent-token-tracker.js +159 -0
  512. package/dist/tracking/ledger-archiver.js +94 -0
  513. package/dist/tracking/ledger-chains.js +245 -0
  514. package/dist/tracking/metrics-store.js +361 -0
  515. package/dist/tracking/native-watcher.js +131 -0
  516. package/dist/tracking/offline-rewind.js +295 -0
  517. package/dist/tracking/pending-violations.js +74 -0
  518. package/dist/tracking/persistence-effectiveness.js +167 -0
  519. package/dist/tracking/prompt-durability.js +202 -0
  520. package/dist/tracking/quality-signals.js +213 -0
  521. package/dist/tracking/redactor.js +73 -0
  522. package/dist/tracking/rewind-engine.js +161 -0
  523. package/dist/tracking/session-history.js +128 -0
  524. package/dist/tracking/session-receipt.js +88 -0
  525. package/dist/tracking/session-summary-writer.js +157 -0
  526. package/dist/tracking/shadow-ledger.js +321 -0
  527. package/dist/tracking/stash-manager.js +258 -0
  528. package/dist/tracking/timeline-fork.js +213 -0
  529. package/dist/tracking/timeline.js +69 -0
  530. package/dist/tracking/token-flow.js +276 -0
  531. package/dist/tracking/turn-segmenter.js +122 -0
  532. package/dist/tracking/weekly-accumulator.js +179 -0
  533. package/dist/tracking/working-snapshots.js +188 -0
  534. package/dist/tracking/workspace-manifest.js +176 -0
  535. package/dist/transport/http.js +102 -0
  536. package/dist/ui/assets/index-BsMTQdhX.js +10 -0
  537. package/dist/ui/index.html +1 -1
  538. package/dist/utils/counterfactual.js +65 -0
  539. package/dist/utils/deep-link.js +34 -0
  540. package/dist/utils/detect.js +193 -0
  541. package/dist/utils/exec.js +73 -0
  542. package/dist/utils/file-logger.js +87 -0
  543. package/dist/utils/format-error.js +29 -0
  544. package/dist/utils/git.js +181 -0
  545. package/dist/utils/log.js +57 -0
  546. package/dist/utils/logger.js +35 -0
  547. package/dist/utils/mcp-content-json.js +8 -0
  548. package/dist/utils/session-logger.js +154 -0
  549. package/dist/utils/startup-log.js +512 -0
  550. package/dist/utils/ui.js +56 -0
  551. package/package.json +5 -3
  552. package/scripts/postinstall.mjs +299 -0
  553. package/dist/ui/assets/index-B-0HTtUR.js +0 -10
@@ -0,0 +1,695 @@
1
+ import { existsSync, mkdirSync, rmSync } from "node:fs";
2
+ import os from "node:os";
3
+ import { join } from "node:path";
4
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
5
+ import { closeMetricsStore, openMetricsStore, } from "../tracking/metrics-store.js";
6
+ import { TokenFlowWriter, aggregateByMechanism, aggregateSession, readTokenFlowEvents, } from "../tracking/token-flow.js";
7
+ describe("token-flow", () => {
8
+ let tmpDir;
9
+ let unerrDir;
10
+ let dbPath;
11
+ beforeEach(() => {
12
+ tmpDir = join(os.tmpdir(), `unerr-tokenflow-test-${Date.now()}-${Math.random().toString(36).slice(2, 8)}`);
13
+ unerrDir = join(tmpDir, ".unerr");
14
+ mkdirSync(unerrDir, { recursive: true });
15
+ dbPath = join(unerrDir, "metrics.db");
16
+ });
17
+ afterEach(() => {
18
+ closeMetricsStore(unerrDir);
19
+ rmSync(tmpDir, { recursive: true, force: true });
20
+ });
21
+ function makeInput(overrides) {
22
+ return {
23
+ session_id: "test-session-001",
24
+ turn: 1,
25
+ mechanism: "graph_query",
26
+ tool: "get_callers",
27
+ tokens_without: 5000,
28
+ tokens_with: 1800,
29
+ tokens_saved: 3200,
30
+ ...overrides,
31
+ };
32
+ }
33
+ // ── Writer Tests ──────────────────────────────────────────────────
34
+ describe("TokenFlowWriter", () => {
35
+ it("creates metrics.db on first write", () => {
36
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
37
+ writer.record(makeInput());
38
+ expect(existsSync(dbPath)).toBe(true);
39
+ });
40
+ it("writes events with auto-populated fields", () => {
41
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
42
+ writer.record(makeInput());
43
+ const [event] = readTokenFlowEvents(unerrDir);
44
+ expect(event).toBeDefined();
45
+ expect(event.id).toBeGreaterThan(0);
46
+ expect(event.ts).toMatch(/^\d{4}-\d{2}-\d{2}T/);
47
+ expect(event.pid).toBe(process.pid);
48
+ expect(event.session_id).toBe("test-session-001");
49
+ expect(event.turn).toBe(1);
50
+ expect(event.mechanism).toBe("graph_query");
51
+ expect(event.tool).toBe("get_callers");
52
+ expect(event.tokens_without).toBe(5000);
53
+ expect(event.tokens_with).toBe(1800);
54
+ expect(event.tokens_saved).toBe(3200);
55
+ });
56
+ it("assigns monotonic IDs across multiple writes", () => {
57
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
58
+ writer.record(makeInput({ turn: 1 }));
59
+ writer.record(makeInput({ turn: 2 }));
60
+ writer.record(makeInput({ turn: 3 }));
61
+ const events = readTokenFlowEvents(unerrDir);
62
+ expect(events).toHaveLength(3);
63
+ const ids = events.map((e) => e.id);
64
+ expect(ids).toEqual([...ids].sort((a, b) => a - b));
65
+ expect(new Set(ids).size).toBe(3);
66
+ });
67
+ it("records detail metadata when provided", () => {
68
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
69
+ writer.record(makeInput({ detail: { counterfactual: "file_read", depth: 3 } }));
70
+ const [event] = readTokenFlowEvents(unerrDir);
71
+ expect(event.detail).toEqual({ counterfactual: "file_read", depth: 3 });
72
+ });
73
+ it("records events with null tool (shell compression)", () => {
74
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
75
+ writer.record(makeInput({
76
+ mechanism: "shell_compression",
77
+ tool: null,
78
+ turn: 0,
79
+ tokens_without: 3400,
80
+ tokens_with: 800,
81
+ tokens_saved: 2600,
82
+ }));
83
+ const [event] = readTokenFlowEvents(unerrDir);
84
+ expect(event.tool).toBeNull();
85
+ expect(event.mechanism).toBe("shell_compression");
86
+ });
87
+ it("getSessionEvents returns in-memory buffer", () => {
88
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
89
+ writer.record(makeInput({ turn: 1 }));
90
+ writer.record(makeInput({ turn: 2 }));
91
+ const events = writer.getSessionEvents();
92
+ expect(events).toHaveLength(2);
93
+ expect(events[0].turn).toBe(1);
94
+ expect(events[1].turn).toBe(2);
95
+ });
96
+ it("getSessionTokensSaved computes running total", () => {
97
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
98
+ writer.record(makeInput({ tokens_saved: 3200 }));
99
+ writer.record(makeInput({ tokens_saved: 1200 }));
100
+ writer.record(makeInput({ tokens_saved: 480 }));
101
+ expect(writer.getSessionTokensSaved()).toBe(4880);
102
+ });
103
+ it("getSessionEfficiency computes correct percentage", () => {
104
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
105
+ writer.record(makeInput({ tokens_without: 10000, tokens_saved: 7000 }));
106
+ writer.record(makeInput({ tokens_without: 5000, tokens_saved: 2000 }));
107
+ // Total without: 15000, total saved: 9000 → 60%
108
+ expect(writer.getSessionEfficiency()).toBe(60);
109
+ });
110
+ it("getSessionEfficiency returns 0 when no events", () => {
111
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
112
+ expect(writer.getSessionEfficiency()).toBe(0);
113
+ });
114
+ it("handles zero-savings events", () => {
115
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
116
+ writer.record(makeInput({
117
+ tokens_without: 100,
118
+ tokens_with: 100,
119
+ tokens_saved: 0,
120
+ }));
121
+ expect(writer.getSessionTokensSaved()).toBe(0);
122
+ expect(writer.getSessionEfficiency()).toBe(0);
123
+ });
124
+ });
125
+ // ── Reader Tests ──────────────────────────────────────────────────
126
+ describe("readTokenFlowEvents", () => {
127
+ it("returns empty array when file does not exist", () => {
128
+ const events = readTokenFlowEvents(unerrDir);
129
+ expect(events).toEqual([]);
130
+ });
131
+ it("reads all events from JSONL file", () => {
132
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
133
+ writer.record(makeInput({ turn: 1 }));
134
+ writer.record(makeInput({ turn: 2 }));
135
+ writer.record(makeInput({ turn: 3 }));
136
+ const events = readTokenFlowEvents(unerrDir);
137
+ expect(events).toHaveLength(3);
138
+ });
139
+ it("filters by session_id", () => {
140
+ const writer1 = new TokenFlowWriter(unerrDir, "session-a");
141
+ writer1.record(makeInput({ session_id: "session-a" }));
142
+ writer1.record(makeInput({ session_id: "session-a" }));
143
+ const writer2 = new TokenFlowWriter(unerrDir, "session-b");
144
+ writer2.record(makeInput({ session_id: "session-b" }));
145
+ const eventsA = readTokenFlowEvents(unerrDir, {
146
+ session_id: "session-a",
147
+ });
148
+ expect(eventsA).toHaveLength(2);
149
+ const eventsB = readTokenFlowEvents(unerrDir, {
150
+ session_id: "session-b",
151
+ });
152
+ expect(eventsB).toHaveLength(1);
153
+ });
154
+ it("filters by mechanism", () => {
155
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
156
+ writer.record(makeInput({ mechanism: "graph_query" }));
157
+ writer.record(makeInput({ mechanism: "shell_compression" }));
158
+ writer.record(makeInput({ mechanism: "format_encoding" }));
159
+ writer.record(makeInput({ mechanism: "graph_query" }));
160
+ const graphEvents = readTokenFlowEvents(unerrDir, {
161
+ mechanism: "graph_query",
162
+ });
163
+ expect(graphEvents).toHaveLength(2);
164
+ });
165
+ it("skips malformed detail JSON gracefully", () => {
166
+ // SQLite stores detail as a TEXT column; if a row ever has a corrupt
167
+ // JSON payload (e.g. external migration), the reader returns it but
168
+ // simply leaves `detail` undefined rather than throwing.
169
+ const store = openMetricsStore(unerrDir);
170
+ store.insertTokenFlow({
171
+ ts: Date.now(),
172
+ ts_iso: new Date().toISOString(),
173
+ session_id: "s1",
174
+ pid: 1,
175
+ turn: 1,
176
+ mechanism: "graph_query",
177
+ tool: "t",
178
+ tokens_without: 100,
179
+ tokens_with: 50,
180
+ tokens_saved: 50,
181
+ detail: "{not valid json",
182
+ });
183
+ store.insertTokenFlow({
184
+ ts: Date.now(),
185
+ ts_iso: new Date().toISOString(),
186
+ session_id: "s1",
187
+ pid: 1,
188
+ turn: 2,
189
+ mechanism: "graph_query",
190
+ tool: "t",
191
+ tokens_without: 200,
192
+ tokens_with: 100,
193
+ tokens_saved: 100,
194
+ detail: null,
195
+ });
196
+ expect(() => readTokenFlowEvents(unerrDir)).not.toThrow();
197
+ const events = readTokenFlowEvents(unerrDir);
198
+ expect(events).toHaveLength(2);
199
+ });
200
+ });
201
+ // ── Aggregation Tests ─────────────────────────────────────────────
202
+ describe("aggregateSession", () => {
203
+ it("computes correct session totals", () => {
204
+ const writer = new TokenFlowWriter(unerrDir, "s1");
205
+ writer.record(makeInput({
206
+ session_id: "s1",
207
+ turn: 1,
208
+ tokens_without: 5000,
209
+ tokens_with: 1800,
210
+ tokens_saved: 3200,
211
+ }));
212
+ writer.record(makeInput({
213
+ session_id: "s1",
214
+ turn: 2,
215
+ tokens_without: 2000,
216
+ tokens_with: 800,
217
+ tokens_saved: 1200,
218
+ }));
219
+ writer.record(makeInput({
220
+ session_id: "s1",
221
+ turn: 3,
222
+ tokens_without: 1000,
223
+ tokens_with: 520,
224
+ tokens_saved: 480,
225
+ }));
226
+ const summary = aggregateSession(writer.getSessionEvents(), "s1");
227
+ expect(summary.session_id).toBe("s1");
228
+ expect(summary.total_turns).toBe(3);
229
+ expect(summary.total_tokens_without).toBe(8000);
230
+ expect(summary.total_tokens_with).toBe(3120);
231
+ expect(summary.total_tokens_saved).toBe(4880);
232
+ expect(summary.efficiency_pct).toBe(61);
233
+ });
234
+ it("filters to correct session when multiple sessions present", () => {
235
+ const events = [
236
+ {
237
+ id: 1,
238
+ ts: "t1",
239
+ session_id: "s1",
240
+ pid: 1,
241
+ turn: 1,
242
+ mechanism: "graph_query",
243
+ tool: "t",
244
+ tokens_without: 100,
245
+ tokens_with: 50,
246
+ tokens_saved: 50,
247
+ },
248
+ {
249
+ id: 2,
250
+ ts: "t2",
251
+ session_id: "s2",
252
+ pid: 1,
253
+ turn: 1,
254
+ mechanism: "graph_query",
255
+ tool: "t",
256
+ tokens_without: 200,
257
+ tokens_with: 100,
258
+ tokens_saved: 100,
259
+ },
260
+ {
261
+ id: 3,
262
+ ts: "t3",
263
+ session_id: "s1",
264
+ pid: 1,
265
+ turn: 2,
266
+ mechanism: "graph_query",
267
+ tool: "t",
268
+ tokens_without: 300,
269
+ tokens_with: 150,
270
+ tokens_saved: 150,
271
+ },
272
+ ];
273
+ const s1Summary = aggregateSession(events, "s1");
274
+ expect(s1Summary.total_tokens_saved).toBe(200);
275
+ expect(s1Summary.total_turns).toBe(2);
276
+ const s2Summary = aggregateSession(events, "s2");
277
+ expect(s2Summary.total_tokens_saved).toBe(100);
278
+ expect(s2Summary.total_turns).toBe(1);
279
+ });
280
+ it("builds mechanism breakdown with percentages", () => {
281
+ const events = [
282
+ {
283
+ id: 1,
284
+ ts: "t1",
285
+ session_id: "s1",
286
+ pid: 1,
287
+ turn: 1,
288
+ mechanism: "graph_query",
289
+ tool: "get_callers",
290
+ tokens_without: 5000,
291
+ tokens_with: 1800,
292
+ tokens_saved: 3200,
293
+ },
294
+ {
295
+ id: 2,
296
+ ts: "t2",
297
+ session_id: "s1",
298
+ pid: 1,
299
+ turn: 2,
300
+ mechanism: "shell_compression",
301
+ tool: null,
302
+ tokens_without: 2000,
303
+ tokens_with: 800,
304
+ tokens_saved: 1200,
305
+ },
306
+ {
307
+ id: 3,
308
+ ts: "t3",
309
+ session_id: "s1",
310
+ pid: 1,
311
+ turn: 3,
312
+ mechanism: "graph_query",
313
+ tool: "search_code",
314
+ tokens_without: 1000,
315
+ tokens_with: 600,
316
+ tokens_saved: 400,
317
+ },
318
+ ];
319
+ const summary = aggregateSession(events, "s1");
320
+ expect(summary.by_mechanism.graph_query).toBeDefined();
321
+ expect(summary.by_mechanism.graph_query.tokens_saved).toBe(3600);
322
+ expect(summary.by_mechanism.graph_query.event_count).toBe(2);
323
+ expect(summary.by_mechanism.shell_compression).toBeDefined();
324
+ expect(summary.by_mechanism.shell_compression.tokens_saved).toBe(1200);
325
+ expect(summary.by_mechanism.shell_compression.event_count).toBe(1);
326
+ // Percentages should sum close to 100
327
+ const totalPct = Object.values(summary.by_mechanism).reduce((sum, m) => sum + m.pct_of_total, 0);
328
+ expect(totalPct).toBeCloseTo(100, 0);
329
+ });
330
+ it("identifies top turns sorted by tokens_saved", () => {
331
+ const events = [
332
+ {
333
+ id: 1,
334
+ ts: "t1",
335
+ session_id: "s1",
336
+ pid: 1,
337
+ turn: 1,
338
+ mechanism: "graph_query",
339
+ tool: "search_code",
340
+ tokens_without: 1000,
341
+ tokens_with: 800,
342
+ tokens_saved: 200,
343
+ },
344
+ {
345
+ id: 2,
346
+ ts: "t2",
347
+ session_id: "s1",
348
+ pid: 1,
349
+ turn: 2,
350
+ mechanism: "graph_query",
351
+ tool: "get_callers",
352
+ tokens_without: 5000,
353
+ tokens_with: 1000,
354
+ tokens_saved: 4000,
355
+ },
356
+ {
357
+ id: 3,
358
+ ts: "t3",
359
+ session_id: "s1",
360
+ pid: 1,
361
+ turn: 3,
362
+ mechanism: "shell_compression",
363
+ tool: null,
364
+ tokens_without: 3000,
365
+ tokens_with: 1200,
366
+ tokens_saved: 1800,
367
+ },
368
+ {
369
+ id: 4,
370
+ ts: "t4",
371
+ session_id: "s1",
372
+ pid: 1,
373
+ turn: 2,
374
+ mechanism: "format_encoding",
375
+ tool: "get_callers",
376
+ tokens_without: 800,
377
+ tokens_with: 500,
378
+ tokens_saved: 300,
379
+ },
380
+ ];
381
+ const summary = aggregateSession(events, "s1");
382
+ expect(summary.top_turns[0].turn).toBe(2);
383
+ expect(summary.top_turns[0].tokens_saved).toBe(4300);
384
+ expect(summary.top_turns[0].tool).toBe("get_callers");
385
+ expect(summary.top_turns[0].primary_mechanism).toBe("graph_query");
386
+ expect(summary.top_turns[1].turn).toBe(3);
387
+ expect(summary.top_turns[1].tokens_saved).toBe(1800);
388
+ });
389
+ it("limits top_turns to 5", () => {
390
+ const events = [];
391
+ for (let i = 1; i <= 10; i++) {
392
+ events.push({
393
+ id: i,
394
+ ts: `t${i}`,
395
+ session_id: "s1",
396
+ pid: 1,
397
+ turn: i,
398
+ mechanism: "graph_query",
399
+ tool: `tool_${i}`,
400
+ tokens_without: i * 1000,
401
+ tokens_with: i * 100,
402
+ tokens_saved: i * 900,
403
+ });
404
+ }
405
+ const summary = aggregateSession(events, "s1");
406
+ expect(summary.top_turns).toHaveLength(5);
407
+ expect(summary.top_turns[0].turn).toBe(10);
408
+ });
409
+ it("returns zero efficiency when no events for session", () => {
410
+ const summary = aggregateSession([], "nonexistent");
411
+ expect(summary.total_turns).toBe(0);
412
+ expect(summary.total_tokens_saved).toBe(0);
413
+ expect(summary.efficiency_pct).toBe(0);
414
+ expect(summary.top_turns).toHaveLength(0);
415
+ });
416
+ it("handles multiple operations within same turn", () => {
417
+ const events = [
418
+ {
419
+ id: 1,
420
+ ts: "t1",
421
+ session_id: "s1",
422
+ pid: 1,
423
+ turn: 3,
424
+ mechanism: "graph_query",
425
+ tool: "get_callers",
426
+ tokens_without: 4800,
427
+ tokens_with: 1800,
428
+ tokens_saved: 3000,
429
+ },
430
+ {
431
+ id: 2,
432
+ ts: "t2",
433
+ session_id: "s1",
434
+ pid: 1,
435
+ turn: 3,
436
+ mechanism: "format_encoding",
437
+ tool: "get_callers",
438
+ tokens_without: 1800,
439
+ tokens_with: 1480,
440
+ tokens_saved: 320,
441
+ },
442
+ {
443
+ id: 3,
444
+ ts: "t3",
445
+ session_id: "s1",
446
+ pid: 1,
447
+ turn: 3,
448
+ mechanism: "session_dedup",
449
+ tool: "get_callers",
450
+ tokens_without: 480,
451
+ tokens_with: 0,
452
+ tokens_saved: 480,
453
+ },
454
+ ];
455
+ const summary = aggregateSession(events, "s1");
456
+ expect(summary.total_turns).toBe(1);
457
+ expect(summary.total_tokens_saved).toBe(3800);
458
+ const turn3 = summary.top_turns[0];
459
+ expect(turn3.turn).toBe(3);
460
+ expect(turn3.tokens_saved).toBe(3800);
461
+ expect(turn3.primary_mechanism).toBe("graph_query");
462
+ });
463
+ });
464
+ describe("aggregateByMechanism", () => {
465
+ it("aggregates across all sessions", () => {
466
+ const events = [
467
+ {
468
+ id: 1,
469
+ ts: "t1",
470
+ session_id: "s1",
471
+ pid: 1,
472
+ turn: 1,
473
+ mechanism: "graph_query",
474
+ tool: "t",
475
+ tokens_without: 100,
476
+ tokens_with: 50,
477
+ tokens_saved: 50,
478
+ },
479
+ {
480
+ id: 2,
481
+ ts: "t2",
482
+ session_id: "s2",
483
+ pid: 1,
484
+ turn: 1,
485
+ mechanism: "graph_query",
486
+ tool: "t",
487
+ tokens_without: 200,
488
+ tokens_with: 100,
489
+ tokens_saved: 100,
490
+ },
491
+ {
492
+ id: 3,
493
+ ts: "t3",
494
+ session_id: "s1",
495
+ pid: 1,
496
+ turn: 2,
497
+ mechanism: "shell_compression",
498
+ tool: null,
499
+ tokens_without: 300,
500
+ tokens_with: 100,
501
+ tokens_saved: 200,
502
+ },
503
+ ];
504
+ const breakdown = aggregateByMechanism(events);
505
+ expect(breakdown.graph_query.tokens_saved).toBe(150);
506
+ expect(breakdown.graph_query.event_count).toBe(2);
507
+ expect(breakdown.shell_compression.tokens_saved).toBe(200);
508
+ expect(breakdown.shell_compression.event_count).toBe(1);
509
+ });
510
+ it("computes correct percentages", () => {
511
+ const events = [
512
+ {
513
+ id: 1,
514
+ ts: "t1",
515
+ session_id: "s1",
516
+ pid: 1,
517
+ turn: 1,
518
+ mechanism: "graph_query",
519
+ tool: "t",
520
+ tokens_without: 1000,
521
+ tokens_with: 250,
522
+ tokens_saved: 750,
523
+ },
524
+ {
525
+ id: 2,
526
+ ts: "t2",
527
+ session_id: "s1",
528
+ pid: 1,
529
+ turn: 2,
530
+ mechanism: "shell_compression",
531
+ tool: null,
532
+ tokens_without: 500,
533
+ tokens_with: 250,
534
+ tokens_saved: 250,
535
+ },
536
+ ];
537
+ const breakdown = aggregateByMechanism(events);
538
+ expect(breakdown.graph_query.pct_of_total).toBe(75);
539
+ expect(breakdown.shell_compression.pct_of_total).toBe(25);
540
+ });
541
+ it("returns empty record for empty events", () => {
542
+ const breakdown = aggregateByMechanism([]);
543
+ expect(Object.keys(breakdown)).toHaveLength(0);
544
+ });
545
+ });
546
+ // ── High-volume insert Tests ──────────────────────────────────────
547
+ // (Replaces the prior "JSONL rotation" test — SQLite doesn't trim rows,
548
+ // it indexes them.)
549
+ describe("high-volume writes", () => {
550
+ it("persists 2200 events without loss", () => {
551
+ const writer = new TokenFlowWriter(unerrDir, "test-session-001");
552
+ for (let i = 0; i < 2200; i++) {
553
+ writer.record(makeInput({ turn: i, tokens_saved: i }));
554
+ }
555
+ const events = readTokenFlowEvents(unerrDir);
556
+ expect(events).toHaveLength(2200);
557
+ // Ordered by id ASC — the last inserted event is the last in the array.
558
+ expect(events[events.length - 1].turn).toBe(2199);
559
+ });
560
+ });
561
+ // ── Cross-Process Tests ───────────────────────────────────────────
562
+ describe("cross-process coordination", () => {
563
+ it("events from different PIDs coexist in same store", () => {
564
+ // Simulate events written by two child processes by inserting via
565
+ // the metrics-store directly with custom pid values.
566
+ const store = openMetricsStore(unerrDir);
567
+ store.insertTokenFlow({
568
+ ts: Date.parse("2026-01-01T00:00:00Z"),
569
+ ts_iso: "2026-01-01T00:00:00Z",
570
+ session_id: "s1",
571
+ pid: 12345,
572
+ turn: 1,
573
+ mechanism: "graph_query",
574
+ tool: "get_callers",
575
+ tokens_without: 5000,
576
+ tokens_with: 1800,
577
+ tokens_saved: 3200,
578
+ detail: null,
579
+ });
580
+ store.insertTokenFlow({
581
+ ts: Date.parse("2026-01-01T00:00:01Z"),
582
+ ts_iso: "2026-01-01T00:00:01Z",
583
+ session_id: "s1",
584
+ pid: 54321,
585
+ turn: 0,
586
+ mechanism: "shell_compression",
587
+ tool: null,
588
+ tokens_without: 3400,
589
+ tokens_with: 800,
590
+ tokens_saved: 2600,
591
+ detail: null,
592
+ });
593
+ const events = readTokenFlowEvents(unerrDir, { session_id: "s1" });
594
+ expect(events).toHaveLength(2);
595
+ const summary = aggregateSession(events, "s1");
596
+ expect(summary.total_tokens_saved).toBe(5800);
597
+ expect(summary.by_mechanism.graph_query.tokens_saved).toBe(3200);
598
+ expect(summary.by_mechanism.shell_compression.tokens_saved).toBe(2600);
599
+ });
600
+ it("session_id filtering isolates concurrent sessions", () => {
601
+ const store = openMetricsStore(unerrDir);
602
+ const baseTs = Date.parse("2026-01-01T00:00:00Z");
603
+ const insert = (sessionId, pid, turn, without, saved, offsetSec) => {
604
+ store.insertTokenFlow({
605
+ ts: baseTs + offsetSec * 1000,
606
+ ts_iso: new Date(baseTs + offsetSec * 1000).toISOString(),
607
+ session_id: sessionId,
608
+ pid,
609
+ turn,
610
+ mechanism: "graph_query",
611
+ tool: "t",
612
+ tokens_without: without,
613
+ tokens_with: without - saved,
614
+ tokens_saved: saved,
615
+ detail: null,
616
+ });
617
+ };
618
+ insert("session-a", 100, 1, 100, 50, 0);
619
+ insert("session-b", 200, 1, 200, 100, 1);
620
+ insert("session-a", 100, 2, 300, 150, 2);
621
+ const summaryA = aggregateSession(readTokenFlowEvents(unerrDir), "session-a");
622
+ const summaryB = aggregateSession(readTokenFlowEvents(unerrDir), "session-b");
623
+ expect(summaryA.total_tokens_saved).toBe(200);
624
+ expect(summaryA.total_turns).toBe(2);
625
+ expect(summaryB.total_tokens_saved).toBe(100);
626
+ expect(summaryB.total_turns).toBe(1);
627
+ });
628
+ });
629
+ // ── All Mechanism Types ───────────────────────────────────────────
630
+ describe("all mechanism types", () => {
631
+ it("handles all 7 mechanism types", () => {
632
+ const writer = new TokenFlowWriter(unerrDir, "s1");
633
+ const mechanisms = [
634
+ "graph_query",
635
+ "session_dedup",
636
+ "shell_compression",
637
+ "format_encoding",
638
+ "smart_truncation",
639
+ "file_read",
640
+ "behavior_automation",
641
+ ];
642
+ for (const [i, mechanism] of mechanisms.entries()) {
643
+ writer.record(makeInput({
644
+ session_id: "s1",
645
+ turn: i + 1,
646
+ mechanism,
647
+ tokens_saved: (i + 1) * 100,
648
+ tokens_without: (i + 1) * 200,
649
+ tokens_with: (i + 1) * 100,
650
+ }));
651
+ }
652
+ const summary = aggregateSession(writer.getSessionEvents(), "s1");
653
+ expect(Object.keys(summary.by_mechanism)).toHaveLength(7);
654
+ for (const mech of mechanisms) {
655
+ expect(summary.by_mechanism[mech]).toBeDefined();
656
+ }
657
+ });
658
+ });
659
+ // ── Performance Smoke Test ────────────────────────────────────────
660
+ describe("performance", () => {
661
+ it("record() completes in under 1ms per write", () => {
662
+ const writer = new TokenFlowWriter(unerrDir, "perf-test");
663
+ const iterations = 100;
664
+ const start = performance.now();
665
+ for (let i = 0; i < iterations; i++) {
666
+ writer.record(makeInput({ turn: i }));
667
+ }
668
+ const elapsed = performance.now() - start;
669
+ const perWrite = elapsed / iterations;
670
+ expect(perWrite).toBeLessThan(1);
671
+ });
672
+ it("aggregateSession handles 100 events in under 5ms", () => {
673
+ const events = [];
674
+ for (let i = 0; i < 100; i++) {
675
+ events.push({
676
+ id: i,
677
+ ts: new Date().toISOString(),
678
+ session_id: "s1",
679
+ pid: 1,
680
+ turn: Math.floor(i / 2) + 1,
681
+ mechanism: i % 2 === 0 ? "graph_query" : "shell_compression",
682
+ tool: i % 2 === 0 ? "get_callers" : null,
683
+ tokens_without: 1000 + i * 10,
684
+ tokens_with: 500 + i * 5,
685
+ tokens_saved: 500 + i * 5,
686
+ });
687
+ }
688
+ const start = performance.now();
689
+ const summary = aggregateSession(events, "s1");
690
+ const elapsed = performance.now() - start;
691
+ expect(elapsed).toBeLessThan(5);
692
+ expect(summary.total_turns).toBe(50);
693
+ });
694
+ });
695
+ });