@unerr-ai/unerr 0.0.1 → 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 (556) hide show
  1. package/README.md +215 -35
  2. package/dist/__tests__/architecture-guard.test.js +122 -0
  3. package/dist/__tests__/arg-validator.test.js +205 -0
  4. package/dist/__tests__/ast-extractor.test.js +203 -0
  5. package/dist/__tests__/auto-bootstrap.test.js +280 -0
  6. package/dist/__tests__/background-indexer.test.js +228 -0
  7. package/dist/__tests__/blast-radius-engine.test.js +200 -0
  8. package/dist/__tests__/bridge-isolation.test.js +37 -0
  9. package/dist/__tests__/budget-enforcer.test.js +53 -0
  10. package/dist/__tests__/cfg-test-detection-perf.test.js +82 -0
  11. package/dist/__tests__/change-narrative.test.js +190 -0
  12. package/dist/__tests__/check-commit.test.js +258 -0
  13. package/dist/__tests__/checksum.test.js +34 -0
  14. package/dist/__tests__/commit-watcher.test.js +154 -0
  15. package/dist/__tests__/community-detection.test.js +179 -0
  16. package/dist/__tests__/community-tools.test.js +299 -0
  17. package/dist/__tests__/components.test.js +449 -0
  18. package/dist/__tests__/compression-log.test.js +174 -0
  19. package/dist/__tests__/compression-quality-monitor.test.js +40 -0
  20. package/dist/__tests__/config-healer.test.js +165 -0
  21. package/dist/__tests__/context-ledger.test.js +58 -0
  22. package/dist/__tests__/convention-detector.test.js +99 -0
  23. package/dist/__tests__/convention-learner.test.js +86 -0
  24. package/dist/__tests__/correction-detector.test.js +330 -0
  25. package/dist/__tests__/daemon-autostart-install.test.js +283 -0
  26. package/dist/__tests__/daemon-bridge.test.js +222 -0
  27. package/dist/__tests__/daemon-dashboard.test.js +202 -0
  28. package/dist/__tests__/daemon-registry.test.js +240 -0
  29. package/dist/__tests__/daemon-supervisor.test.js +318 -0
  30. package/dist/__tests__/daemon-version-check.test.js +275 -0
  31. package/dist/__tests__/decision-point-detector.test.js +98 -0
  32. package/dist/__tests__/deep-link.test.js +143 -0
  33. package/dist/__tests__/disallowed-tools.test.js +115 -0
  34. package/dist/__tests__/drift-tracker.test.js +582 -0
  35. package/dist/__tests__/durability-scorer.test.js +152 -0
  36. package/dist/__tests__/efficiency-tracker.test.js +65 -0
  37. package/dist/__tests__/enrich.test.js +144 -0
  38. package/dist/__tests__/entity-rewind.test.js +248 -0
  39. package/dist/__tests__/ephemeral.test.js +111 -0
  40. package/dist/__tests__/exploration-cost.test.js +93 -0
  41. package/dist/__tests__/fact-generator.test.js +197 -0
  42. package/dist/__tests__/file-l0-graph.test.js +244 -0
  43. package/dist/__tests__/file-logger.test.js +82 -0
  44. package/dist/__tests__/file-outline.test.js +141 -0
  45. package/dist/__tests__/file-read-protocol.test.js +188 -0
  46. package/dist/__tests__/format-encoder.test.js +233 -0
  47. package/dist/__tests__/git-attribution.test.js +259 -0
  48. package/dist/__tests__/graph-temporal-joiner.test.js +219 -0
  49. package/dist/__tests__/health-grade-enhanced.test.js +138 -0
  50. package/dist/__tests__/health-map-data.test.js +173 -0
  51. package/dist/__tests__/helpers/mcp-harness.js +45 -0
  52. package/dist/__tests__/helpers/mcp-harness.test.js +68 -0
  53. package/dist/__tests__/hook-dedup.test.js +112 -0
  54. package/dist/__tests__/hook-runner.test.js +253 -0
  55. package/dist/__tests__/indexer-cfg.test.js +185 -0
  56. package/dist/__tests__/indexer-cross-file.test.js +172 -0
  57. package/dist/__tests__/indexer-extraction.test.js +245 -0
  58. package/dist/__tests__/indexer-incremental.test.js +232 -0
  59. package/dist/__tests__/indexer-language-expansion.test.js +165 -0
  60. package/dist/__tests__/init-push.test.js +131 -0
  61. package/dist/__tests__/instruction-writer.test.js +179 -0
  62. package/dist/__tests__/intelligence-integration.test.js +217 -0
  63. package/dist/__tests__/intent-correlator.test.js +175 -0
  64. package/dist/__tests__/intent-detector.test.js +235 -0
  65. package/dist/__tests__/intent-encoder.test.js +167 -0
  66. package/dist/__tests__/java-build-tool-detection.test.js +174 -0
  67. package/dist/__tests__/layer3-sprint-q.test.js +160 -0
  68. package/dist/__tests__/layer3-sprint-r.test.js +91 -0
  69. package/dist/__tests__/layer3-sprint-s.test.js +183 -0
  70. package/dist/__tests__/layer3-sprint-t.test.js +201 -0
  71. package/dist/__tests__/layer3-sprint-u.test.js +174 -0
  72. package/dist/__tests__/layer4-sprint-ba2.test.js +354 -0
  73. package/dist/__tests__/layer4-sprint-ba4.test.js +84 -0
  74. package/dist/__tests__/layer4-sprint-vs.test.js +105 -0
  75. package/dist/__tests__/ledger-chains.test.js +162 -0
  76. package/dist/__tests__/lifecycle-machine.test.js +226 -0
  77. package/dist/__tests__/local-chat-provider.test.js +170 -0
  78. package/dist/__tests__/local-convention-detector.test.js +308 -0
  79. package/dist/__tests__/local-embeddings.test.js +422 -0
  80. package/dist/__tests__/local-graph.test.js +540 -0
  81. package/dist/__tests__/local-indexer.test.js +228 -0
  82. package/dist/__tests__/local-intelligence-l3.test.js +332 -0
  83. package/dist/__tests__/local-llm.test.js +253 -0
  84. package/dist/__tests__/local-mode-offline.test.js +187 -0
  85. package/dist/__tests__/local-mode-stats.test.js +273 -0
  86. package/dist/__tests__/local-mode-tui.test.js +343 -0
  87. package/dist/__tests__/local-parse.test.js +199 -0
  88. package/dist/__tests__/log-tailer.test.js +208 -0
  89. package/dist/__tests__/loop-breaker.test.js +276 -0
  90. package/dist/__tests__/loop-miner.test.js +226 -0
  91. package/dist/__tests__/mcp-config.test.js +126 -0
  92. package/dist/__tests__/mcp-content-json.test.js +10 -0
  93. package/dist/__tests__/mcp-envelope.test.js +124 -0
  94. package/dist/__tests__/metrics-store.test.js +223 -0
  95. package/dist/__tests__/native-watcher.test.js +191 -0
  96. package/dist/__tests__/navigation-hooks-agent-aware.test.js +145 -0
  97. package/dist/__tests__/negative-knowledge.test.js +116 -0
  98. package/dist/__tests__/network-boundary.test.js +190 -0
  99. package/dist/__tests__/network-firewall.test.js +112 -0
  100. package/dist/__tests__/nudge-invariants.test.js +160 -0
  101. package/dist/__tests__/nudge-v2.test.js +225 -0
  102. package/dist/__tests__/offline-rewind.test.js +251 -0
  103. package/dist/__tests__/open-threads.test.js +89 -0
  104. package/dist/__tests__/output-compressor.test.js +93 -0
  105. package/dist/__tests__/pending-violations.test.js +112 -0
  106. package/dist/__tests__/persistence-effectiveness.test.js +143 -0
  107. package/dist/__tests__/provider-factory.test.js +42 -0
  108. package/dist/__tests__/providers.test.js +24 -0
  109. package/dist/__tests__/proxy.test.js +314 -0
  110. package/dist/__tests__/query-router.test.js +1018 -0
  111. package/dist/__tests__/reasoning-quality-route.test.js +138 -0
  112. package/dist/__tests__/redactor.test.js +120 -0
  113. package/dist/__tests__/resource-monitor.test.js +57 -0
  114. package/dist/__tests__/response-envelope.test.js +100 -0
  115. package/dist/__tests__/risk-classifier.test.js +101 -0
  116. package/dist/__tests__/risk-signal-scope.test.js +75 -0
  117. package/dist/__tests__/rule-evaluator.test.js +280 -0
  118. package/dist/__tests__/scip-decoder.test.js +49 -0
  119. package/dist/__tests__/scip-downloader.test.js +201 -0
  120. package/dist/__tests__/scip-merger.test.js +103 -0
  121. package/dist/__tests__/search-index.test.js +422 -0
  122. package/dist/__tests__/semantic-enrichment.test.js +360 -0
  123. package/dist/__tests__/session-brief-builder.test.js +187 -0
  124. package/dist/__tests__/session-context.test.js +221 -0
  125. package/dist/__tests__/session-continuity.test.js +144 -0
  126. package/dist/__tests__/session-dedup.test.js +74 -0
  127. package/dist/__tests__/session-event-wiring.test.js +206 -0
  128. package/dist/__tests__/session-events.test.js +149 -0
  129. package/dist/__tests__/session-legend.test.js +20 -0
  130. package/dist/__tests__/session-persistence.test.js +131 -0
  131. package/dist/__tests__/session-resume-block.test.js +107 -0
  132. package/dist/__tests__/session-resume.test.js +97 -0
  133. package/dist/__tests__/session-summary-writer.test.js +134 -0
  134. package/dist/__tests__/shadow-ledger.test.js +203 -0
  135. package/dist/__tests__/shell-classifier.test.js +151 -0
  136. package/dist/__tests__/shell-compression-floor.test.js +189 -0
  137. package/dist/__tests__/shell-compression-v2.test.js +339 -0
  138. package/dist/__tests__/shell-compressor.test.js +35 -0
  139. package/dist/__tests__/shell-hooks.test.js +128 -0
  140. package/dist/__tests__/shell-strategies.test.js +644 -0
  141. package/dist/__tests__/shell-tee.test.js +133 -0
  142. package/dist/__tests__/signal-dedup.test.js +158 -0
  143. package/dist/__tests__/signal-reinforcer.test.js +77 -0
  144. package/dist/__tests__/signal-scorer.test.js +251 -0
  145. package/dist/__tests__/signal-show-store.test.js +108 -0
  146. package/dist/__tests__/smart-truncate.test.js +215 -0
  147. package/dist/__tests__/snapshot-v2.test.js +113 -0
  148. package/dist/__tests__/sprint-l1-local-mode.test.js +130 -0
  149. package/dist/__tests__/sprint-l10-boot.test.js +220 -0
  150. package/dist/__tests__/sprint-l9-offline-commands.test.js +189 -0
  151. package/dist/__tests__/sprint-q-persistent-context.test.js +198 -0
  152. package/dist/__tests__/sprint-s1-wiring.test.js +215 -0
  153. package/dist/__tests__/sprint-s2-wiring.test.js +256 -0
  154. package/dist/__tests__/sprint-s3-wiring.test.js +195 -0
  155. package/dist/__tests__/sprint-s4-wiring.test.js +213 -0
  156. package/dist/__tests__/sprint-s6-hooks.test.js +222 -0
  157. package/dist/__tests__/sprint-s7-persistent.test.js +263 -0
  158. package/dist/__tests__/sprint-s8-value.test.js +167 -0
  159. package/dist/__tests__/sprint-s9-behavioral.test.js +179 -0
  160. package/dist/__tests__/sprint3-intelligence.test.js +297 -0
  161. package/dist/__tests__/sprint5-mcp-server.test.js +136 -0
  162. package/dist/__tests__/startup-display.test.js +302 -0
  163. package/dist/__tests__/startup-log-file.test.js +97 -0
  164. package/dist/__tests__/stash-manager.test.js +229 -0
  165. package/dist/__tests__/state-detector.test.js +92 -0
  166. package/dist/__tests__/status-dashboard.test.js +142 -0
  167. package/dist/__tests__/temporal-facts.test.js +292 -0
  168. package/dist/__tests__/temporal-routes.test.js +142 -0
  169. package/dist/__tests__/test-detector.test.js +174 -0
  170. package/dist/__tests__/theme.test.js +72 -0
  171. package/dist/__tests__/timeline-agents.test.js +122 -0
  172. package/dist/__tests__/timeline-bootstrap.test.js +176 -0
  173. package/dist/__tests__/timeline-filters.test.js +193 -0
  174. package/dist/__tests__/timeline-markers.test.js +151 -0
  175. package/dist/__tests__/timeline-routes.test.js +156 -0
  176. package/dist/__tests__/timeline-store.test.js +171 -0
  177. package/dist/__tests__/token-counter.test.js +86 -0
  178. package/dist/__tests__/token-estimator.test.js +96 -0
  179. package/dist/__tests__/token-flow-api.test.js +239 -0
  180. package/dist/__tests__/token-flow-instrumentation.test.js +437 -0
  181. package/dist/__tests__/token-flow-persistence.test.js +356 -0
  182. package/dist/__tests__/token-flow-routes.test.js +199 -0
  183. package/dist/__tests__/token-flow.test.js +695 -0
  184. package/dist/__tests__/tool-clusters.test.js +177 -0
  185. package/dist/__tests__/transport-mux.test.js +283 -0
  186. package/dist/__tests__/turn-segmenter.test.js +166 -0
  187. package/dist/__tests__/uninstall.test.js +141 -0
  188. package/dist/__tests__/warm-start-policy.test.js +271 -0
  189. package/dist/__tests__/wire-cap-nudge.test.js +77 -0
  190. package/dist/__tests__/worker-pool.test.js +101 -0
  191. package/dist/behaviors/agent-llm-bridge.js +166 -0
  192. package/dist/behaviors/architecture-guard.js +256 -0
  193. package/dist/behaviors/auto-doc.js +247 -0
  194. package/dist/behaviors/cascade-guard.js +289 -0
  195. package/dist/behaviors/change-narrative.js +270 -0
  196. package/dist/behaviors/convention-drift.js +290 -0
  197. package/dist/behaviors/framework.js +235 -0
  198. package/dist/behaviors/guard-formatter.js +44 -0
  199. package/dist/behaviors/incomplete-work.js +270 -0
  200. package/dist/behaviors/loop-breaker.js +300 -0
  201. package/dist/behaviors/session-continuity.js +208 -0
  202. package/dist/cli.js +6446 -2227
  203. package/dist/commands/branches.js +97 -0
  204. package/dist/commands/check-commit.js +225 -0
  205. package/dist/commands/compress-output.js +64 -0
  206. package/dist/commands/config-verify.js +243 -0
  207. package/dist/commands/daemon.js +905 -0
  208. package/dist/commands/dashboard.js +52 -0
  209. package/dist/commands/debug.js +200 -0
  210. package/dist/commands/enrich.js +184 -0
  211. package/dist/commands/exec.js +233 -0
  212. package/dist/commands/gain.js +156 -0
  213. package/dist/commands/hook.js +88 -0
  214. package/dist/commands/index.js +88 -0
  215. package/dist/commands/init.js +74 -0
  216. package/dist/commands/install.js +505 -0
  217. package/dist/commands/learn.js +116 -0
  218. package/dist/commands/manifest.js +193 -0
  219. package/dist/commands/rewind.js +103 -0
  220. package/dist/commands/serve.js +19 -0
  221. package/dist/commands/setup-wizard.js +414 -0
  222. package/dist/commands/skills.js +64 -0
  223. package/dist/commands/stats.js +20 -0
  224. package/dist/commands/status.js +654 -0
  225. package/dist/commands/timeline.js +139 -0
  226. package/dist/commands/uninstall.js +230 -0
  227. package/dist/components/App.js +109 -0
  228. package/dist/components/Banner.js +12 -0
  229. package/dist/components/ConfirmPrompt.js +25 -0
  230. package/dist/components/DriftSummary.js +23 -0
  231. package/dist/components/GradeBadge.js +15 -0
  232. package/dist/components/HealthCard.js +18 -0
  233. package/dist/components/InkSpinner.js +22 -0
  234. package/dist/components/InputBox.js +17 -0
  235. package/dist/components/KeyValue.js +13 -0
  236. package/dist/components/MessageList.js +14 -0
  237. package/dist/components/ProgressBar.js +26 -0
  238. package/dist/components/Section.js +16 -0
  239. package/dist/components/SessionSummaryCard.js +73 -0
  240. package/dist/components/StartupDisplay.js +24 -0
  241. package/dist/components/StatusDashboard.js +57 -0
  242. package/dist/components/StatusLine.js +8 -0
  243. package/dist/components/StepLine.js +22 -0
  244. package/dist/components/Theme.js +20 -0
  245. package/dist/components/ToolProgress.js +8 -0
  246. package/dist/components/ViolationList.js +21 -0
  247. package/dist/components/render.js +13 -0
  248. package/dist/config/agent-registry.js +237 -0
  249. package/dist/config/claude-settings-hooks.js +304 -0
  250. package/dist/config/hook-installer.js +65 -0
  251. package/dist/config/instruction-writer.js +388 -0
  252. package/dist/config/mcp-config-writer.js +266 -0
  253. package/dist/config/settings.js +174 -0
  254. package/dist/config/tool-detector.js +42 -0
  255. package/dist/config/value-surfacing.js +119 -0
  256. package/dist/core/context-assembly.js +108 -0
  257. package/dist/core/conversation.js +33 -0
  258. package/dist/core/local-chat-provider.js +475 -0
  259. package/dist/core/provider-factory.js +55 -0
  260. package/dist/core/providers.js +90 -0
  261. package/dist/core/query-engine.js +174 -0
  262. package/dist/daemon/api.js +312 -0
  263. package/dist/daemon/autostart.js +119 -0
  264. package/dist/daemon/bootstrap.js +39 -0
  265. package/dist/daemon/client.js +164 -0
  266. package/dist/daemon/detect-ci.js +81 -0
  267. package/dist/daemon/platform-linux.js +146 -0
  268. package/dist/daemon/platform-macos.js +134 -0
  269. package/dist/daemon/platform-windows.js +116 -0
  270. package/dist/daemon/process-manager.js +299 -0
  271. package/dist/daemon/protocol.js +23 -0
  272. package/dist/daemon/registry.js +270 -0
  273. package/dist/daemon/settings-schema.js +72 -0
  274. package/dist/daemon/system-health.js +134 -0
  275. package/dist/daemon/version-checker.js +262 -0
  276. package/dist/daemon/warm-start.js +223 -0
  277. package/dist/entrypoints/cli.js +1043 -0
  278. package/dist/entrypoints/daemon.js +380 -0
  279. package/dist/entrypoints/repl.js +147 -0
  280. package/dist/hooks/adapters/claude-code.js +90 -0
  281. package/dist/hooks/adapters/cline.js +100 -0
  282. package/dist/hooks/adapters/cursor.js +98 -0
  283. package/dist/hooks/hook-dedup.js +79 -0
  284. package/dist/hooks/hook-runner.js +113 -0
  285. package/dist/hooks/navigation-hooks.js +175 -0
  286. package/dist/hooks/prompt-hooks.js +63 -0
  287. package/dist/hooks/shell-hooks.js +47 -0
  288. package/dist/ignore.js +111 -0
  289. package/dist/intelligence/approach-suggester.js +61 -0
  290. package/dist/intelligence/ast-extractor.js +2615 -0
  291. package/dist/intelligence/ast-worker.js +34 -0
  292. package/dist/intelligence/background-indexer.js +121 -0
  293. package/dist/intelligence/blast-radius.js +200 -0
  294. package/dist/intelligence/community-detection.js +691 -0
  295. package/dist/intelligence/community-detector.js +184 -0
  296. package/dist/intelligence/computation-scheduler.js +75 -0
  297. package/dist/intelligence/confidence-propagation.js +47 -0
  298. package/dist/intelligence/convention-detector.js +242 -0
  299. package/dist/intelligence/convention-learner.js +205 -0
  300. package/dist/intelligence/convention-matcher.js +205 -0
  301. package/dist/intelligence/cozo-schema.js +376 -0
  302. package/dist/intelligence/decision-point-detector.js +90 -0
  303. package/dist/intelligence/deep-dive-tools.js +586 -0
  304. package/dist/intelligence/durability-scorer.js +84 -0
  305. package/dist/intelligence/exploration-cost.js +204 -0
  306. package/dist/intelligence/exploration-pattern-tracker.js +61 -0
  307. package/dist/intelligence/fact-generator.js +322 -0
  308. package/dist/intelligence/facts-schema.js +90 -0
  309. package/dist/intelligence/file-intelligence.js +59 -0
  310. package/dist/intelligence/graph-holder.js +220 -0
  311. package/dist/intelligence/graph-temporal-joiner.js +238 -0
  312. package/dist/intelligence/health-grade.js +423 -0
  313. package/dist/intelligence/health-grader.js +200 -0
  314. package/dist/intelligence/health-map-data.js +259 -0
  315. package/dist/intelligence/import-symbols.js +136 -0
  316. package/dist/intelligence/incremental-indexer.js +658 -0
  317. package/dist/intelligence/indexer/centrality.js +62 -0
  318. package/dist/intelligence/indexer/cfg-context.js +95 -0
  319. package/dist/intelligence/indexer/confidence.js +34 -0
  320. package/dist/intelligence/indexer/cross-file-resolver.js +104 -0
  321. package/dist/intelligence/indexer/edge-repair.js +89 -0
  322. package/dist/intelligence/indexer/entity-key.js +17 -0
  323. package/dist/intelligence/indexer/export-map.js +132 -0
  324. package/dist/intelligence/indexer/git-cochange.js +128 -0
  325. package/dist/intelligence/indexer/graph-patch.js +147 -0
  326. package/dist/intelligence/indexer/incremental.js +78 -0
  327. package/dist/intelligence/indexer/ingest.js +160 -0
  328. package/dist/intelligence/indexer/language-detect.js +226 -0
  329. package/dist/intelligence/indexer/metadata.js +63 -0
  330. package/dist/intelligence/indexer/mutation-tracker.js +79 -0
  331. package/dist/intelligence/indexer/orchestrator.js +155 -0
  332. package/dist/intelligence/indexer/plugin-interface.js +31 -0
  333. package/dist/intelligence/indexer/plugins/csharp.js +440 -0
  334. package/dist/intelligence/indexer/plugins/go.js +335 -0
  335. package/dist/intelligence/indexer/plugins/java.js +370 -0
  336. package/dist/intelligence/indexer/plugins/python.js +358 -0
  337. package/dist/intelligence/indexer/plugins/regex-fallback.js +82 -0
  338. package/dist/intelligence/indexer/plugins/ruby.js +290 -0
  339. package/dist/intelligence/indexer/plugins/rust.js +484 -0
  340. package/dist/intelligence/indexer/plugins/tier2-generic.js +310 -0
  341. package/dist/intelligence/indexer/plugins/typescript.js +456 -0
  342. package/dist/intelligence/indexer/resource-monitor.js +93 -0
  343. package/dist/intelligence/indexer/scip/decoder.js +253 -0
  344. package/dist/intelligence/indexer/scip/detector.js +232 -0
  345. package/dist/intelligence/indexer/scip/downloader.js +427 -0
  346. package/dist/intelligence/indexer/scip/fallback.js +34 -0
  347. package/dist/intelligence/indexer/scip/merger.js +109 -0
  348. package/dist/intelligence/indexer/scip/orchestrator.js +433 -0
  349. package/dist/intelligence/indexer/scip/runner.js +98 -0
  350. package/dist/intelligence/indexer/snapshot.js +66 -0
  351. package/dist/intelligence/indexer/test-detector.js +196 -0
  352. package/dist/intelligence/indexer/watch-integration.js +61 -0
  353. package/dist/intelligence/indexer/worker.js +85 -0
  354. package/dist/intelligence/local-convention-detector.js +437 -0
  355. package/dist/intelligence/local-embeddings.js +190 -0
  356. package/dist/intelligence/local-graph.js +1946 -0
  357. package/dist/intelligence/local-indexer.js +1575 -0
  358. package/dist/intelligence/local-llm.js +163 -0
  359. package/dist/intelligence/local-rule-generator.js +154 -0
  360. package/dist/intelligence/local-snapshot.js +213 -0
  361. package/dist/intelligence/negative-knowledge.js +103 -0
  362. package/dist/intelligence/persistent-db.js +85 -0
  363. package/dist/intelligence/query-router.js +2556 -0
  364. package/dist/intelligence/risk-classifier.js +116 -0
  365. package/dist/intelligence/rule-evaluator.js +380 -0
  366. package/dist/intelligence/rule-generator.js +49 -0
  367. package/dist/intelligence/search-index.js +173 -0
  368. package/dist/intelligence/semantic/docstring-extractor.js +67 -0
  369. package/dist/intelligence/semantic/embedding-store.js +52 -0
  370. package/dist/intelligence/semantic/enrichment-orchestrator.js +48 -0
  371. package/dist/intelligence/semantic/git-message-miner.js +114 -0
  372. package/dist/intelligence/semantic/identifier-tokenizer.js +51 -0
  373. package/dist/intelligence/semantic/node2vec-embeddings.js +71 -0
  374. package/dist/intelligence/semantic/node2vec-walks.js +103 -0
  375. package/dist/intelligence/semantic/path-domain-inference.js +112 -0
  376. package/dist/intelligence/semantic/similarity-engine.js +60 -0
  377. package/dist/intelligence/semantic/tfidf-vectors.js +88 -0
  378. package/dist/intelligence/session-brief-builder.js +159 -0
  379. package/dist/intelligence/session-context.js +221 -0
  380. package/dist/intelligence/session-health-monitor.js +211 -0
  381. package/dist/intelligence/session-narrative.js +197 -0
  382. package/dist/intelligence/session-pattern-analyzer.js +218 -0
  383. package/dist/intelligence/signal-scorer.js +390 -0
  384. package/dist/intelligence/signal-show-store.js +182 -0
  385. package/dist/intelligence/smart-truncate.js +158 -0
  386. package/dist/intelligence/subgraph-cache.js +88 -0
  387. package/dist/intelligence/temporal-facts.js +494 -0
  388. package/dist/intelligence/token-estimator.js +100 -0
  389. package/dist/intelligence/tool-injector.js +87 -0
  390. package/dist/intelligence/tree-sitter-loader.js +71 -0
  391. package/dist/intelligence/worker-pool.js +116 -0
  392. package/dist/proxy/arg-validator.js +79 -0
  393. package/dist/proxy/auto-bootstrap.js +167 -0
  394. package/dist/proxy/bridge.js +147 -0
  395. package/dist/proxy/budget-enforcer.js +70 -0
  396. package/dist/proxy/compression-quality-monitor.js +160 -0
  397. package/dist/proxy/compression-stats.js +51 -0
  398. package/dist/proxy/context-rot-detector.js +137 -0
  399. package/dist/proxy/drift-detector.js +139 -0
  400. package/dist/proxy/efficiency-tracker.js +79 -0
  401. package/dist/proxy/fact-ranking.js +154 -0
  402. package/dist/proxy/format-encoder.js +266 -0
  403. package/dist/proxy/http-transport.js +90 -0
  404. package/dist/proxy/lifecycle-actor.js +55 -0
  405. package/dist/proxy/lifecycle-machine.js +187 -0
  406. package/dist/proxy/log-tailer.js +265 -0
  407. package/dist/proxy/model-pricing.js +98 -0
  408. package/dist/proxy/network-firewall.js +141 -0
  409. package/dist/proxy/nudge-state.js +93 -0
  410. package/dist/proxy/output-compressor.js +185 -0
  411. package/dist/proxy/pid-lock.js +291 -0
  412. package/dist/proxy/proxy-context.js +11 -0
  413. package/dist/proxy/proxy.js +2633 -0
  414. package/dist/proxy/response-enrichment.js +32 -0
  415. package/dist/proxy/response-envelope.js +313 -0
  416. package/dist/proxy/session-dedup.js +82 -0
  417. package/dist/proxy/session-legend.js +30 -0
  418. package/dist/proxy/session-persistence.js +210 -0
  419. package/dist/proxy/session-resume.js +94 -0
  420. package/dist/proxy/session-stats.js +513 -0
  421. package/dist/proxy/shell-classifier.js +1346 -0
  422. package/dist/proxy/shell-compression-log.js +93 -0
  423. package/dist/proxy/shell-compressor.js +390 -0
  424. package/dist/proxy/shell-graph-boost.js +202 -0
  425. package/dist/proxy/shell-monitor-map.js +18 -0
  426. package/dist/proxy/shell-stats.js +54 -0
  427. package/dist/proxy/shell-strategies/cloud.js +215 -0
  428. package/dist/proxy/shell-strategies/diff.js +159 -0
  429. package/dist/proxy/shell-strategies/error-diagnostic.js +796 -0
  430. package/dist/proxy/shell-strategies/filter-dsl.js +358 -0
  431. package/dist/proxy/shell-strategies/git-status.js +177 -0
  432. package/dist/proxy/shell-strategies/key-value.js +193 -0
  433. package/dist/proxy/shell-strategies/log-text.js +154 -0
  434. package/dist/proxy/shell-strategies/omni.js +188 -0
  435. package/dist/proxy/shell-strategies/progress.js +55 -0
  436. package/dist/proxy/shell-strategies/redact.js +76 -0
  437. package/dist/proxy/shell-strategies/structured.js +241 -0
  438. package/dist/proxy/shell-strategies/tabular.js +243 -0
  439. package/dist/proxy/shell-strategies/test-results-types.js +13 -0
  440. package/dist/proxy/shell-strategies/test-results.js +784 -0
  441. package/dist/proxy/shell-strategies/tree-paths.js +144 -0
  442. package/dist/proxy/shell-strategies/yaml.js +182 -0
  443. package/dist/proxy/shell-tee.js +111 -0
  444. package/dist/proxy/signal-dedup.js +171 -0
  445. package/dist/proxy/startup-renderer.js +158 -0
  446. package/dist/proxy/task-token-display.js +38 -0
  447. package/dist/proxy/token-counter.js +61 -0
  448. package/dist/proxy/tool-clusters.js +273 -0
  449. package/dist/proxy/tool-definitions.js +525 -0
  450. package/dist/proxy/transport-mux.js +229 -0
  451. package/dist/proxy/wire-cap.js +268 -0
  452. package/dist/schemas/api/skills.js +19 -0
  453. package/dist/schemas/common/errors.js +7 -0
  454. package/dist/schemas/common/headers.js +5 -0
  455. package/dist/schemas/entities/edge.js +25 -0
  456. package/dist/schemas/entities/entity.js +22 -0
  457. package/dist/schemas/entities/rule.js +18 -0
  458. package/dist/schemas/index.js +14 -0
  459. package/dist/server/event-bus.js +59 -0
  460. package/dist/server/http.js +156 -0
  461. package/dist/server/middleware.js +70 -0
  462. package/dist/server/routes/drift.js +97 -0
  463. package/dist/server/routes/intelligence.js +1217 -0
  464. package/dist/server/routes/reasoning-quality.js +444 -0
  465. package/dist/server/routes/session.js +86 -0
  466. package/dist/server/routes/stream.js +120 -0
  467. package/dist/server/routes/system.js +73 -0
  468. package/dist/server/routes/temporal.js +170 -0
  469. package/dist/server/routes/timeline.js +232 -0
  470. package/dist/server/routes/token-flow.js +403 -0
  471. package/dist/skills/effectiveness-tracker.js +93 -0
  472. package/dist/skills/local-pack.js +380 -0
  473. package/dist/skills/resolver.js +495 -0
  474. package/dist/state-detector.js +83 -0
  475. package/dist/timeline/intent-detector.js +263 -0
  476. package/dist/timeline/loop-miner.js +140 -0
  477. package/dist/timeline/open-threads.js +49 -0
  478. package/dist/timeline/signal-reinforcer.js +62 -0
  479. package/dist/timeline/timeline-bootstrap.js +151 -0
  480. package/dist/timeline/timeline-store.js +618 -0
  481. package/dist/tools/coding/bash.js +49 -0
  482. package/dist/tools/coding/file-edit.js +72 -0
  483. package/dist/tools/coding/file-outline.js +227 -0
  484. package/dist/tools/coding/file-read-protocol.js +425 -0
  485. package/dist/tools/coding/file-read.js +35 -0
  486. package/dist/tools/coding/file-write.js +43 -0
  487. package/dist/tools/coding/glob-tool.js +109 -0
  488. package/dist/tools/coding/grep.js +162 -0
  489. package/dist/tools/coding/index.js +27 -0
  490. package/dist/tools/intelligence/index.js +269 -0
  491. package/dist/tools/intelligence/record-fact.js +48 -0
  492. package/dist/tools/intelligence/timeline-markers.js +130 -0
  493. package/dist/tools/registry.js +47 -0
  494. package/dist/tools/types.js +8 -0
  495. package/dist/tracking/auto-snapshot-triggers.js +246 -0
  496. package/dist/tracking/branch-context.js +115 -0
  497. package/dist/tracking/branch-snapshot.js +217 -0
  498. package/dist/tracking/causal-bridge.js +317 -0
  499. package/dist/tracking/circuit-breaker.js +147 -0
  500. package/dist/tracking/commit-watcher.js +114 -0
  501. package/dist/tracking/context-ledger.js +119 -0
  502. package/dist/tracking/correction-detector.js +324 -0
  503. package/dist/tracking/drift-tracker.js +874 -0
  504. package/dist/tracking/durability-tracker.js +94 -0
  505. package/dist/tracking/entity-rewind.js +200 -0
  506. package/dist/tracking/file-hash-state.js +114 -0
  507. package/dist/tracking/git-attribution.js +132 -0
  508. package/dist/tracking/git-trailers.js +171 -0
  509. package/dist/tracking/intelligence-counter.js +46 -0
  510. package/dist/tracking/intent-correlator.js +202 -0
  511. package/dist/tracking/intent-encoder.js +52 -0
  512. package/dist/tracking/intent-token-tracker.js +159 -0
  513. package/dist/tracking/ledger-archiver.js +94 -0
  514. package/dist/tracking/ledger-chains.js +245 -0
  515. package/dist/tracking/metrics-store.js +361 -0
  516. package/dist/tracking/native-watcher.js +131 -0
  517. package/dist/tracking/offline-rewind.js +295 -0
  518. package/dist/tracking/pending-violations.js +74 -0
  519. package/dist/tracking/persistence-effectiveness.js +167 -0
  520. package/dist/tracking/prompt-durability.js +202 -0
  521. package/dist/tracking/quality-signals.js +213 -0
  522. package/dist/tracking/redactor.js +73 -0
  523. package/dist/tracking/rewind-engine.js +161 -0
  524. package/dist/tracking/session-history.js +128 -0
  525. package/dist/tracking/session-receipt.js +88 -0
  526. package/dist/tracking/session-summary-writer.js +157 -0
  527. package/dist/tracking/shadow-ledger.js +321 -0
  528. package/dist/tracking/stash-manager.js +258 -0
  529. package/dist/tracking/timeline-fork.js +213 -0
  530. package/dist/tracking/timeline.js +69 -0
  531. package/dist/tracking/token-flow.js +276 -0
  532. package/dist/tracking/turn-segmenter.js +122 -0
  533. package/dist/tracking/weekly-accumulator.js +179 -0
  534. package/dist/tracking/working-snapshots.js +188 -0
  535. package/dist/tracking/workspace-manifest.js +176 -0
  536. package/dist/transport/http.js +102 -0
  537. package/dist/ui/assets/index-7gl3mIuY.css +1 -0
  538. package/dist/ui/assets/index-BsMTQdhX.js +10 -0
  539. package/dist/ui/index.html +2 -2
  540. package/dist/utils/counterfactual.js +65 -0
  541. package/dist/utils/deep-link.js +34 -0
  542. package/dist/utils/detect.js +193 -0
  543. package/dist/utils/exec.js +73 -0
  544. package/dist/utils/file-logger.js +87 -0
  545. package/dist/utils/format-error.js +29 -0
  546. package/dist/utils/git.js +181 -0
  547. package/dist/utils/log.js +57 -0
  548. package/dist/utils/logger.js +35 -0
  549. package/dist/utils/mcp-content-json.js +8 -0
  550. package/dist/utils/session-logger.js +154 -0
  551. package/dist/utils/startup-log.js +512 -0
  552. package/dist/utils/ui.js +56 -0
  553. package/package.json +5 -3
  554. package/scripts/postinstall.mjs +299 -0
  555. package/dist/ui/assets/index-BISLlJyc.js +0 -10
  556. package/dist/ui/assets/index-BUChTv4H.css +0 -1
