@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,223 @@
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
+ describe("MetricsStore", () => {
7
+ let dir;
8
+ beforeEach(() => {
9
+ dir = join(os.tmpdir(), `unerr-metrics-${Date.now()}-${Math.random()}`);
10
+ mkdirSync(dir, { recursive: true });
11
+ });
12
+ afterEach(() => {
13
+ closeMetricsStore(dir);
14
+ rmSync(dir, { recursive: true, force: true });
15
+ });
16
+ it("creates metrics.db on first open and is idempotent", () => {
17
+ openMetricsStore(dir);
18
+ expect(existsSync(join(dir, "metrics.db"))).toBe(true);
19
+ // Re-opening returns the same instance (no error)
20
+ expect(openMetricsStore(dir)).toBe(openMetricsStore(dir));
21
+ });
22
+ it("inserts + reads compression events with monotonic id", () => {
23
+ const s = openMetricsStore(dir);
24
+ const id1 = s.insertCompression({
25
+ ts: Date.now(),
26
+ ts_iso: new Date().toISOString(),
27
+ command: "ls",
28
+ category: "log_text",
29
+ confidence: 0.9,
30
+ raw_bytes: 1024,
31
+ compressed_bytes: 512,
32
+ saved_pct: 50,
33
+ omni_fallback: 0,
34
+ tee_file: null,
35
+ });
36
+ const id2 = s.insertCompression({
37
+ ts: Date.now() + 1,
38
+ ts_iso: new Date().toISOString(),
39
+ command: "ps",
40
+ category: "tabular",
41
+ confidence: 0.95,
42
+ raw_bytes: 2048,
43
+ compressed_bytes: 256,
44
+ saved_pct: 87.5,
45
+ omni_fallback: 0,
46
+ tee_file: ".unerr/tee/foo.txt",
47
+ });
48
+ expect(id2).toBeGreaterThan(id1);
49
+ const recent = s.recentCompression(10);
50
+ expect(recent).toHaveLength(2);
51
+ // ORDER BY id DESC — newest first
52
+ expect(recent[0]?.command).toBe("ps");
53
+ expect(recent[1]?.command).toBe("ls");
54
+ });
55
+ it("supports id > lastSeen polling for compression", () => {
56
+ const s = openMetricsStore(dir);
57
+ for (let i = 0; i < 3; i++) {
58
+ s.insertCompression({
59
+ ts: Date.now() + i,
60
+ ts_iso: new Date().toISOString(),
61
+ command: `cmd-${i}`,
62
+ category: "log_text",
63
+ confidence: 0.9,
64
+ raw_bytes: 100,
65
+ compressed_bytes: 50,
66
+ saved_pct: 50,
67
+ omni_fallback: 0,
68
+ tee_file: null,
69
+ });
70
+ }
71
+ const all = s.compressionSince(0);
72
+ expect(all).toHaveLength(3);
73
+ expect(all.map((r) => r.command)).toEqual(["cmd-0", "cmd-1", "cmd-2"]);
74
+ const afterFirst = s.compressionSince(all[0].id);
75
+ expect(afterFirst).toHaveLength(2);
76
+ expect(afterFirst[0]?.command).toBe("cmd-1");
77
+ });
78
+ it("inserts + reads file_read events", () => {
79
+ const s = openMetricsStore(dir);
80
+ s.insertFileRead({
81
+ ts: Date.now(),
82
+ ts_iso: new Date().toISOString(),
83
+ file: "src/foo.ts",
84
+ mode: "outline",
85
+ total_lines: 500,
86
+ returned_lines: 30,
87
+ saved_pct: 94,
88
+ entity: null,
89
+ token_estimate: 200,
90
+ });
91
+ const all = s.fileReadsSince(0);
92
+ expect(all).toHaveLength(1);
93
+ expect(all[0]?.file).toBe("src/foo.ts");
94
+ expect(all[0]?.mode).toBe("outline");
95
+ });
96
+ it("inserts + reads token_flow events; filters by session", () => {
97
+ const s = openMetricsStore(dir);
98
+ s.insertTokenFlow({
99
+ ts: Date.now(),
100
+ ts_iso: new Date().toISOString(),
101
+ session_id: "s-1",
102
+ pid: 1234,
103
+ turn: 1,
104
+ mechanism: "graph_query",
105
+ tool: "search_code",
106
+ tokens_without: 1000,
107
+ tokens_with: 200,
108
+ tokens_saved: 800,
109
+ detail: JSON.stringify({ query: "foo" }),
110
+ });
111
+ s.insertTokenFlow({
112
+ ts: Date.now() + 1,
113
+ ts_iso: new Date().toISOString(),
114
+ session_id: "s-2",
115
+ pid: 1234,
116
+ turn: 1,
117
+ mechanism: "shell_compression",
118
+ tool: null,
119
+ tokens_without: 5000,
120
+ tokens_with: 500,
121
+ tokens_saved: 4500,
122
+ detail: null,
123
+ });
124
+ expect(s.allTokenFlow()).toHaveLength(2);
125
+ expect(s.tokenFlowBySession("s-1")).toHaveLength(1);
126
+ expect(s.tokenFlowBySession("s-2")[0]?.mechanism).toBe("shell_compression");
127
+ });
128
+ it("upserts session_history (one row per session_id)", () => {
129
+ const s = openMetricsStore(dir);
130
+ s.upsertSessionHistory({
131
+ session_id: "s-1",
132
+ started_at: "2026-05-12T10:00:00Z",
133
+ ended_at: "2026-05-12T10:30:00Z",
134
+ duration_ms: 1_800_000,
135
+ tool_calls: 50,
136
+ tokens_saved: 12000,
137
+ tokens_processed: 50000,
138
+ efficiency: 24,
139
+ dollars_saved: 0.15,
140
+ model_id: "claude-opus-4-7",
141
+ entity_count: 30,
142
+ agent_name: "claude-code",
143
+ token_flow_summary: null,
144
+ });
145
+ // Second upsert for the same session_id should update, not append.
146
+ s.upsertSessionHistory({
147
+ session_id: "s-1",
148
+ started_at: "2026-05-12T10:00:00Z",
149
+ ended_at: "2026-05-12T11:00:00Z",
150
+ duration_ms: 3_600_000,
151
+ tool_calls: 80,
152
+ tokens_saved: 20000,
153
+ tokens_processed: 80000,
154
+ efficiency: 25,
155
+ dollars_saved: 0.25,
156
+ model_id: "claude-opus-4-7",
157
+ entity_count: 45,
158
+ agent_name: "claude-code",
159
+ token_flow_summary: JSON.stringify({ top_mechanism: "graph_query" }),
160
+ });
161
+ const all = s.allSessionHistory();
162
+ expect(all).toHaveLength(1);
163
+ expect(all[0]?.tool_calls).toBe(80);
164
+ expect(all[0]?.duration_ms).toBe(3_600_000);
165
+ });
166
+ it("upserts session_summary and reads by id", () => {
167
+ const s = openMetricsStore(dir);
168
+ s.upsertSessionSummary({
169
+ session_id: "sum-1",
170
+ written_at: "2026-05-12T10:30:00Z",
171
+ started_at: "2026-05-12T10:00:00Z",
172
+ ended_at: "2026-05-12T10:30:00Z",
173
+ duration_ms: 1_800_000,
174
+ tool_calls: 50,
175
+ chains: 5,
176
+ files_modified: JSON.stringify(["a.ts", "b.ts"]),
177
+ entities_touched: JSON.stringify(["fooFn", "barClass"]),
178
+ tools_used: JSON.stringify({ search_code: 10, file_read: 20 }),
179
+ feature_areas: JSON.stringify(["intelligence"]),
180
+ facts_recorded: 2,
181
+ facts_surfaced: JSON.stringify(["f-1"]),
182
+ revert_count: 0,
183
+ rot_score: 0.1,
184
+ token_estimate: 50000,
185
+ branch: "main",
186
+ });
187
+ const r = s.sessionSummary("sum-1");
188
+ expect(r?.session_id).toBe("sum-1");
189
+ expect(r?.chains).toBe(5);
190
+ expect(JSON.parse(r.files_modified)).toEqual(["a.ts", "b.ts"]);
191
+ expect(s.sessionSummary("never-existed")).toBeNull();
192
+ });
193
+ it("reset() empties tables and resets autoincrement", () => {
194
+ const s = openMetricsStore(dir);
195
+ s.insertCompression({
196
+ ts: Date.now(),
197
+ ts_iso: new Date().toISOString(),
198
+ command: "x",
199
+ category: "log_text",
200
+ confidence: 1,
201
+ raw_bytes: 1,
202
+ compressed_bytes: 1,
203
+ saved_pct: 0,
204
+ omni_fallback: 0,
205
+ tee_file: null,
206
+ });
207
+ s.reset();
208
+ expect(s.recentCompression(10)).toEqual([]);
209
+ const firstAfterReset = s.insertCompression({
210
+ ts: Date.now(),
211
+ ts_iso: new Date().toISOString(),
212
+ command: "y",
213
+ category: "log_text",
214
+ confidence: 1,
215
+ raw_bytes: 1,
216
+ compressed_bytes: 1,
217
+ saved_pct: 0,
218
+ omni_fallback: 0,
219
+ tee_file: null,
220
+ });
221
+ expect(firstAfterReset).toBe(1);
222
+ });
223
+ });
@@ -0,0 +1,191 @@
1
+ /**
2
+ * Native watcher tests — @parcel/watcher integration with debounced events.
3
+ */
4
+ import { mkdirSync, realpathSync, rmSync, unlinkSync, writeFileSync, } from "node:fs";
5
+ import { tmpdir } from "node:os";
6
+ import { join } from "node:path";
7
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
8
+ import { createNativeWatcher, } from "../tracking/native-watcher.js";
9
+ let tempDir;
10
+ let watcher = null;
11
+ function makeTempDir() {
12
+ const raw = join(tmpdir(), `unerr-watcher-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
13
+ mkdirSync(raw, { recursive: true });
14
+ return realpathSync(raw);
15
+ }
16
+ /**
17
+ * Wait until the predicate matches at least one event, or timeout.
18
+ * Returns all collected events at the time the predicate was satisfied.
19
+ */
20
+ function waitForMatch(collected, predicate, timeoutMs = 3000) {
21
+ return new Promise((resolve) => {
22
+ const start = Date.now();
23
+ const check = () => {
24
+ if (collected.some(predicate)) {
25
+ resolve([...collected]);
26
+ return;
27
+ }
28
+ if (Date.now() - start > timeoutMs) {
29
+ resolve([...collected]);
30
+ return;
31
+ }
32
+ setTimeout(check, 20);
33
+ };
34
+ check();
35
+ });
36
+ }
37
+ function delay(ms) {
38
+ return new Promise((resolve) => setTimeout(resolve, ms));
39
+ }
40
+ /**
41
+ * Start watcher and drain the initial FSEvents directory-creation event
42
+ * that fires when subscribing to a newly-created temp directory.
43
+ */
44
+ async function startAndDrain(w, events) {
45
+ await w.start();
46
+ await delay(500);
47
+ events.length = 0;
48
+ }
49
+ beforeEach(() => {
50
+ tempDir = makeTempDir();
51
+ });
52
+ afterEach(async () => {
53
+ if (watcher?.isRunning()) {
54
+ await watcher.stop();
55
+ }
56
+ watcher = null;
57
+ try {
58
+ rmSync(tempDir, { recursive: true, force: true });
59
+ }
60
+ catch {
61
+ /* ignore */
62
+ }
63
+ });
64
+ describe("NativeWatcher", () => {
65
+ it("can be created and started without errors", async () => {
66
+ const events = [];
67
+ watcher = createNativeWatcher({
68
+ projectRoot: tempDir,
69
+ onEvents: (batch) => events.push(...batch),
70
+ });
71
+ await watcher.start();
72
+ expect(watcher.isRunning()).toBe(true);
73
+ await watcher.stop();
74
+ expect(watcher.isRunning()).toBe(false);
75
+ });
76
+ it("detects file creation", async () => {
77
+ const events = [];
78
+ watcher = createNativeWatcher({
79
+ projectRoot: tempDir,
80
+ debounceMs: 30,
81
+ onEvents: (batch) => events.push(...batch),
82
+ });
83
+ await startAndDrain(watcher, events);
84
+ writeFileSync(join(tempDir, "new-file.txt"), "hello");
85
+ const result = await waitForMatch(events, (e) => e.path.includes("new-file.txt"));
86
+ const createEvent = result.find((e) => e.path.includes("new-file.txt"));
87
+ expect(createEvent).toBeDefined();
88
+ expect(createEvent?.type).toBe("create");
89
+ });
90
+ it("detects file modification", async () => {
91
+ const filePath = join(tempDir, "existing.txt");
92
+ writeFileSync(filePath, "initial content");
93
+ const events = [];
94
+ watcher = createNativeWatcher({
95
+ projectRoot: tempDir,
96
+ debounceMs: 30,
97
+ onEvents: (batch) => events.push(...batch),
98
+ });
99
+ await startAndDrain(watcher, events);
100
+ writeFileSync(filePath, "updated content");
101
+ const result = await waitForMatch(events, (e) => e.path.includes("existing.txt"));
102
+ const updateEvent = result.find((e) => e.path.includes("existing.txt"));
103
+ expect(updateEvent).toBeDefined();
104
+ expect(["create", "update"]).toContain(updateEvent?.type);
105
+ });
106
+ it("detects file deletion", async () => {
107
+ const filePath = join(tempDir, "to-delete.txt");
108
+ writeFileSync(filePath, "will be deleted");
109
+ const events = [];
110
+ watcher = createNativeWatcher({
111
+ projectRoot: tempDir,
112
+ debounceMs: 30,
113
+ onEvents: (batch) => events.push(...batch),
114
+ });
115
+ await startAndDrain(watcher, events);
116
+ unlinkSync(filePath);
117
+ const result = await waitForMatch(events, (e) => e.path.includes("to-delete.txt"));
118
+ const deleteEvent = result.find((e) => e.path.includes("to-delete.txt"));
119
+ expect(deleteEvent).toBeDefined();
120
+ expect(deleteEvent?.type).toBe("delete");
121
+ });
122
+ it("debounces rapid writes into a single batch", async () => {
123
+ const batches = [];
124
+ watcher = createNativeWatcher({
125
+ projectRoot: tempDir,
126
+ debounceMs: 150,
127
+ onEvents: (batch) => batches.push([...batch]),
128
+ });
129
+ await watcher.start();
130
+ await delay(500);
131
+ batches.length = 0;
132
+ for (let i = 0; i < 5; i++) {
133
+ writeFileSync(join(tempDir, `rapid-${i}.txt`), `content-${i}`);
134
+ }
135
+ await delay(500);
136
+ expect(batches.length).toBeLessThanOrEqual(2);
137
+ const allEvents = batches.flat();
138
+ expect(allEvents.length).toBeGreaterThanOrEqual(1);
139
+ });
140
+ it("ignores .git directory changes", async () => {
141
+ const gitDir = join(tempDir, ".git");
142
+ mkdirSync(gitDir, { recursive: true });
143
+ const events = [];
144
+ watcher = createNativeWatcher({
145
+ projectRoot: tempDir,
146
+ debounceMs: 30,
147
+ onEvents: (batch) => events.push(...batch),
148
+ });
149
+ await startAndDrain(watcher, events);
150
+ writeFileSync(join(gitDir, "HEAD"), "ref: refs/heads/main");
151
+ writeFileSync(join(tempDir, "tracked.txt"), "should see this");
152
+ const result = await waitForMatch(events, (e) => e.path.includes("tracked.txt"));
153
+ const gitEvents = result.filter((e) => e.path.includes("/.git/") || e.path.endsWith("/.git"));
154
+ expect(gitEvents.length).toBe(0);
155
+ const trackedEvent = result.find((e) => e.path.includes("tracked.txt"));
156
+ expect(trackedEvent).toBeDefined();
157
+ });
158
+ it("ignores node_modules directory changes", async () => {
159
+ const nmDir = join(tempDir, "node_modules", "some-pkg");
160
+ mkdirSync(nmDir, { recursive: true });
161
+ const events = [];
162
+ watcher = createNativeWatcher({
163
+ projectRoot: tempDir,
164
+ debounceMs: 30,
165
+ onEvents: (batch) => events.push(...batch),
166
+ });
167
+ await startAndDrain(watcher, events);
168
+ writeFileSync(join(nmDir, "index.js"), "module.exports = {}");
169
+ writeFileSync(join(tempDir, "src-file.ts"), "export const x = 1;");
170
+ const result = await waitForMatch(events, (e) => e.path.includes("src-file.ts"));
171
+ const nmEvents = result.filter((e) => e.path.includes("node_modules"));
172
+ expect(nmEvents.length).toBe(0);
173
+ const srcEvent = result.find((e) => e.path.includes("src-file.ts"));
174
+ expect(srcEvent).toBeDefined();
175
+ });
176
+ it("stops cleanly and flushes pending events", async () => {
177
+ const events = [];
178
+ watcher = createNativeWatcher({
179
+ projectRoot: tempDir,
180
+ debounceMs: 5000,
181
+ onEvents: (batch) => events.push(...batch),
182
+ });
183
+ await startAndDrain(watcher, events);
184
+ writeFileSync(join(tempDir, "pending.txt"), "content");
185
+ await delay(300);
186
+ await watcher.stop();
187
+ expect(watcher.isRunning()).toBe(false);
188
+ const pendingEvent = events.find((e) => e.path.includes("pending.txt"));
189
+ expect(pendingEvent).toBeDefined();
190
+ });
191
+ });
@@ -0,0 +1,145 @@
1
+ import { beforeEach, describe, expect, it } from "vitest";
2
+ import { resetHookDedup } from "../hooks/hook-dedup.js";
3
+ import { runPostReadHook, runPreEditHook, runPreReadHook, } from "../hooks/navigation-hooks.js";
4
+ /**
5
+ * Tests for agent-aware navigation hook behavior.
6
+ *
7
+ * Claude Code hooks detect via `hook_event_name` field.
8
+ * Cursor hooks detect via `tool_name` + `cwd` fields (no hook_event_name).
9
+ * Cline hooks detect via `tool` + `params` + `event` fields.
10
+ *
11
+ * The key behavioral difference: Read-before-Edit warnings are Claude Code only.
12
+ */
13
+ // ── Helpers ──────────────────────────────────────────────────────────
14
+ function claudeCodePayload(toolInput) {
15
+ return JSON.stringify({
16
+ hook_event_name: "PreToolUse",
17
+ tool_input: toolInput,
18
+ });
19
+ }
20
+ function cursorPayload(toolInput) {
21
+ return JSON.stringify({
22
+ tool_name: "Read",
23
+ tool_input: toolInput,
24
+ cwd: "/project",
25
+ });
26
+ }
27
+ function clinePayload(path) {
28
+ return JSON.stringify({
29
+ tool: "read_file",
30
+ params: { path },
31
+ event: "pre_tool",
32
+ });
33
+ }
34
+ // ── preReadHook: Claude Code ─────────────────────────────────────────
35
+ describe("preReadHook — Claude Code", () => {
36
+ it("passthrough for targeted Read (offset/limit)", () => {
37
+ const result = JSON.parse(runPreReadHook(claudeCodePayload({ file_path: "src/foo.ts", offset: 10, limit: 20 })));
38
+ // Empty object = passthrough (no nudge)
39
+ expect(result).toEqual({});
40
+ });
41
+ it("nudges for full-file Read (no offset/limit)", () => {
42
+ const result = JSON.parse(runPreReadHook(claudeCodePayload({ file_path: "src/foo.ts" })));
43
+ const msg = result.hookSpecificOutput?.systemMessage ?? "";
44
+ expect(msg).toContain("ONLY for the Edit workflow");
45
+ expect(msg).toContain("offset/limit");
46
+ expect(msg).toContain("file_read");
47
+ });
48
+ it("still nudges for non-code files (preRead has no isCodeFile gate)", () => {
49
+ const result = JSON.parse(runPreReadHook(claudeCodePayload({ file_path: "README.md" })));
50
+ const msg = result.hookSpecificOutput?.systemMessage ?? "";
51
+ expect(msg).toContain("file_read");
52
+ });
53
+ });
54
+ // ── preReadHook: Non-Claude Code (Cursor) ────────────────────────────
55
+ describe("preReadHook — Cursor (non-Claude Code)", () => {
56
+ it("nudges toward file_read (no Edit workflow mention)", () => {
57
+ const result = JSON.parse(runPreReadHook(cursorPayload({ file_path: "src/foo.ts" })));
58
+ // Cursor adapter uses `agent_message` at root level for pre-tool-use nudges
59
+ const msg = result.agent_message ?? "";
60
+ // Should NOT mention Edit workflow or offset/limit requirement
61
+ expect(msg).not.toContain("ONLY for the Edit workflow");
62
+ // Should mention file_read as the preferred tool
63
+ expect(msg).toContain("file_read");
64
+ });
65
+ it("still nudges even with offset/limit (Cursor doesn't need targeted Read)", () => {
66
+ const result = JSON.parse(runPreReadHook(cursorPayload({ file_path: "src/foo.ts", offset: 10, limit: 20 })));
67
+ // Cursor adapter uses `agent_message` for pre-tool-use nudges
68
+ const msg = result.agent_message ?? "";
69
+ // Non-Claude Code: always nudge toward file_read, even with offset/limit
70
+ expect(msg).toContain("file_read");
71
+ });
72
+ });
73
+ // ── preEditHook: Claude Code ─────────────────────────────────────────
74
+ describe("preEditHook — Claude Code", () => {
75
+ it("includes Read prerequisite warning", () => {
76
+ const result = JSON.parse(runPreEditHook(claudeCodePayload({
77
+ file_path: "src/foo.ts",
78
+ old_string: "const x = 1",
79
+ new_string: "const x = 2",
80
+ })));
81
+ const msg = result.hookSpecificOutput?.systemMessage ?? "";
82
+ expect(msg).toContain("CRITICAL: Edit REQUIRES built-in Read");
83
+ expect(msg).toContain("file_read (MCP) does NOT satisfy this");
84
+ });
85
+ it("includes blast radius warning for signature changes", () => {
86
+ const result = JSON.parse(runPreEditHook(claudeCodePayload({
87
+ file_path: "src/foo.ts",
88
+ old_string: "export function doSomething(x: number)",
89
+ new_string: "export function doSomething(x: string)",
90
+ })));
91
+ const msg = result.hookSpecificOutput?.systemMessage ?? "";
92
+ expect(msg).toContain("CRITICAL: Edit REQUIRES built-in Read");
93
+ expect(msg).toContain("function/class signature");
94
+ expect(msg).toContain("get_references");
95
+ });
96
+ });
97
+ // ── preEditHook: Non-Claude Code (Cursor) ────────────────────────────
98
+ describe("preEditHook — Cursor (non-Claude Code)", () => {
99
+ it("does NOT include Read prerequisite warning", () => {
100
+ const stdin = JSON.stringify({
101
+ tool_name: "Edit",
102
+ tool_input: {
103
+ file_path: "src/foo.ts",
104
+ old_string: "const x = 1",
105
+ new_string: "const x = 2",
106
+ },
107
+ cwd: "/project",
108
+ });
109
+ const result = JSON.parse(runPreEditHook(stdin));
110
+ // Cursor adapter uses `agent_message` at root level for pre-tool-use nudges
111
+ const msg = result.agent_message ?? "";
112
+ // Should NOT mention Edit requires Read
113
+ expect(msg).not.toContain("CRITICAL: Edit REQUIRES built-in Read");
114
+ expect(msg).not.toContain("file_read (MCP) does NOT satisfy this");
115
+ // Should still mention get_references for blast radius
116
+ expect(msg).toContain("get_references");
117
+ });
118
+ });
119
+ // ── postReadHook: Claude Code vs Cursor ──────────────────────────────
120
+ describe("postReadHook — agent-aware enrichment", () => {
121
+ beforeEach(() => {
122
+ resetHookDedup();
123
+ });
124
+ it("Claude Code: mentions Edit workflow", () => {
125
+ const result = JSON.parse(runPostReadHook(claudeCodePayload({ file_path: "src/post-read-cc.ts" })));
126
+ const msg = result.hookSpecificOutput?.additionalContext ?? "";
127
+ expect(msg).toContain("Edit needs built-in Read first");
128
+ expect(msg).toContain("file_read");
129
+ });
130
+ it("Cursor: generic file_read suggestion (no Edit mention)", () => {
131
+ // Cursor postToolUse payload includes tool_output to distinguish from preToolUse
132
+ // Use a unique file path to avoid dedup collision with the Claude Code test above
133
+ const stdin = JSON.stringify({
134
+ tool_name: "Read",
135
+ tool_input: { file_path: "src/post-read-cursor.ts" },
136
+ tool_output: "file contents here",
137
+ cwd: "/project",
138
+ });
139
+ const result = JSON.parse(runPostReadHook(stdin));
140
+ // Cursor adapter uses `additional_context` at root level for post-tool-use enrichment
141
+ const msg = result.additional_context ?? "";
142
+ expect(msg).not.toContain("Edit needs built-in Read first");
143
+ expect(msg).toContain("file_read");
144
+ });
145
+ });
@@ -0,0 +1,116 @@
1
+ import { describe, expect, it } from "vitest";
2
+ import { detectAntiPatterns, detectInstableEntities, } from "../intelligence/negative-knowledge.js";
3
+ describe("detectAntiPatterns", () => {
4
+ it("detects modified-then-reverted pattern", () => {
5
+ const entries = [
6
+ {
7
+ id: "target",
8
+ ts: "2026-04-30T10:00:00Z",
9
+ tool: "sync_local_diff",
10
+ args_summary: { files: ["src/ok.ts"] },
11
+ result_summary: {},
12
+ },
13
+ {
14
+ id: "bad1",
15
+ ts: "2026-04-30T10:01:00Z",
16
+ tool: "sync_local_diff",
17
+ args_summary: { files: ["src/bad.ts"] },
18
+ result_summary: {},
19
+ },
20
+ {
21
+ id: "bad2",
22
+ ts: "2026-04-30T10:02:00Z",
23
+ tool: "sync_local_diff",
24
+ args_summary: { files: ["src/worse.ts"] },
25
+ result_summary: {},
26
+ },
27
+ {
28
+ id: "rewind",
29
+ ts: "2026-04-30T10:05:00Z",
30
+ tool: "revert_to_working_state",
31
+ args_summary: {},
32
+ result_summary: { rewind_target_id: "target" },
33
+ },
34
+ ];
35
+ const corrections = detectAntiPatterns(entries, "rewind");
36
+ expect(corrections.length).toBe(2);
37
+ expect(corrections[0]?.entityKey).toBe("src/bad.ts");
38
+ expect(corrections[0]?.pattern).toBe("modified-then-reverted");
39
+ expect(corrections[1]?.entityKey).toBe("src/worse.ts");
40
+ });
41
+ it("returns empty when rewind not found", () => {
42
+ const entries = [
43
+ {
44
+ id: "e1",
45
+ ts: "2026-04-30T10:00:00Z",
46
+ tool: "sync_local_diff",
47
+ args_summary: { files: ["a.ts"] },
48
+ result_summary: {},
49
+ },
50
+ ];
51
+ expect(detectAntiPatterns(entries, "nonexistent")).toEqual([]);
52
+ });
53
+ it("returns empty when no target entry found", () => {
54
+ const entries = [
55
+ {
56
+ id: "rewind",
57
+ ts: "2026-04-30T10:00:00Z",
58
+ tool: "revert_to_working_state",
59
+ args_summary: {},
60
+ result_summary: { rewind_target_id: "missing" },
61
+ },
62
+ ];
63
+ expect(detectAntiPatterns(entries, "rewind")).toEqual([]);
64
+ });
65
+ });
66
+ describe("detectInstableEntities", () => {
67
+ it("detects entities modified 3+ times rapidly", () => {
68
+ const now = Date.now();
69
+ const entries = [
70
+ {
71
+ id: "e1",
72
+ ts: new Date(now - 5 * 60000).toISOString(),
73
+ tool: "sync_local_diff",
74
+ args_summary: { files: ["src/flaky.ts"] },
75
+ result_summary: {},
76
+ },
77
+ {
78
+ id: "e2",
79
+ ts: new Date(now - 3 * 60000).toISOString(),
80
+ tool: "sync_local_diff",
81
+ args_summary: { files: ["src/flaky.ts"] },
82
+ result_summary: {},
83
+ },
84
+ {
85
+ id: "e3",
86
+ ts: new Date(now - 1 * 60000).toISOString(),
87
+ tool: "sync_local_diff",
88
+ args_summary: { files: ["src/flaky.ts"] },
89
+ result_summary: {},
90
+ },
91
+ ];
92
+ const corrections = detectInstableEntities(entries);
93
+ expect(corrections.length).toBe(1);
94
+ expect(corrections[0]?.entityKey).toBe("src/flaky.ts");
95
+ expect(corrections[0]?.pattern).toBe("rapid-modification");
96
+ });
97
+ it("ignores stable entities", () => {
98
+ const entries = [
99
+ {
100
+ id: "e1",
101
+ ts: "2026-04-01T10:00:00Z",
102
+ tool: "sync_local_diff",
103
+ args_summary: { files: ["src/stable.ts"] },
104
+ result_summary: {},
105
+ },
106
+ {
107
+ id: "e2",
108
+ ts: "2026-04-15T10:00:00Z",
109
+ tool: "sync_local_diff",
110
+ args_summary: { files: ["src/stable.ts"] },
111
+ result_summary: {},
112
+ },
113
+ ];
114
+ expect(detectInstableEntities(entries)).toEqual([]);
115
+ });
116
+ });