@unerr-ai/unerr 0.1.0 → 0.1.2

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 +996 -710
  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 +312 -0
  553. package/dist/ui/assets/index-B-0HTtUR.js +0 -10
@@ -0,0 +1,193 @@
1
+ /**
2
+ * TR-1 / TR-2: filter + pagination + listSessions + getActivityBuckets store
3
+ * methods, plus the route surface that exposes them.
4
+ */
5
+ import { mkdirSync, rmSync } from "node:fs";
6
+ import { tmpdir } from "node:os";
7
+ import { join } from "node:path";
8
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
9
+ import { createTimelineRoutes } from "../server/routes/timeline.js";
10
+ import { CozoTimelineStore } from "../timeline/timeline-store.js";
11
+ let tempDir;
12
+ let store;
13
+ beforeEach(async () => {
14
+ tempDir = join(tmpdir(), `unerr-tf-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
15
+ mkdirSync(join(tempDir, ".unerr"), { recursive: true });
16
+ store = await CozoTimelineStore.create(tempDir);
17
+ });
18
+ afterEach(() => {
19
+ try {
20
+ store.close();
21
+ }
22
+ catch {
23
+ /* ignore */
24
+ }
25
+ try {
26
+ rmSync(tempDir, { recursive: true, force: true });
27
+ }
28
+ catch {
29
+ /* ignore */
30
+ }
31
+ });
32
+ function turn(overrides) {
33
+ return {
34
+ session_id: overrides.session_id ?? "s1",
35
+ started_at: overrides.started_at ?? 1_000,
36
+ ended_at: overrides.ended_at ?? 2_000,
37
+ opened_by: overrides.opened_by ?? "first_call",
38
+ closed_reason: overrides.closed_reason ?? "session_end",
39
+ tool_count: overrides.tool_count ?? 3,
40
+ file_count: overrides.file_count ?? 2,
41
+ edit_count: overrides.edit_count ?? 1,
42
+ title: overrides.title ?? "",
43
+ outcome: overrides.outcome ?? "unknown",
44
+ ...overrides,
45
+ };
46
+ }
47
+ describe("listTurns — filters + pagination", () => {
48
+ it("paginates via offset + limit", async () => {
49
+ for (let i = 0; i < 7; i++) {
50
+ await store.upsertTurn(turn({ turn_id: `t${i}`, started_at: 1000 + i * 1000 }));
51
+ }
52
+ const page1 = await store.listTurns({ limit: 3, offset: 0 });
53
+ const page2 = await store.listTurns({ limit: 3, offset: 3 });
54
+ const page3 = await store.listTurns({ limit: 3, offset: 6 });
55
+ expect(page1.map((t) => t.turn_id)).toEqual(["t6", "t5", "t4"]);
56
+ expect(page2.map((t) => t.turn_id)).toEqual(["t3", "t2", "t1"]);
57
+ expect(page3.map((t) => t.turn_id)).toEqual(["t0"]);
58
+ });
59
+ it("filters by fromTs / toTs (inclusive)", async () => {
60
+ await store.upsertTurn(turn({ turn_id: "a", started_at: 100 }));
61
+ await store.upsertTurn(turn({ turn_id: "b", started_at: 500 }));
62
+ await store.upsertTurn(turn({ turn_id: "c", started_at: 900 }));
63
+ const within = await store.listTurns({ fromTs: 200, toTs: 800 });
64
+ expect(within.map((t) => t.turn_id)).toEqual(["b"]);
65
+ });
66
+ it("filters by query — case-insensitive substring on title", async () => {
67
+ await store.upsertTurn(turn({ turn_id: "a", title: "Refactor Auth Middleware" }));
68
+ await store.upsertTurn(turn({ turn_id: "b", title: "Payments cleanup" }));
69
+ await store.upsertTurn(turn({ turn_id: "c", title: "auth follow-up" }));
70
+ const hits = await store.listTurns({ query: "auth" });
71
+ expect(hits.map((t) => t.turn_id).sort()).toEqual(["a", "c"]);
72
+ });
73
+ it("query escapes regex metacharacters in the user input", async () => {
74
+ await store.upsertTurn(turn({ turn_id: "a", title: "fix [bug] in handler" }));
75
+ await store.upsertTurn(turn({ turn_id: "b", title: "fix bug" }));
76
+ const hits = await store.listTurns({ query: "[bug]" });
77
+ expect(hits.map((t) => t.turn_id)).toEqual(["a"]);
78
+ });
79
+ it("countTurns mirrors the same filter semantics", async () => {
80
+ await store.upsertTurn(turn({ turn_id: "a", session_id: "s1", started_at: 100 }));
81
+ await store.upsertTurn(turn({ turn_id: "b", session_id: "s1", started_at: 200 }));
82
+ await store.upsertTurn(turn({ turn_id: "c", session_id: "s2", started_at: 150 }));
83
+ expect(await store.countTurns({})).toBe(3);
84
+ expect(await store.countTurns({ sessionId: "s1" })).toBe(2);
85
+ expect(await store.countTurns({ fromTs: 110, toTs: 250 })).toBe(2);
86
+ });
87
+ });
88
+ describe("listSessions", () => {
89
+ it("aggregates per-session stats and orders by last_seen desc", async () => {
90
+ await store.upsertTurn(turn({
91
+ turn_id: "x1",
92
+ session_id: "sa",
93
+ started_at: 100,
94
+ ended_at: 200,
95
+ edit_count: 1,
96
+ file_count: 2,
97
+ }));
98
+ await store.upsertTurn(turn({
99
+ turn_id: "x2",
100
+ session_id: "sa",
101
+ started_at: 300,
102
+ ended_at: 400,
103
+ edit_count: 2,
104
+ file_count: 3,
105
+ }));
106
+ await store.upsertTurn(turn({
107
+ turn_id: "x3",
108
+ session_id: "sb",
109
+ started_at: 500,
110
+ ended_at: 600,
111
+ edit_count: 1,
112
+ file_count: 1,
113
+ }));
114
+ const list = await store.listSessions();
115
+ expect(list.map((s) => s.session_id)).toEqual(["sb", "sa"]);
116
+ const sa = list.find((s) => s.session_id === "sa");
117
+ expect(sa.turn_count).toBe(2);
118
+ expect(sa.edit_count).toBe(3);
119
+ expect(sa.first_seen).toBe(100);
120
+ expect(sa.last_seen).toBe(400);
121
+ });
122
+ });
123
+ describe("getActivityBuckets", () => {
124
+ it("fills the requested range with zero days and counts turns per bucket", async () => {
125
+ const day = 24 * 60 * 60_000;
126
+ const t0 = day; // bucket-aligned for predictability
127
+ await store.upsertTurn(turn({
128
+ turn_id: "a",
129
+ started_at: t0,
130
+ ended_at: t0 + 5_000,
131
+ edit_count: 1,
132
+ tool_count: 4,
133
+ }));
134
+ await store.upsertTurn(turn({
135
+ turn_id: "b",
136
+ started_at: t0 + day + 1_000,
137
+ ended_at: t0 + day + 5_000,
138
+ edit_count: 3,
139
+ tool_count: 7,
140
+ }));
141
+ const buckets = await store.getActivityBuckets({
142
+ fromTs: t0,
143
+ toTs: t0 + 3 * day,
144
+ bucketMs: day,
145
+ });
146
+ expect(buckets).toHaveLength(4);
147
+ expect(buckets[0]?.turns).toBe(1);
148
+ expect(buckets[1]?.turns).toBe(1);
149
+ expect(buckets[1]?.edits).toBe(3);
150
+ expect(buckets[1]?.tools).toBe(7);
151
+ expect(buckets[2]?.turns).toBe(0);
152
+ expect(buckets[3]?.turns).toBe(0);
153
+ });
154
+ });
155
+ describe("HTTP routes — pagination + sessions + heatmap", () => {
156
+ it("GET /turns returns paginated envelope (total, returned, offset, limit)", async () => {
157
+ for (let i = 0; i < 5; i++) {
158
+ await store.upsertTurn(turn({ turn_id: `r${i}`, started_at: 1000 + i * 1000 }));
159
+ }
160
+ const app = createTimelineRoutes({ store });
161
+ const res = await app.request("/turns?limit=2&offset=2");
162
+ const body = (await res.json());
163
+ expect(body.total).toBe(5);
164
+ expect(body.returned).toBe(2);
165
+ expect(body.offset).toBe(2);
166
+ expect(body.limit).toBe(2);
167
+ expect(body.data.map((t) => t.turn_id)).toEqual(["r2", "r1"]);
168
+ });
169
+ it("GET /turns?q= performs a substring search", async () => {
170
+ await store.upsertTurn(turn({ turn_id: "a", title: "Auth refactor" }));
171
+ await store.upsertTurn(turn({ turn_id: "b", title: "Payments" }));
172
+ const app = createTimelineRoutes({ store });
173
+ const res = await app.request("/turns?q=auth");
174
+ const body = (await res.json());
175
+ expect(body.total).toBe(1);
176
+ expect(body.data[0]?.turn_id).toBe("a");
177
+ });
178
+ it("GET /sessions returns the aggregated session list", async () => {
179
+ await store.upsertTurn(turn({ turn_id: "x", session_id: "sa", started_at: 100, ended_at: 200 }));
180
+ const app = createTimelineRoutes({ store });
181
+ const res = await app.request("/sessions");
182
+ const body = (await res.json());
183
+ expect(body.data).toHaveLength(1);
184
+ expect(body.data[0]?.session_id).toBe("sa");
185
+ expect(body.data[0]?.turn_count).toBe(1);
186
+ });
187
+ it("GET /heatmap returns the requested number of buckets", async () => {
188
+ const app = createTimelineRoutes({ store });
189
+ const res = await app.request("/heatmap?days=7");
190
+ const body = (await res.json());
191
+ expect(body.data.length).toBe(7);
192
+ });
193
+ });
@@ -0,0 +1,151 @@
1
+ /**
2
+ * ST-2a: timeline marker tool handlers.
3
+ * Verifies dual-write (ledger + timeline.db.markers), validation, and turn
4
+ * stamping inheritance from the segmenter.
5
+ */
6
+ import { mkdirSync, readFileSync, rmSync } from "node:fs";
7
+ import { tmpdir } from "node:os";
8
+ import { join } from "node:path";
9
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
10
+ import { CozoTimelineStore } from "../timeline/timeline-store.js";
11
+ import { MARKER_TOOLS, handleMarkerCall, isMarkerTool, } from "../tools/intelligence/timeline-markers.js";
12
+ import { ShadowLedger } from "../tracking/shadow-ledger.js";
13
+ let tempDir;
14
+ let unerrDir;
15
+ let ledger;
16
+ let store;
17
+ beforeEach(async () => {
18
+ tempDir = join(tmpdir(), `unerr-mk-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
19
+ unerrDir = join(tempDir, ".unerr");
20
+ mkdirSync(unerrDir, { recursive: true });
21
+ ledger = new ShadowLedger(unerrDir);
22
+ store = await CozoTimelineStore.create(tempDir);
23
+ });
24
+ afterEach(() => {
25
+ try {
26
+ store.close();
27
+ }
28
+ catch {
29
+ /* ignore */
30
+ }
31
+ try {
32
+ rmSync(tempDir, { recursive: true, force: true });
33
+ }
34
+ catch {
35
+ /* ignore */
36
+ }
37
+ });
38
+ describe("isMarkerTool / MARKER_TOOLS", () => {
39
+ it("recognises all four marker names", () => {
40
+ expect(MARKER_TOOLS).toEqual([
41
+ "mark_intent",
42
+ "mark_decision",
43
+ "mark_blocker",
44
+ "mark_resolution",
45
+ ]);
46
+ for (const t of MARKER_TOOLS) {
47
+ expect(isMarkerTool(t)).toBe(true);
48
+ }
49
+ expect(isMarkerTool("file_read")).toBe(false);
50
+ expect(isMarkerTool("mark_anything_else")).toBe(false);
51
+ });
52
+ });
53
+ describe("handleMarkerCall — happy paths", () => {
54
+ it("mark_intent writes a ledger row AND a markers row", async () => {
55
+ const res = await handleMarkerCall("mark_intent", { text: "refactor auth" }, { ledger, store, branch: "main", headSha: "deadbeef" });
56
+ const body = JSON.parse(res.content[0].text);
57
+ expect(body.ok).toBe(true);
58
+ expect(body.type).toBe("mark_intent");
59
+ expect(body.marker_id).toMatch(/^[a-f0-9]{12}$/);
60
+ expect(body.turn_id).toMatch(/^[a-f0-9]{12}$/);
61
+ // Ledger row
62
+ const ledgerLines = readFileSync(join(unerrDir, "ledger", "shadow.jsonl"), "utf-8")
63
+ .trim()
64
+ .split("\n")
65
+ .map((l) => JSON.parse(l));
66
+ expect(ledgerLines).toHaveLength(1);
67
+ expect(ledgerLines[0].tool).toBe("mark_intent");
68
+ expect(ledgerLines[0].args_summary.text).toBe("refactor auth");
69
+ expect(ledgerLines[0].turn_id).toBe(body.turn_id);
70
+ // Timeline.db row
71
+ const markers = await store.listMarkers();
72
+ expect(markers).toHaveLength(1);
73
+ expect(markers[0].marker_id).toBe(body.marker_id);
74
+ expect(markers[0].type).toBe("mark_intent");
75
+ expect(markers[0].text).toBe("refactor auth");
76
+ expect(markers[0].turn_id).toBe(body.turn_id);
77
+ });
78
+ it("mark_decision persists alternatives (capped)", async () => {
79
+ await handleMarkerCall("mark_decision", {
80
+ text: "JWT over session cookies",
81
+ alternatives: ["session cookies", "OAuth proxy", "API tokens"],
82
+ }, { ledger, store, branch: "main", headSha: "x" });
83
+ const lines = readFileSync(join(unerrDir, "ledger", "shadow.jsonl"), "utf-8")
84
+ .trim()
85
+ .split("\n")
86
+ .map((l) => JSON.parse(l));
87
+ expect(lines[0].args_summary.alternatives).toEqual([
88
+ "session cookies",
89
+ "OAuth proxy",
90
+ "API tokens",
91
+ ]);
92
+ });
93
+ it("mark_blocker carries file_path through to timeline.db", async () => {
94
+ const res = await handleMarkerCall("mark_blocker", { text: "type error in verify", file_path: "src/auth/token.ts" }, { ledger, store, branch: "main", headSha: "x" });
95
+ const body = JSON.parse(res.content[0].text);
96
+ const markers = await store.listMarkers({ type: "mark_blocker" });
97
+ expect(markers).toHaveLength(1);
98
+ expect(markers[0].marker_id).toBe(body.marker_id);
99
+ expect(markers[0].file_path).toBe("src/auth/token.ts");
100
+ });
101
+ it("redacts secrets in BOTH the ledger row AND the timeline.db markers row", async () => {
102
+ // Regression — live test on 2026-05-12 found the redactor only ran on
103
+ // shadow.jsonl; raw tokens leaked into timeline.db.markers.text.
104
+ const tokenish = "GITHUB_TOKEN=ghp_AAAAAAAAAAAAAAAAAAAAAAAAAAAAAA";
105
+ const res = await handleMarkerCall("mark_blocker", { text: `auth fails when ${tokenish} is set`, file_path: "src/auth.ts" }, { ledger, store, branch: "main", headSha: "x" });
106
+ const body = JSON.parse(res.content[0].text);
107
+ expect(body.ok).toBe(true);
108
+ // Ledger redacted
109
+ const ledgerLines = readFileSync(join(unerrDir, "ledger", "shadow.jsonl"), "utf-8")
110
+ .trim()
111
+ .split("\n")
112
+ .map((l) => JSON.parse(l));
113
+ const ledgerRow = ledgerLines.find((r) => r.id === body.marker_id);
114
+ expect(ledgerRow.args_summary.text).not.toContain("ghp_");
115
+ expect(ledgerRow.args_summary.text).toContain("<redacted>");
116
+ // Timeline.db markers row also redacted (the bug)
117
+ const markers = await store.listMarkers({ type: "mark_blocker" });
118
+ const row = markers.find((m) => m.marker_id === body.marker_id);
119
+ expect(row).toBeDefined();
120
+ expect(row.text).not.toContain("ghp_");
121
+ expect(row.text).toContain("<redacted>");
122
+ });
123
+ it("mark_resolution links to blocker via blocker_ref", async () => {
124
+ const blocker = await handleMarkerCall("mark_blocker", { text: "type error" }, { ledger, store, branch: "main", headSha: "x" });
125
+ const blockerId = JSON.parse(blocker.content[0].text).marker_id;
126
+ const res = await handleMarkerCall("mark_resolution", { blocker_ref: blockerId, text: "fixed by import bump" }, { ledger, store, branch: "main", headSha: "x" });
127
+ const body = JSON.parse(res.content[0].text);
128
+ expect(body.ok).toBe(true);
129
+ const resolutions = await store.listMarkers({ type: "mark_resolution" });
130
+ expect(resolutions).toHaveLength(1);
131
+ expect(resolutions[0].blocker_ref).toBe(blockerId);
132
+ });
133
+ });
134
+ describe("handleMarkerCall — validation", () => {
135
+ it("rejects empty text", async () => {
136
+ const res = await handleMarkerCall("mark_intent", { text: " " }, { ledger, store, branch: "main", headSha: "x" });
137
+ const body = JSON.parse(res.content[0].text);
138
+ expect(body.error).toMatch(/required/);
139
+ });
140
+ it("rejects over-cap text per tool", async () => {
141
+ const long = "x".repeat(200);
142
+ const res = await handleMarkerCall("mark_intent", { text: long }, { ledger, store, branch: "main", headSha: "x" });
143
+ const body = JSON.parse(res.content[0].text);
144
+ expect(body.error).toMatch(/80/);
145
+ });
146
+ it("mark_resolution requires blocker_ref", async () => {
147
+ const res = await handleMarkerCall("mark_resolution", { text: "fixed" }, { ledger, store, branch: "main", headSha: "x" });
148
+ const body = JSON.parse(res.content[0].text);
149
+ expect(body.error).toMatch(/blocker_ref required/);
150
+ });
151
+ });
@@ -0,0 +1,156 @@
1
+ /**
2
+ * ST-3b: Timeline HTTP routes — smoke test against the Hono app.
3
+ */
4
+ import { mkdirSync, rmSync } 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 { createTimelineRoutes } from "../server/routes/timeline.js";
9
+ import { CozoTimelineStore } from "../timeline/timeline-store.js";
10
+ let tempDir;
11
+ let store;
12
+ beforeEach(async () => {
13
+ tempDir = join(tmpdir(), `unerr-rt-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
14
+ mkdirSync(join(tempDir, ".unerr"), { recursive: true });
15
+ store = await CozoTimelineStore.create(tempDir);
16
+ });
17
+ afterEach(() => {
18
+ try {
19
+ store.close();
20
+ }
21
+ catch {
22
+ /* ignore */
23
+ }
24
+ try {
25
+ rmSync(tempDir, { recursive: true, force: true });
26
+ }
27
+ catch {
28
+ /* ignore */
29
+ }
30
+ });
31
+ async function jsonOf(res) {
32
+ return await res.json();
33
+ }
34
+ describe("Timeline routes", () => {
35
+ it("GET /health reports the db path", async () => {
36
+ const app = createTimelineRoutes({ store });
37
+ const res = await app.request("/health");
38
+ expect(res.status).toBe(200);
39
+ const body = (await jsonOf(res));
40
+ expect(body.data.ok).toBe(true);
41
+ expect(body.data.db_path).toBe(store.dbPath);
42
+ });
43
+ it("GET /turns returns turns in newest-first order", async () => {
44
+ await store.upsertTurn({
45
+ turn_id: "t1",
46
+ session_id: "s1",
47
+ started_at: 100,
48
+ ended_at: 200,
49
+ opened_by: "first_call",
50
+ closed_reason: "session_end",
51
+ tool_count: 2,
52
+ file_count: 1,
53
+ edit_count: 0,
54
+ title: "early",
55
+ outcome: "unknown",
56
+ });
57
+ await store.upsertTurn({
58
+ turn_id: "t2",
59
+ session_id: "s1",
60
+ started_at: 300,
61
+ ended_at: 400,
62
+ opened_by: "idle_gap",
63
+ closed_reason: "idle_gap",
64
+ tool_count: 5,
65
+ file_count: 2,
66
+ edit_count: 1,
67
+ title: "late",
68
+ outcome: "unknown",
69
+ });
70
+ const app = createTimelineRoutes({ store });
71
+ const res = await app.request("/turns");
72
+ const body = (await jsonOf(res));
73
+ expect(body.data.map((t) => t.turn_id)).toEqual(["t2", "t1"]);
74
+ });
75
+ it("GET /loops uses the ledger getter for live detection", async () => {
76
+ const t = (offsetSec) => new Date(Date.parse("2026-05-12T10:00:00Z") + offsetSec * 1000).toISOString();
77
+ let id = 0;
78
+ const make = (tool, file, ts) => {
79
+ id += 1;
80
+ return {
81
+ id: `e${id}`,
82
+ ts,
83
+ tool,
84
+ args_summary: { file_path: file },
85
+ result_summary: {},
86
+ branch: "main",
87
+ head_sha: "x",
88
+ session_id: "s1",
89
+ correlation_id: null,
90
+ };
91
+ };
92
+ const entries = [
93
+ make("file_read", "a.ts", t(0)),
94
+ make("file_read", "a.ts", t(10)),
95
+ make("file_read", "a.ts", t(20)),
96
+ make("file_read", "a.ts", t(30)),
97
+ make("file_read", "a.ts", t(40)),
98
+ ];
99
+ const app = createTimelineRoutes({
100
+ store,
101
+ getRecentLedgerEntries: () => entries,
102
+ });
103
+ const res = await app.request("/loops");
104
+ const body = (await jsonOf(res));
105
+ expect(body.data).toHaveLength(1);
106
+ expect(body.data[0]?.kind).toBe("file_reread");
107
+ });
108
+ it("GET /resume returns null when no recent turn exists", async () => {
109
+ const app = createTimelineRoutes({ store });
110
+ const res = await app.request("/resume");
111
+ const body = (await jsonOf(res));
112
+ expect(body.data).toBeNull();
113
+ });
114
+ it("GET /resume surfaces dominant intent + open blockers for the latest fresh turn", async () => {
115
+ const now = Date.now();
116
+ await store.upsertTurn({
117
+ turn_id: "tx",
118
+ session_id: "sx",
119
+ started_at: now - 60_000,
120
+ ended_at: now - 30_000,
121
+ opened_by: "first_call",
122
+ closed_reason: "session_end",
123
+ tool_count: 3,
124
+ file_count: 1,
125
+ edit_count: 0,
126
+ title: "auth refactor",
127
+ outcome: "unknown",
128
+ });
129
+ await store.insertMarker({
130
+ marker_id: "i1",
131
+ type: "mark_intent",
132
+ text: "harden auth flow",
133
+ session_id: "sx",
134
+ turn_id: "tx",
135
+ ts: now - 50_000,
136
+ blocker_ref: "",
137
+ file_path: "",
138
+ });
139
+ await store.insertMarker({
140
+ marker_id: "b1",
141
+ type: "mark_blocker",
142
+ text: "type error in verify",
143
+ session_id: "sx",
144
+ turn_id: "tx",
145
+ ts: now - 40_000,
146
+ blocker_ref: "",
147
+ file_path: "src/auth.ts",
148
+ });
149
+ const app = createTimelineRoutes({ store });
150
+ const res = await app.request("/resume");
151
+ const body = (await jsonOf(res));
152
+ expect(body.data.intent).toBe("harden auth flow");
153
+ expect(body.data.session_id).toBe("sx");
154
+ expect(body.data.open_threads.map((t) => t.text)).toContain("type error in verify");
155
+ });
156
+ });
@@ -0,0 +1,171 @@
1
+ /**
2
+ * ST-1b: CozoTimelineStore — third CozoDB instance at `.unerr/timeline.db`.
3
+ * Tests cover open/init, idempotence, and round-trip CRUD on turns + markers.
4
+ */
5
+ import { existsSync, mkdirSync, rmSync } from "node:fs";
6
+ import { tmpdir } from "node:os";
7
+ import { join } from "node:path";
8
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
9
+ import { CozoTimelineStore, } from "../timeline/timeline-store.js";
10
+ let tempDir;
11
+ beforeEach(() => {
12
+ tempDir = join(tmpdir(), `unerr-tl-test-${Date.now()}-${Math.random().toString(36).slice(2)}`);
13
+ mkdirSync(tempDir, { recursive: true });
14
+ });
15
+ afterEach(() => {
16
+ try {
17
+ rmSync(tempDir, { recursive: true, force: true });
18
+ }
19
+ catch {
20
+ /* ignore */
21
+ }
22
+ });
23
+ function makeTurn(overrides = {}) {
24
+ return {
25
+ turn_id: overrides.turn_id ?? "t1",
26
+ session_id: overrides.session_id ?? "s1",
27
+ started_at: overrides.started_at ?? 1000,
28
+ ended_at: overrides.ended_at ?? 2000,
29
+ opened_by: overrides.opened_by ?? "first_call",
30
+ closed_reason: overrides.closed_reason ?? "session_end",
31
+ tool_count: overrides.tool_count ?? 3,
32
+ file_count: overrides.file_count ?? 2,
33
+ edit_count: overrides.edit_count ?? 1,
34
+ title: overrides.title ?? "",
35
+ outcome: overrides.outcome ?? "unknown",
36
+ };
37
+ }
38
+ function makeMarker(overrides = {}) {
39
+ return {
40
+ marker_id: overrides.marker_id ?? "m1",
41
+ type: overrides.type ?? "mark_intent",
42
+ text: overrides.text ?? "refactor auth",
43
+ session_id: overrides.session_id ?? "s1",
44
+ turn_id: overrides.turn_id ?? "t1",
45
+ ts: overrides.ts ?? 1500,
46
+ blocker_ref: overrides.blocker_ref ?? "",
47
+ file_path: overrides.file_path ?? "",
48
+ };
49
+ }
50
+ describe("CozoTimelineStore", () => {
51
+ it("creates .unerr/timeline.db on first call and reports isNew=true", async () => {
52
+ const store = await CozoTimelineStore.create(tempDir);
53
+ try {
54
+ expect(store.isNew).toBe(true);
55
+ expect(store.dbPath).toBe(join(tempDir, ".unerr", "timeline.db"));
56
+ expect(existsSync(store.dbPath)).toBe(true);
57
+ }
58
+ finally {
59
+ store.close();
60
+ }
61
+ });
62
+ it("second create on the same path reports isNew=false and schema-init is idempotent", async () => {
63
+ const a = await CozoTimelineStore.create(tempDir);
64
+ a.close();
65
+ const b = await CozoTimelineStore.create(tempDir);
66
+ try {
67
+ expect(b.isNew).toBe(false);
68
+ }
69
+ finally {
70
+ b.close();
71
+ }
72
+ });
73
+ it("initialises all six expected relations", async () => {
74
+ const store = await CozoTimelineStore.create(tempDir);
75
+ try {
76
+ const rels = await store.getDb().run("::relations");
77
+ const names = new Set(rels.rows.map((r) => r[0]));
78
+ for (const expected of [
79
+ "turns",
80
+ "intents",
81
+ "intent_sessions",
82
+ "markers",
83
+ "derived_signals",
84
+ "signal_reinforcement",
85
+ ]) {
86
+ expect(names.has(expected)).toBe(true);
87
+ }
88
+ }
89
+ finally {
90
+ store.close();
91
+ }
92
+ });
93
+ it("round-trips a turn through upsertTurn → listTurns", async () => {
94
+ const store = await CozoTimelineStore.create(tempDir);
95
+ try {
96
+ const turn = makeTurn({ title: "auth refactor" });
97
+ await store.upsertTurn(turn);
98
+ const list = await store.listTurns();
99
+ expect(list).toHaveLength(1);
100
+ expect(list[0]?.turn_id).toBe(turn.turn_id);
101
+ expect(list[0]?.title).toBe("auth refactor");
102
+ expect(list[0]?.opened_by).toBe("first_call");
103
+ }
104
+ finally {
105
+ store.close();
106
+ }
107
+ });
108
+ it("upsertTurn replaces an existing turn with the same id", async () => {
109
+ const store = await CozoTimelineStore.create(tempDir);
110
+ try {
111
+ await store.upsertTurn(makeTurn({ title: "v1", tool_count: 1 }));
112
+ await store.upsertTurn(makeTurn({ title: "v2", tool_count: 5 }));
113
+ const list = await store.listTurns();
114
+ expect(list).toHaveLength(1);
115
+ expect(list[0]?.title).toBe("v2");
116
+ expect(list[0]?.tool_count).toBe(5);
117
+ }
118
+ finally {
119
+ store.close();
120
+ }
121
+ });
122
+ it("listTurns filters by session and orders newest-first", async () => {
123
+ const store = await CozoTimelineStore.create(tempDir);
124
+ try {
125
+ await store.upsertTurn(makeTurn({ turn_id: "t1", session_id: "a", started_at: 100 }));
126
+ await store.upsertTurn(makeTurn({ turn_id: "t2", session_id: "a", started_at: 200 }));
127
+ await store.upsertTurn(makeTurn({ turn_id: "t3", session_id: "b", started_at: 150 }));
128
+ const inA = await store.listTurns({ sessionId: "a" });
129
+ expect(inA.map((t) => t.turn_id)).toEqual(["t2", "t1"]);
130
+ const all = await store.listTurns();
131
+ expect(all.map((t) => t.turn_id)).toEqual(["t2", "t3", "t1"]);
132
+ }
133
+ finally {
134
+ store.close();
135
+ }
136
+ });
137
+ it("round-trips markers through insertMarker → listMarkers with type filter", async () => {
138
+ const store = await CozoTimelineStore.create(tempDir);
139
+ try {
140
+ await store.insertMarker(makeMarker({
141
+ marker_id: "m1",
142
+ type: "mark_intent",
143
+ text: "auth",
144
+ ts: 100,
145
+ }));
146
+ await store.insertMarker(makeMarker({
147
+ marker_id: "m2",
148
+ type: "mark_blocker",
149
+ text: "type error",
150
+ ts: 200,
151
+ }));
152
+ await store.insertMarker(makeMarker({
153
+ marker_id: "m3",
154
+ type: "mark_resolution",
155
+ text: "fixed",
156
+ ts: 300,
157
+ blocker_ref: "m2",
158
+ }));
159
+ const all = await store.listMarkers();
160
+ expect(all.map((m) => m.marker_id)).toEqual(["m3", "m2", "m1"]);
161
+ const blockers = await store.listMarkers({ type: "mark_blocker" });
162
+ expect(blockers).toHaveLength(1);
163
+ expect(blockers[0]?.text).toBe("type error");
164
+ const resolutions = await store.listMarkers({ type: "mark_resolution" });
165
+ expect(resolutions[0]?.blocker_ref).toBe("m2");
166
+ }
167
+ finally {
168
+ store.close();
169
+ }
170
+ });
171
+ });