@@ -0,0 +1,475 @@
1
+ /**
2
+ * Local Chat Provider — BYO-LLM streaming chat for the REPL.
3
+ *
4
+ * Implements the ChatProvider interface by routing to Ollama (/api/chat),
5
+ * LM Studio / OpenAI-compatible (/v1/chat/completions), or anthropic-direct.
6
+ * All providers use Server-Sent Events or NDJSON streaming.
7
+ *
8
+ * SECURITY: API keys are NEVER logged or included in MCP responses.
9
+ */
10
+ import Anthropic from "@anthropic-ai/sdk";
11
+ // ── Convert unerr Tool → ChatToolDef ────────────────────────
12
+ export function toChatToolDefs(tools) {
13
+ return tools.map((t) => ({
14
+ type: "function",
15
+ function: {
16
+ name: t.name,
17
+ description: t.description,
18
+ parameters: t.inputSchema,
19
+ },
20
+ }));
21
+ }
22
+ // ── Anthropic Chat Provider ─────────────────────────────────
23
+ export class AnthropicChatProvider {
24
+ providerName = "anthropic";
25
+ modelId;
26
+ client;
27
+ constructor(apiKey, model) {
28
+ this.client = new Anthropic({ apiKey });
29
+ this.modelId = model;
30
+ }
31
+ async streamChat(messages, tools, systemPrompt, maxTokens, onChunk) {
32
+ // Convert ChatMessage[] → Anthropic format
33
+ const anthropicMessages = this.toAnthropicMessages(messages);
34
+ const anthropicTools = tools.map((t) => ({
35
+ name: t.function.name,
36
+ description: t.function.description,
37
+ input_schema: t.function.parameters,
38
+ }));
39
+ const stream = this.client.messages.stream({
40
+ model: this.modelId,
41
+ max_tokens: maxTokens,
42
+ system: systemPrompt,
43
+ messages: anthropicMessages,
44
+ tools: anthropicTools.length > 0 ? anthropicTools : undefined,
45
+ });
46
+ let text = "";
47
+ stream.on("text", (t) => {
48
+ text += t;
49
+ onChunk({ type: "text_delta", text: t });
50
+ });
51
+ const message = await stream.finalMessage();
52
+ const toolCalls = [];
53
+ for (const block of message.content) {
54
+ if (block.type === "tool_use") {
55
+ toolCalls.push({
56
+ id: block.id,
57
+ name: block.name,
58
+ args: block.input,
59
+ });
60
+ }
61
+ }
62
+ onChunk({ type: "done" });
63
+ return {
64
+ text,
65
+ toolCalls,
66
+ inputTokens: message.usage.input_tokens,
67
+ outputTokens: message.usage.output_tokens,
68
+ };
69
+ }
70
+ toAnthropicMessages(messages) {
71
+ const result = [];
72
+ for (const msg of messages) {
73
+ if (msg.role === "system")
74
+ continue; // System prompt handled separately
75
+ if (msg.role === "tool") {
76
+ result.push({
77
+ role: "user",
78
+ content: [
79
+ {
80
+ type: "tool_result",
81
+ tool_use_id: msg.tool_call_id ?? "",
82
+ content: msg.content,
83
+ },
84
+ ],
85
+ });
86
+ }
87
+ else if (msg.role === "assistant" && msg.tool_calls?.length) {
88
+ const blocks = [];
89
+ if (msg.content)
90
+ blocks.push({ type: "text", text: msg.content });
91
+ for (const tc of msg.tool_calls) {
92
+ blocks.push({
93
+ type: "tool_use",
94
+ id: tc.id,
95
+ name: tc.function.name,
96
+ input: JSON.parse(tc.function.arguments),
97
+ });
98
+ }
99
+ result.push({ role: "assistant", content: blocks });
100
+ }
101
+ else {
102
+ result.push({
103
+ role: msg.role,
104
+ content: msg.content,
105
+ });
106
+ }
107
+ }
108
+ return result;
109
+ }
110
+ }
111
+ // ── Local Chat Provider ─────────────────────────────────────
112
+ const DEFAULT_BASE_URLS = {
113
+ ollama: "http://localhost:11434",
114
+ "lm-studio": "http://localhost:1234",
115
+ "openai-compatible": "http://localhost:8080",
116
+ "anthropic-direct": "https://api.anthropic.com",
117
+ };
118
+ export class LocalChatProvider {
119
+ providerName;
120
+ modelId;
121
+ baseUrl;
122
+ apiKey;
123
+ useOllamaApi;
124
+ constructor(config) {
125
+ this.providerName = config.provider;
126
+ this.modelId = config.chatModel;
127
+ this.baseUrl = (config.baseUrl ??
128
+ DEFAULT_BASE_URLS[config.provider] ??
129
+ "http://localhost:8080").replace(/\/+$/, "");
130
+ this.apiKey = config.apiKey;
131
+ this.useOllamaApi = config.provider === "ollama";
132
+ }
133
+ async streamChat(messages, tools, systemPrompt, maxTokens, onChunk) {
134
+ // anthropic-direct uses the Anthropic SDK directly
135
+ if (this.providerName === "anthropic-direct") {
136
+ return this.streamAnthropicDirect(messages, tools, systemPrompt, maxTokens, onChunk);
137
+ }
138
+ if (this.useOllamaApi) {
139
+ return this.streamOllama(messages, tools, systemPrompt, maxTokens, onChunk);
140
+ }
141
+ return this.streamOpenAiCompatible(messages, tools, systemPrompt, maxTokens, onChunk);
142
+ }
143
+ async streamAnthropicDirect(messages, tools, systemPrompt, maxTokens, onChunk) {
144
+ if (!this.apiKey) {
145
+ throw new Error("[LocalChatProvider] anthropic-direct requires an API key. " +
146
+ "Set localLlm.apiKey in settings.json.");
147
+ }
148
+ const provider = new AnthropicChatProvider(this.apiKey, this.modelId);
149
+ return provider.streamChat(messages, tools, systemPrompt, maxTokens, onChunk);
150
+ }
151
+ // ── Ollama: POST /api/chat with NDJSON streaming ──────────
152
+ async streamOllama(messages, tools, systemPrompt, _maxTokens, onChunk) {
153
+ const ollamaMessages = [
154
+ { role: "system", content: systemPrompt },
155
+ ...messages.map((m) => ({
156
+ role: m.role === "tool" ? "user" : m.role,
157
+ content: m.content,
158
+ })),
159
+ ];
160
+ const body = {
161
+ model: this.modelId,
162
+ messages: ollamaMessages,
163
+ stream: true,
164
+ };
165
+ // Ollama supports tools via the OpenAI-compatible format
166
+ if (tools.length > 0) {
167
+ body.tools = tools;
168
+ }
169
+ const response = await fetch(`${this.baseUrl}/api/chat`, {
170
+ method: "POST",
171
+ headers: { "Content-Type": "application/json" },
172
+ body: JSON.stringify(body),
173
+ });
174
+ if (!response.ok) {
175
+ const errText = await response.text().catch(() => "");
176
+ throw new Error(`[LocalChatProvider] Ollama chat failed: ${response.status} ${response.statusText}${errText ? ` — ${errText.slice(0, 200)}` : ""}`);
177
+ }
178
+ return this.parseNdjsonStream(response, onChunk);
179
+ }
180
+ async parseNdjsonStream(response, onChunk) {
181
+ const reader = response.body?.getReader();
182
+ if (!reader)
183
+ throw new Error("[LocalChatProvider] No response body");
184
+ const decoder = new TextDecoder();
185
+ let text = "";
186
+ const toolCalls = [];
187
+ let inputTokens = 0;
188
+ let outputTokens = 0;
189
+ let buffer = "";
190
+ while (true) {
191
+ const { done, value } = await reader.read();
192
+ if (done)
193
+ break;
194
+ buffer += decoder.decode(value, { stream: true });
195
+ const lines = buffer.split("\n");
196
+ buffer = lines.pop() ?? "";
197
+ for (const line of lines) {
198
+ if (!line.trim())
199
+ continue;
200
+ try {
201
+ const chunk = JSON.parse(line);
202
+ if (chunk.message?.content) {
203
+ text += chunk.message.content;
204
+ onChunk({ type: "text_delta", text: chunk.message.content });
205
+ }
206
+ if (chunk.message?.tool_calls) {
207
+ for (const tc of chunk.message.tool_calls) {
208
+ toolCalls.push({
209
+ id: tc.id ?? `tool_${Date.now()}`,
210
+ name: tc.function?.name ?? "",
211
+ args: typeof tc.function?.arguments === "string"
212
+ ? JSON.parse(tc.function.arguments)
213
+ : (tc.function?.arguments ?? {}),
214
+ });
215
+ }
216
+ }
217
+ if (chunk.eval_count)
218
+ outputTokens = chunk.eval_count;
219
+ if (chunk.prompt_eval_count)
220
+ inputTokens = chunk.prompt_eval_count;
221
+ }
222
+ catch {
223
+ // Skip malformed JSON lines
224
+ }
225
+ }
226
+ }
227
+ onChunk({ type: "done" });
228
+ return { text, toolCalls, inputTokens, outputTokens };
229
+ }
230
+ // ── OpenAI-Compatible: POST /v1/chat/completions with SSE ──
231
+ async streamOpenAiCompatible(messages, tools, systemPrompt, maxTokens, onChunk) {
232
+ const openaiMessages = [
233
+ { role: "system", content: systemPrompt },
234
+ ...messages.map((m) => {
235
+ if (m.tool_calls) {
236
+ return { role: m.role, content: m.content, tool_calls: m.tool_calls };
237
+ }
238
+ if (m.tool_call_id) {
239
+ return {
240
+ role: m.role,
241
+ content: m.content,
242
+ tool_call_id: m.tool_call_id,
243
+ };
244
+ }
245
+ return { role: m.role, content: m.content };
246
+ }),
247
+ ];
248
+ const body = {
249
+ model: this.modelId,
250
+ messages: openaiMessages,
251
+ max_tokens: maxTokens,
252
+ stream: true,
253
+ };
254
+ if (tools.length > 0) {
255
+ body.tools = tools;
256
+ }
257
+ const headers = {
258
+ "Content-Type": "application/json",
259
+ };
260
+ if (this.apiKey) {
261
+ headers.Authorization = `Bearer ${this.apiKey}`;
262
+ }
263
+ const response = await fetch(`${this.baseUrl}/v1/chat/completions`, {
264
+ method: "POST",
265
+ headers,
266
+ body: JSON.stringify(body),
267
+ });
268
+ if (!response.ok) {
269
+ const errText = await response.text().catch(() => "");
270
+ throw new Error(`[LocalChatProvider] Chat request failed: ${response.status} ${response.statusText}${errText ? ` — ${errText.slice(0, 200)}` : ""}`);
271
+ }
272
+ return this.parseSseStream(response, onChunk);
273
+ }
274
+ async parseSseStream(response, onChunk) {
275
+ const reader = response.body?.getReader();
276
+ if (!reader)
277
+ throw new Error("[LocalChatProvider] No response body");
278
+ const decoder = new TextDecoder();
279
+ let text = "";
280
+ const toolCalls = [];
281
+ const pendingToolArgs = new Map();
282
+ let buffer = "";
283
+ while (true) {
284
+ const { done, value } = await reader.read();
285
+ if (done)
286
+ break;
287
+ buffer += decoder.decode(value, { stream: true });
288
+ const lines = buffer.split("\n");
289
+ buffer = lines.pop() ?? "";
290
+ for (const line of lines) {
291
+ if (!line.startsWith("data: "))
292
+ continue;
293
+ const data = line.slice(6).trim();
294
+ if (data === "[DONE]")
295
+ continue;
296
+ try {
297
+ const chunk = JSON.parse(data);
298
+ const delta = chunk.choices?.[0]?.delta;
299
+ if (!delta)
300
+ continue;
301
+ if (delta.content) {
302
+ text += delta.content;
303
+ onChunk({ type: "text_delta", text: delta.content });
304
+ }
305
+ if (delta.tool_calls) {
306
+ for (const tc of delta.tool_calls) {
307
+ const idx = tc.index ?? 0;
308
+ if (tc.id) {
309
+ pendingToolArgs.set(idx, {
310
+ id: tc.id,
311
+ name: tc.function?.name ?? "",
312
+ args: tc.function?.arguments ?? "",
313
+ });
314
+ }
315
+ else if (pendingToolArgs.has(idx) && tc.function?.arguments) {
316
+ const pending = pendingToolArgs.get(idx);
317
+ if (pending)
318
+ pending.args += tc.function.arguments;
319
+ }
320
+ }
321
+ }
322
+ }
323
+ catch {
324
+ // Skip malformed SSE data
325
+ }
326
+ }
327
+ }
328
+ // Finalize pending tool calls
329
+ for (const [, pending] of pendingToolArgs) {
330
+ try {
331
+ toolCalls.push({
332
+ id: pending.id,
333
+ name: pending.name,
334
+ args: pending.args ? JSON.parse(pending.args) : {},
335
+ });
336
+ }
337
+ catch {
338
+ toolCalls.push({
339
+ id: pending.id,
340
+ name: pending.name,
341
+ args: {},
342
+ });
343
+ }
344
+ }
345
+ onChunk({ type: "done" });
346
+ // OpenAI-compatible endpoints don't always return token counts in streamed responses
347
+ return { text, toolCalls, inputTokens: 0, outputTokens: 0 };
348
+ }
349
+ }
350
+ // ── AI SDK Chat Provider ─────────────────────────────────────
351
+ /**
352
+ * Chat provider backed by Vercel AI SDK streamText().
353
+ *
354
+ * Works with any AI SDK-compatible LanguageModel (Anthropic, OpenAI, Google,
355
+ * Ollama, any OpenAI-compatible). This is the recommended provider for new
356
+ * integrations — it replaces both AnthropicChatProvider and LocalChatProvider.
357
+ */
358
+ export class AiSdkChatProvider {
359
+ providerName;
360
+ modelId;
361
+ model;
362
+ constructor(model, providerName, modelId) {
363
+ this.model = model;
364
+ this.providerName = providerName;
365
+ this.modelId = modelId;
366
+ }
367
+ async streamChat(messages, tools, systemPrompt, maxTokens, onChunk) {
368
+ const { streamText } = await import("ai");
369
+ const modelMessages = this.toModelMessages(messages);
370
+ const aiTools = this.toAiTools(tools);
371
+ const result = streamText({
372
+ model: this.model,
373
+ system: systemPrompt,
374
+ messages: modelMessages,
375
+ tools: Object.keys(aiTools).length > 0 ? aiTools : undefined,
376
+ maxOutputTokens: maxTokens,
377
+ toolChoice: tools.length > 0 ? "auto" : undefined,
378
+ });
379
+ let text = "";
380
+ const collectedToolCalls = [];
381
+ for await (const part of result.fullStream) {
382
+ if (part.type === "text-delta") {
383
+ const delta = part.text ??
384
+ part.textDelta ??
385
+ "";
386
+ text += delta;
387
+ onChunk({ type: "text_delta", text: delta });
388
+ }
389
+ else if (part.type === "tool-call") {
390
+ const tc = part;
391
+ const toolCallId = tc.toolCallId ?? "";
392
+ const toolName = tc.toolName ?? "";
393
+ const args = tc.args ?? tc.input ?? {};
394
+ onChunk({ type: "tool_use_start", toolCallId, toolName });
395
+ onChunk({
396
+ type: "tool_use_delta",
397
+ toolCallId,
398
+ toolArgs: JSON.stringify(args),
399
+ });
400
+ onChunk({ type: "tool_use_end", toolCallId });
401
+ collectedToolCalls.push({
402
+ id: toolCallId,
403
+ name: toolName,
404
+ args: args,
405
+ });
406
+ }
407
+ }
408
+ let inputTokens = 0;
409
+ let outputTokens = 0;
410
+ try {
411
+ const usage = await result.usage;
412
+ const u = usage;
413
+ inputTokens = u?.promptTokens ?? 0;
414
+ outputTokens = u?.completionTokens ?? 0;
415
+ }
416
+ catch {
417
+ /* token counts unavailable for some providers */
418
+ }
419
+ onChunk({ type: "done" });
420
+ return { text, toolCalls: collectedToolCalls, inputTokens, outputTokens };
421
+ }
422
+ toModelMessages(messages) {
423
+ const result = [];
424
+ for (const msg of messages) {
425
+ if (msg.role === "system")
426
+ continue;
427
+ if (msg.role === "user") {
428
+ result.push({ role: "user", content: msg.content });
429
+ }
430
+ else if (msg.role === "tool") {
431
+ result.push({
432
+ role: "tool",
433
+ content: [
434
+ {
435
+ type: "tool-result",
436
+ toolCallId: msg.tool_call_id ?? "",
437
+ result: msg.content,
438
+ },
439
+ ],
440
+ });
441
+ }
442
+ else if (msg.role === "assistant") {
443
+ if (msg.tool_calls?.length) {
444
+ const parts = [];
445
+ if (msg.content) {
446
+ parts.push({ type: "text", text: msg.content });
447
+ }
448
+ for (const tc of msg.tool_calls) {
449
+ parts.push({
450
+ type: "tool-call",
451
+ toolCallId: tc.id,
452
+ toolName: tc.function.name,
453
+ args: JSON.parse(tc.function.arguments),
454
+ });
455
+ }
456
+ result.push({ role: "assistant", content: parts });
457
+ }
458
+ else {
459
+ result.push({ role: "assistant", content: msg.content });
460
+ }
461
+ }
462
+ }
463
+ return result;
464
+ }
465
+ toAiTools(tools) {
466
+ const result = {};
467
+ for (const t of tools) {
468
+ result[t.function.name] = {
469
+ description: t.function.description,
470
+ parameters: t.function.parameters,
471
+ };
472
+ }
473
+ return result;
474
+ }
475
+ }
@@ -0,0 +1,55 @@
1
+ /**
2
+ * Provider Factory — creates AI SDK provider from settings configuration.
3
+ *
4
+ * Reads the LLM configuration (provider, model, apiKey, baseUrl) and
5
+ * returns a ready-to-use LanguageModel for streamText().
6
+ *
7
+ * Supports two config shapes:
8
+ * 1. Direct ProviderConfig: { provider, model, apiKey, baseUrl }
9
+ * 2. Settings-based: reads from loadSettings().llm
10
+ */
11
+ import { createLanguageModel, } from "./providers.js";
12
+ const DEFAULT_MODELS = {
13
+ anthropic: "claude-sonnet-4-20250514",
14
+ openai: "gpt-4o",
15
+ google: "gemini-2.0-flash",
16
+ ollama: "llama3",
17
+ "openai-compatible": "default",
18
+ };
19
+ /**
20
+ * Resolve a ProviderConfig from loose settings.
21
+ * Fills in defaults for missing fields.
22
+ */
23
+ export function resolveProviderConfig(settings) {
24
+ const provider = (settings.provider ?? "anthropic");
25
+ const model = settings.model ?? DEFAULT_MODELS[provider] ?? "default";
26
+ return {
27
+ provider,
28
+ model,
29
+ apiKey: settings.apiKey,
30
+ baseUrl: settings.baseUrl,
31
+ };
32
+ }
33
+ /**
34
+ * Create a LanguageModel from settings configuration.
35
+ * This is the primary entry point for the rest of the codebase.
36
+ */
37
+ export async function createModelFromSettings(settings) {
38
+ const config = resolveProviderConfig(settings);
39
+ return createLanguageModel(config);
40
+ }
41
+ /**
42
+ * Create a LanguageModel from environment variables.
43
+ * Falls back to Anthropic with claude-sonnet-4-20250514.
44
+ */
45
+ export async function createModelFromEnv() {
46
+ const provider = (process.env.UNERR_LLM_PROVIDER ??
47
+ "anthropic");
48
+ const model = process.env.UNERR_MODEL ?? DEFAULT_MODELS[provider] ?? "default";
49
+ return createLanguageModel({
50
+ provider,
51
+ model,
52
+ apiKey: process.env.ANTHROPIC_API_KEY ?? process.env.OPENAI_API_KEY,
53
+ baseUrl: process.env.UNERR_LLM_BASE_URL,
54
+ });
55
+ }
@@ -0,0 +1,90 @@
1
+ /**
2
+ * AI Provider Registry — unified provider abstraction via Vercel AI SDK.
3
+ *
4
+ * Supports: Anthropic, OpenAI, Google Gemini, Ollama (local), any OpenAI-compatible.
5
+ * Each provider returns an AI SDK LanguageModel that plugs into streamText().
6
+ *
7
+ * Design: the registry is a pure function map — no state, no singletons.
8
+ * Provider creation is lazy (only when called) to avoid importing unused SDKs.
9
+ */
10
+ /**
11
+ * Create an AI SDK LanguageModel from provider configuration.
12
+ * Throws on invalid config or missing API keys (except Ollama).
13
+ */
14
+ export async function createLanguageModel(config) {
15
+ switch (config.provider) {
16
+ case "anthropic":
17
+ return createAnthropicModel(config);
18
+ case "openai":
19
+ return createOpenAIModel(config);
20
+ case "google":
21
+ return createGoogleModel(config);
22
+ case "ollama":
23
+ return createOllamaModel(config);
24
+ case "openai-compatible":
25
+ return createOpenAICompatibleModel(config);
26
+ default: {
27
+ const _exhaustive = config.provider;
28
+ throw new Error(`Unknown provider: ${config.provider}`);
29
+ }
30
+ }
31
+ }
32
+ async function createAnthropicModel(config) {
33
+ const { createAnthropic } = await import("@ai-sdk/anthropic");
34
+ const anthropic = createAnthropic({
35
+ apiKey: config.apiKey ?? process.env.ANTHROPIC_API_KEY,
36
+ });
37
+ return anthropic(config.model);
38
+ }
39
+ async function createOpenAIModel(config) {
40
+ const { createOpenAI } = await import("@ai-sdk/openai");
41
+ const openai = createOpenAI({
42
+ apiKey: config.apiKey ?? process.env.OPENAI_API_KEY,
43
+ ...(config.baseUrl ? { baseURL: config.baseUrl } : {}),
44
+ });
45
+ return openai(config.model);
46
+ }
47
+ async function createGoogleModel(config) {
48
+ const { createGoogleGenerativeAI } = await import("@ai-sdk/google");
49
+ const google = createGoogleGenerativeAI({
50
+ apiKey: config.apiKey ?? process.env.GOOGLE_GENERATIVE_AI_API_KEY,
51
+ });
52
+ return google(config.model);
53
+ }
54
+ /**
55
+ * Ollama uses the OpenAI-compatible endpoint with a local baseURL.
56
+ * No API key required — Ollama runs locally.
57
+ */
58
+ async function createOllamaModel(config) {
59
+ const { createOpenAI } = await import("@ai-sdk/openai");
60
+ const ollama = createOpenAI({
61
+ baseURL: config.baseUrl ?? "http://localhost:11434/v1",
62
+ apiKey: "ollama",
63
+ });
64
+ return ollama(config.model);
65
+ }
66
+ /**
67
+ * Any OpenAI-compatible API (LM Studio, vLLM, Together, Fireworks, etc.)
68
+ * Requires baseUrl to be set.
69
+ */
70
+ async function createOpenAICompatibleModel(config) {
71
+ if (!config.baseUrl) {
72
+ throw new Error("openai-compatible provider requires baseUrl to be set");
73
+ }
74
+ const { createOpenAI } = await import("@ai-sdk/openai");
75
+ const compatible = createOpenAI({
76
+ baseURL: config.baseUrl,
77
+ apiKey: config.apiKey ?? "no-key",
78
+ });
79
+ return compatible(config.model);
80
+ }
81
+ /**
82
+ * List of all supported provider names.
83
+ */
84
+ export const SUPPORTED_PROVIDERS = [
85
+ "anthropic",
86
+ "openai",
87
+ "google",
88
+ "ollama",
89
+ "openai-compatible",
90
+ ];