opencandle 0.4.0 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (564) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +186 -117
  3. package/dist/analysts/contracts.d.ts +1 -3
  4. package/dist/analysts/contracts.js +1 -11
  5. package/dist/analysts/contracts.js.map +1 -1
  6. package/dist/analysts/orchestrator.d.ts +1 -3
  7. package/dist/analysts/orchestrator.js +1 -26
  8. package/dist/analysts/orchestrator.js.map +1 -1
  9. package/dist/cli.js +32 -8
  10. package/dist/cli.js.map +1 -1
  11. package/dist/config.d.ts +19 -3
  12. package/dist/config.js +69 -3
  13. package/dist/config.js.map +1 -1
  14. package/dist/index.d.ts +1 -1
  15. package/dist/index.js +1 -1
  16. package/dist/index.js.map +1 -1
  17. package/dist/infra/browser.d.ts +1 -3
  18. package/dist/infra/browser.js +4 -2
  19. package/dist/infra/browser.js.map +1 -1
  20. package/dist/infra/cache.d.ts +8 -11
  21. package/dist/infra/cache.js +17 -15
  22. package/dist/infra/cache.js.map +1 -1
  23. package/dist/infra/http-client.d.ts +4 -1
  24. package/dist/infra/http-client.js +59 -6
  25. package/dist/infra/http-client.js.map +1 -1
  26. package/dist/infra/index.d.ts +3 -3
  27. package/dist/infra/index.js +3 -3
  28. package/dist/infra/index.js.map +1 -1
  29. package/dist/infra/native-dependencies.js +2 -2
  30. package/dist/infra/native-dependencies.js.map +1 -1
  31. package/dist/infra/node-version.js.map +1 -1
  32. package/dist/infra/opencandle-paths.d.ts +0 -3
  33. package/dist/infra/opencandle-paths.js +4 -11
  34. package/dist/infra/opencandle-paths.js.map +1 -1
  35. package/dist/infra/rate-limiter.d.ts +4 -0
  36. package/dist/infra/rate-limiter.js +17 -10
  37. package/dist/infra/rate-limiter.js.map +1 -1
  38. package/dist/market-state/alert-conditions.d.ts +34 -0
  39. package/dist/market-state/alert-conditions.js +23 -0
  40. package/dist/market-state/alert-conditions.js.map +1 -0
  41. package/dist/market-state/alert-runner.d.ts +55 -0
  42. package/dist/market-state/alert-runner.js +634 -0
  43. package/dist/market-state/alert-runner.js.map +1 -0
  44. package/dist/market-state/daily-report.d.ts +26 -0
  45. package/dist/market-state/daily-report.js +179 -0
  46. package/dist/market-state/daily-report.js.map +1 -0
  47. package/dist/market-state/local-automation-service.d.ts +25 -0
  48. package/dist/market-state/local-automation-service.js +119 -0
  49. package/dist/market-state/local-automation-service.js.map +1 -0
  50. package/dist/market-state/notification-delivery.d.ts +14 -0
  51. package/dist/market-state/notification-delivery.js +139 -0
  52. package/dist/market-state/notification-delivery.js.map +1 -0
  53. package/dist/market-state/resolve-for-mutation.d.ts +10 -0
  54. package/dist/market-state/resolve-for-mutation.js +15 -0
  55. package/dist/market-state/resolve-for-mutation.js.map +1 -0
  56. package/dist/market-state/resolve.d.ts +14 -0
  57. package/dist/market-state/resolve.js +89 -0
  58. package/dist/market-state/resolve.js.map +1 -0
  59. package/dist/market-state/service.d.ts +527 -0
  60. package/dist/market-state/service.js +1099 -0
  61. package/dist/market-state/service.js.map +1 -0
  62. package/dist/memory/index.d.ts +7 -7
  63. package/dist/memory/index.js +6 -6
  64. package/dist/memory/index.js.map +1 -1
  65. package/dist/memory/manager.d.ts +9 -0
  66. package/dist/memory/manager.js +39 -22
  67. package/dist/memory/manager.js.map +1 -1
  68. package/dist/memory/retrieval.js +7 -4
  69. package/dist/memory/retrieval.js.map +1 -1
  70. package/dist/memory/sqlite.js +385 -3
  71. package/dist/memory/sqlite.js.map +1 -1
  72. package/dist/memory/storage.d.ts +3 -2
  73. package/dist/memory/storage.js +1 -2
  74. package/dist/memory/storage.js.map +1 -1
  75. package/dist/memory/tool-defaults.js +64 -28
  76. package/dist/memory/tool-defaults.js.map +1 -1
  77. package/dist/memory/types.js +4 -0
  78. package/dist/memory/types.js.map +1 -1
  79. package/dist/monitor.d.ts +2 -0
  80. package/dist/monitor.js +104 -0
  81. package/dist/monitor.js.map +1 -0
  82. package/dist/onboarding/connect.js +4 -6
  83. package/dist/onboarding/connect.js.map +1 -1
  84. package/dist/onboarding/credential-interceptor.js +1 -1
  85. package/dist/onboarding/credential-interceptor.js.map +1 -1
  86. package/dist/onboarding/degradation-accumulator.js +1 -3
  87. package/dist/onboarding/degradation-accumulator.js.map +1 -1
  88. package/dist/onboarding/providers.js +3 -16
  89. package/dist/onboarding/providers.js.map +1 -1
  90. package/dist/onboarding/state.js.map +1 -1
  91. package/dist/onboarding/tool-helpers.js +1 -1
  92. package/dist/onboarding/tool-helpers.js.map +1 -1
  93. package/dist/onboarding/tool-tags.js +6 -4
  94. package/dist/onboarding/tool-tags.js.map +1 -1
  95. package/dist/onboarding/validation.js +1 -1
  96. package/dist/onboarding/validation.js.map +1 -1
  97. package/dist/pi/opencandle-extension.d.ts +8 -0
  98. package/dist/pi/opencandle-extension.js +637 -59
  99. package/dist/pi/opencandle-extension.js.map +1 -1
  100. package/dist/pi/session.d.ts +1 -1
  101. package/dist/pi/session.js +3 -1
  102. package/dist/pi/session.js.map +1 -1
  103. package/dist/pi/setup.js +17 -2
  104. package/dist/pi/setup.js.map +1 -1
  105. package/dist/pi/tool-adapter.js +5 -2
  106. package/dist/pi/tool-adapter.js.map +1 -1
  107. package/dist/prompts/context-builder.d.ts +18 -3
  108. package/dist/prompts/context-builder.js +117 -18
  109. package/dist/prompts/context-builder.js.map +1 -1
  110. package/dist/prompts/disclaimer.js +1 -1
  111. package/dist/prompts/disclaimer.js.map +1 -1
  112. package/dist/prompts/policy-cards.d.ts +13 -0
  113. package/dist/prompts/policy-cards.js +197 -0
  114. package/dist/prompts/policy-cards.js.map +1 -0
  115. package/dist/prompts/sections.d.ts +1 -1
  116. package/dist/prompts/sections.js +3 -3
  117. package/dist/prompts/sections.js.map +1 -1
  118. package/dist/prompts/symbol-preflight.d.ts +20 -0
  119. package/dist/prompts/symbol-preflight.js +49 -0
  120. package/dist/prompts/symbol-preflight.js.map +1 -0
  121. package/dist/prompts/workflow-prompts.d.ts +1 -1
  122. package/dist/prompts/workflow-prompts.js +209 -19
  123. package/dist/prompts/workflow-prompts.js.map +1 -1
  124. package/dist/providers/alpha-vantage.d.ts +1 -1
  125. package/dist/providers/alpha-vantage.js +49 -8
  126. package/dist/providers/alpha-vantage.js.map +1 -1
  127. package/dist/providers/coingecko.js +1 -1
  128. package/dist/providers/coingecko.js.map +1 -1
  129. package/dist/providers/errors.d.ts +5 -0
  130. package/dist/providers/errors.js +11 -0
  131. package/dist/providers/errors.js.map +1 -0
  132. package/dist/providers/exa-search.d.ts +2 -2
  133. package/dist/providers/exa-search.js +19 -11
  134. package/dist/providers/exa-search.js.map +1 -1
  135. package/dist/providers/fear-greed.js +1 -1
  136. package/dist/providers/fear-greed.js.map +1 -1
  137. package/dist/providers/finnhub.js +3 -5
  138. package/dist/providers/finnhub.js.map +1 -1
  139. package/dist/providers/fred.js +2 -2
  140. package/dist/providers/fred.js.map +1 -1
  141. package/dist/providers/index.d.ts +7 -6
  142. package/dist/providers/index.js +6 -5
  143. package/dist/providers/index.js.map +1 -1
  144. package/dist/providers/reddit.js +2 -2
  145. package/dist/providers/reddit.js.map +1 -1
  146. package/dist/providers/sec-edgar.d.ts +9 -1
  147. package/dist/providers/sec-edgar.js +181 -6
  148. package/dist/providers/sec-edgar.js.map +1 -1
  149. package/dist/providers/tradingview.d.ts +47 -0
  150. package/dist/providers/tradingview.js +275 -0
  151. package/dist/providers/tradingview.js.map +1 -0
  152. package/dist/providers/twitter.js +6 -8
  153. package/dist/providers/twitter.js.map +1 -1
  154. package/dist/providers/web-search.js +26 -12
  155. package/dist/providers/web-search.js.map +1 -1
  156. package/dist/providers/with-fallback.js +4 -2
  157. package/dist/providers/with-fallback.js.map +1 -1
  158. package/dist/providers/wrap-provider.d.ts +2 -3
  159. package/dist/providers/wrap-provider.js +14 -8
  160. package/dist/providers/wrap-provider.js.map +1 -1
  161. package/dist/providers/yahoo-finance.d.ts +3 -1
  162. package/dist/providers/yahoo-finance.js +226 -11
  163. package/dist/providers/yahoo-finance.js.map +1 -1
  164. package/dist/routing/classify-intent.d.ts +9 -0
  165. package/dist/routing/classify-intent.js +153 -3
  166. package/dist/routing/classify-intent.js.map +1 -1
  167. package/dist/routing/defaults.d.ts +1 -1
  168. package/dist/routing/defaults.js +3 -3
  169. package/dist/routing/defaults.js.map +1 -1
  170. package/dist/routing/entity-extractor.d.ts +2 -0
  171. package/dist/routing/entity-extractor.js +377 -26
  172. package/dist/routing/entity-extractor.js.map +1 -1
  173. package/dist/routing/fund-symbols.d.ts +2 -0
  174. package/dist/routing/fund-symbols.js +55 -0
  175. package/dist/routing/fund-symbols.js.map +1 -0
  176. package/dist/routing/horizon.d.ts +1 -0
  177. package/dist/routing/horizon.js +10 -0
  178. package/dist/routing/horizon.js.map +1 -0
  179. package/dist/routing/index.d.ts +12 -6
  180. package/dist/routing/index.js +8 -4
  181. package/dist/routing/index.js.map +1 -1
  182. package/dist/routing/legacy-rule-router.d.ts +9 -0
  183. package/dist/routing/legacy-rule-router.js +12 -0
  184. package/dist/routing/legacy-rule-router.js.map +1 -0
  185. package/dist/routing/planning.d.ts +54 -0
  186. package/dist/routing/planning.js +562 -0
  187. package/dist/routing/planning.js.map +1 -0
  188. package/dist/routing/route-manifest.d.ts +35 -0
  189. package/dist/routing/route-manifest.js +242 -0
  190. package/dist/routing/route-manifest.js.map +1 -0
  191. package/dist/routing/router-llm-client.js.map +1 -1
  192. package/dist/routing/router-prompt.js +46 -45
  193. package/dist/routing/router-prompt.js.map +1 -1
  194. package/dist/routing/router-types.d.ts +10 -0
  195. package/dist/routing/router.d.ts +1 -0
  196. package/dist/routing/router.js +572 -13
  197. package/dist/routing/router.js.map +1 -1
  198. package/dist/routing/slot-resolver.d.ts +1 -1
  199. package/dist/routing/slot-resolver.js +45 -7
  200. package/dist/routing/slot-resolver.js.map +1 -1
  201. package/dist/routing/symbol-disambiguator.d.ts +11 -0
  202. package/dist/routing/symbol-disambiguator.js +52 -0
  203. package/dist/routing/symbol-disambiguator.js.map +1 -0
  204. package/dist/routing/turn-context.d.ts +44 -0
  205. package/dist/routing/turn-context.js +45 -0
  206. package/dist/routing/turn-context.js.map +1 -0
  207. package/dist/routing/types.d.ts +15 -1
  208. package/dist/runtime/answer-contracts.d.ts +82 -0
  209. package/dist/runtime/answer-contracts.js +442 -0
  210. package/dist/runtime/answer-contracts.js.map +1 -0
  211. package/dist/runtime/artifact-contracts.d.ts +14 -0
  212. package/dist/runtime/artifact-contracts.js +57 -0
  213. package/dist/runtime/artifact-contracts.js.map +1 -0
  214. package/dist/runtime/planning-evidence.d.ts +99 -0
  215. package/dist/runtime/planning-evidence.js +466 -0
  216. package/dist/runtime/planning-evidence.js.map +1 -0
  217. package/dist/runtime/prompt-step.d.ts +1 -9
  218. package/dist/runtime/prompt-step.js +0 -10
  219. package/dist/runtime/prompt-step.js.map +1 -1
  220. package/dist/runtime/run-context.d.ts +5 -2
  221. package/dist/runtime/run-context.js +8 -1
  222. package/dist/runtime/run-context.js.map +1 -1
  223. package/dist/runtime/session-coordinator.d.ts +29 -3
  224. package/dist/runtime/session-coordinator.js +204 -31
  225. package/dist/runtime/session-coordinator.js.map +1 -1
  226. package/dist/runtime/session-title.d.ts +14 -0
  227. package/dist/runtime/session-title.js +50 -0
  228. package/dist/runtime/session-title.js.map +1 -0
  229. package/dist/runtime/tool-defaults-wrapper.js +1 -3
  230. package/dist/runtime/tool-defaults-wrapper.js.map +1 -1
  231. package/dist/runtime/validation.js.map +1 -1
  232. package/dist/runtime/workflow-events.js.map +1 -1
  233. package/dist/runtime/workflow-runner.d.ts +3 -3
  234. package/dist/runtime/workflow-runner.js +1 -1
  235. package/dist/runtime/workflow-runner.js.map +1 -1
  236. package/dist/sentiment/adapters/finnhub.d.ts +1 -1
  237. package/dist/sentiment/adapters/finnhub.js +6 -1
  238. package/dist/sentiment/adapters/finnhub.js.map +1 -1
  239. package/dist/sentiment/adapters/reddit.d.ts +2 -2
  240. package/dist/sentiment/adapters/twitter.d.ts +1 -1
  241. package/dist/sentiment/adapters/web.d.ts +1 -1
  242. package/dist/sentiment/index.d.ts +9 -11
  243. package/dist/sentiment/index.js +9 -20
  244. package/dist/sentiment/index.js.map +1 -1
  245. package/dist/sentiment/keywords.js +26 -4
  246. package/dist/sentiment/keywords.js.map +1 -1
  247. package/dist/sentiment/pipeline.d.ts +2 -2
  248. package/dist/sentiment/pipeline.js +1 -1
  249. package/dist/sentiment/pipeline.js.map +1 -1
  250. package/dist/sentiment/scorer.js +1 -1
  251. package/dist/sentiment/store.d.ts +1 -1
  252. package/dist/sentiment/store.js +1 -1
  253. package/dist/sentiment/store.js.map +1 -1
  254. package/dist/sentiment/trends.d.ts +1 -1
  255. package/dist/sentiment/trends.js.map +1 -1
  256. package/dist/sentiment/types.js.map +1 -1
  257. package/dist/system-prompt.js +7 -3
  258. package/dist/system-prompt.js.map +1 -1
  259. package/dist/tool-kit.d.ts +7 -7
  260. package/dist/tool-kit.js +4 -4
  261. package/dist/tool-kit.js.map +1 -1
  262. package/dist/tools/fundamentals/company-overview.js +12 -7
  263. package/dist/tools/fundamentals/company-overview.js.map +1 -1
  264. package/dist/tools/fundamentals/comps.js +19 -10
  265. package/dist/tools/fundamentals/comps.js.map +1 -1
  266. package/dist/tools/fundamentals/dcf.js +24 -12
  267. package/dist/tools/fundamentals/dcf.js.map +1 -1
  268. package/dist/tools/fundamentals/earnings.js +9 -4
  269. package/dist/tools/fundamentals/earnings.js.map +1 -1
  270. package/dist/tools/fundamentals/financials.js +9 -4
  271. package/dist/tools/fundamentals/financials.js.map +1 -1
  272. package/dist/tools/fundamentals/sec-filings.d.ts +1 -0
  273. package/dist/tools/fundamentals/sec-filings.js +36 -4
  274. package/dist/tools/fundamentals/sec-filings.js.map +1 -1
  275. package/dist/tools/index.d.ts +23 -18
  276. package/dist/tools/index.js +53 -38
  277. package/dist/tools/index.js.map +1 -1
  278. package/dist/tools/interaction/ask-user.js +15 -3
  279. package/dist/tools/interaction/ask-user.js.map +1 -1
  280. package/dist/tools/interaction/twitter-login.js +13 -3
  281. package/dist/tools/interaction/twitter-login.js.map +1 -1
  282. package/dist/tools/macro/fear-greed.js +1 -1
  283. package/dist/tools/macro/fear-greed.js.map +1 -1
  284. package/dist/tools/macro/fred-data.d.ts +1 -1
  285. package/dist/tools/macro/fred-data.js +44 -9
  286. package/dist/tools/macro/fred-data.js.map +1 -1
  287. package/dist/tools/market/crypto-history.js +21 -3
  288. package/dist/tools/market/crypto-history.js.map +1 -1
  289. package/dist/tools/market/crypto-price.js +4 -2
  290. package/dist/tools/market/crypto-price.js.map +1 -1
  291. package/dist/tools/market/screen-stocks.d.ts +18 -0
  292. package/dist/tools/market/screen-stocks.js +252 -0
  293. package/dist/tools/market/screen-stocks.js.map +1 -0
  294. package/dist/tools/market/search-ticker.js +161 -9
  295. package/dist/tools/market/search-ticker.js.map +1 -1
  296. package/dist/tools/market/stock-history.d.ts +2 -2
  297. package/dist/tools/market/stock-history.js +27 -8
  298. package/dist/tools/market/stock-history.js.map +1 -1
  299. package/dist/tools/market/stock-quote.js +6 -4
  300. package/dist/tools/market/stock-quote.js.map +1 -1
  301. package/dist/tools/options/greeks.js +1 -2
  302. package/dist/tools/options/greeks.js.map +1 -1
  303. package/dist/tools/options/option-chain.js +27 -9
  304. package/dist/tools/options/option-chain.js.map +1 -1
  305. package/dist/tools/portfolio/alerts.d.ts +15 -0
  306. package/dist/tools/portfolio/alerts.js +357 -0
  307. package/dist/tools/portfolio/alerts.js.map +1 -0
  308. package/dist/tools/portfolio/correlation.d.ts +1 -1
  309. package/dist/tools/portfolio/correlation.js +34 -14
  310. package/dist/tools/portfolio/correlation.js.map +1 -1
  311. package/dist/tools/portfolio/daily-report.d.ts +8 -0
  312. package/dist/tools/portfolio/daily-report.js +83 -0
  313. package/dist/tools/portfolio/daily-report.js.map +1 -0
  314. package/dist/tools/portfolio/holdings-overlap.d.ts +8 -0
  315. package/dist/tools/portfolio/holdings-overlap.js +112 -0
  316. package/dist/tools/portfolio/holdings-overlap.js.map +1 -0
  317. package/dist/tools/portfolio/notifications.d.ts +7 -0
  318. package/dist/tools/portfolio/notifications.js +43 -0
  319. package/dist/tools/portfolio/notifications.js.map +1 -0
  320. package/dist/tools/portfolio/predictions.d.ts +12 -6
  321. package/dist/tools/portfolio/predictions.js +338 -88
  322. package/dist/tools/portfolio/predictions.js.map +1 -1
  323. package/dist/tools/portfolio/risk-analysis.d.ts +1 -1
  324. package/dist/tools/portfolio/risk-analysis.js +46 -7
  325. package/dist/tools/portfolio/risk-analysis.js.map +1 -1
  326. package/dist/tools/portfolio/tracker.d.ts +4 -3
  327. package/dist/tools/portfolio/tracker.js +247 -102
  328. package/dist/tools/portfolio/tracker.js.map +1 -1
  329. package/dist/tools/portfolio/watchlist.d.ts +6 -4
  330. package/dist/tools/portfolio/watchlist.js +209 -101
  331. package/dist/tools/portfolio/watchlist.js.map +1 -1
  332. package/dist/tools/sentiment/reddit-sentiment.js +24 -11
  333. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  334. package/dist/tools/sentiment/sentiment-summary.js +71 -14
  335. package/dist/tools/sentiment/sentiment-summary.js.map +1 -1
  336. package/dist/tools/sentiment/sentiment-trend.d.ts +1 -1
  337. package/dist/tools/sentiment/sentiment-trend.js +12 -2
  338. package/dist/tools/sentiment/sentiment-trend.js.map +1 -1
  339. package/dist/tools/sentiment/twitter-sentiment.js +13 -6
  340. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
  341. package/dist/tools/sentiment/untrusted-text.d.ts +2 -0
  342. package/dist/tools/sentiment/untrusted-text.js +17 -0
  343. package/dist/tools/sentiment/untrusted-text.js.map +1 -0
  344. package/dist/tools/sentiment/web-search.js +37 -12
  345. package/dist/tools/sentiment/web-search.js.map +1 -1
  346. package/dist/tools/sentiment/web-sentiment.js +16 -4
  347. package/dist/tools/sentiment/web-sentiment.js.map +1 -1
  348. package/dist/tools/technical/backtest.d.ts +3 -3
  349. package/dist/tools/technical/backtest.js +65 -44
  350. package/dist/tools/technical/backtest.js.map +1 -1
  351. package/dist/tools/technical/indicators.js +24 -8
  352. package/dist/tools/technical/indicators.js.map +1 -1
  353. package/dist/types/index.d.ts +3 -3
  354. package/dist/types/index.js.map +1 -1
  355. package/dist/types/market.d.ts +1 -0
  356. package/dist/types/options.d.ts +10 -0
  357. package/dist/types/portfolio.d.ts +41 -4
  358. package/dist/workflows/compare-assets.d.ts +0 -3
  359. package/dist/workflows/compare-assets.js +55 -10
  360. package/dist/workflows/compare-assets.js.map +1 -1
  361. package/dist/workflows/index.d.ts +3 -4
  362. package/dist/workflows/index.js +3 -3
  363. package/dist/workflows/index.js.map +1 -1
  364. package/dist/workflows/options-screener.d.ts +0 -3
  365. package/dist/workflows/options-screener.js +88 -14
  366. package/dist/workflows/options-screener.js.map +1 -1
  367. package/dist/workflows/portfolio-builder.d.ts +0 -3
  368. package/dist/workflows/portfolio-builder.js +7 -11
  369. package/dist/workflows/portfolio-builder.js.map +1 -1
  370. package/gui/server/ask-user-bridge.ts +82 -0
  371. package/gui/server/automation-heartbeat.ts +97 -0
  372. package/gui/server/background-quotes.ts +97 -1
  373. package/gui/server/chat-event-adapter.ts +32 -10
  374. package/gui/server/chat-run-session.ts +16 -0
  375. package/gui/server/gui-session-manager.ts +5 -0
  376. package/gui/server/invoke-tool.ts +144 -1
  377. package/gui/server/live-chat-event-adapter.ts +21 -6
  378. package/gui/server/market-state-api.ts +315 -0
  379. package/gui/server/model-setup.ts +149 -2
  380. package/gui/server/private-api-access.ts +62 -0
  381. package/gui/server/projector.ts +58 -11
  382. package/gui/server/prompt-observation.ts +58 -0
  383. package/gui/server/quote-snapshot-store.ts +50 -0
  384. package/gui/server/server.ts +236 -376
  385. package/gui/server/session-actions.ts +186 -1
  386. package/gui/server/session-entry-wait.ts +81 -0
  387. package/gui/server/shutdown.ts +47 -0
  388. package/gui/server/tool-invoke-ack.ts +49 -0
  389. package/gui/server/tool-metadata.ts +23 -10
  390. package/gui/server/websocket.ts +13 -3
  391. package/gui/server/writer-lock.ts +6 -2
  392. package/gui/server/ws-hub.ts +292 -0
  393. package/gui/shared/chat-events.ts +16 -1
  394. package/gui/shared/event-reducer.ts +24 -6
  395. package/gui/web/dist/assets/CatalogOverlay-eJ2cBk33.js +1 -0
  396. package/gui/web/dist/assets/index-2KZtKBmu.css +1 -0
  397. package/gui/web/dist/assets/index-CveNgtDg.js +69 -0
  398. package/gui/web/dist/index.html +2 -2
  399. package/package.json +22 -12
  400. package/src/analysts/contracts.ts +10 -23
  401. package/src/analysts/orchestrator.ts +8 -43
  402. package/src/cli.ts +37 -13
  403. package/src/config.ts +99 -7
  404. package/src/index.ts +1 -1
  405. package/src/infra/browser.ts +4 -2
  406. package/src/infra/cache.ts +41 -30
  407. package/src/infra/http-client.ts +72 -6
  408. package/src/infra/index.ts +7 -10
  409. package/src/infra/native-dependencies.ts +8 -3
  410. package/src/infra/node-version.ts +3 -1
  411. package/src/infra/opencandle-paths.ts +3 -14
  412. package/src/infra/rate-limiter.ts +32 -20
  413. package/src/market-state/alert-conditions.ts +82 -0
  414. package/src/market-state/alert-runner.ts +863 -0
  415. package/src/market-state/daily-report.ts +247 -0
  416. package/src/market-state/local-automation-service.ts +162 -0
  417. package/src/market-state/notification-delivery.ts +158 -0
  418. package/src/market-state/resolve-for-mutation.ts +24 -0
  419. package/src/market-state/resolve.ts +112 -0
  420. package/src/market-state/service.ts +2344 -0
  421. package/src/memory/index.ts +7 -7
  422. package/src/memory/manager.ts +57 -26
  423. package/src/memory/retrieval.ts +8 -7
  424. package/src/memory/sqlite.ts +407 -6
  425. package/src/memory/storage.ts +8 -17
  426. package/src/memory/tool-defaults.ts +60 -39
  427. package/src/memory/types.ts +7 -3
  428. package/src/monitor.ts +121 -0
  429. package/src/onboarding/connect.ts +10 -33
  430. package/src/onboarding/credential-interceptor.ts +3 -15
  431. package/src/onboarding/degradation-accumulator.ts +1 -3
  432. package/src/onboarding/providers.ts +9 -40
  433. package/src/onboarding/state.ts +4 -15
  434. package/src/onboarding/tool-helpers.ts +2 -9
  435. package/src/onboarding/tool-tags.ts +6 -6
  436. package/src/onboarding/validation.ts +14 -20
  437. package/src/pi/opencandle-extension.ts +795 -120
  438. package/src/pi/session.ts +7 -5
  439. package/src/pi/setup.ts +61 -33
  440. package/src/pi/tool-adapter.ts +5 -2
  441. package/src/prompts/context-builder.ts +143 -21
  442. package/src/prompts/disclaimer.ts +1 -1
  443. package/src/prompts/policy-cards.ts +220 -0
  444. package/src/prompts/sections.ts +4 -4
  445. package/src/prompts/symbol-preflight.ts +80 -0
  446. package/src/prompts/workflow-prompts.ts +231 -28
  447. package/src/providers/alpha-vantage.ts +82 -40
  448. package/src/providers/coingecko.ts +2 -5
  449. package/src/providers/errors.ts +9 -0
  450. package/src/providers/exa-search.ts +24 -22
  451. package/src/providers/fear-greed.ts +1 -1
  452. package/src/providers/finnhub.ts +7 -6
  453. package/src/providers/fred.ts +3 -3
  454. package/src/providers/index.ts +14 -6
  455. package/src/providers/reddit.ts +17 -6
  456. package/src/providers/sec-edgar.ts +235 -5
  457. package/src/providers/tradingview.ts +399 -0
  458. package/src/providers/twitter.ts +6 -8
  459. package/src/providers/web-search.ts +30 -20
  460. package/src/providers/with-fallback.ts +8 -7
  461. package/src/providers/wrap-provider.ts +15 -10
  462. package/src/providers/yahoo-finance.ts +292 -20
  463. package/src/routing/classify-intent.ts +186 -4
  464. package/src/routing/defaults.ts +4 -4
  465. package/src/routing/entity-extractor.ts +428 -28
  466. package/src/routing/fund-symbols.ts +58 -0
  467. package/src/routing/horizon.ts +7 -0
  468. package/src/routing/index.ts +60 -16
  469. package/src/routing/legacy-rule-router.ts +13 -0
  470. package/src/routing/planning.ts +823 -0
  471. package/src/routing/route-manifest.ts +309 -0
  472. package/src/routing/router-llm-client.ts +4 -4
  473. package/src/routing/router-prompt.ts +52 -52
  474. package/src/routing/router-types.ts +18 -0
  475. package/src/routing/router.ts +717 -20
  476. package/src/routing/slot-resolver.ts +75 -14
  477. package/src/routing/symbol-disambiguator.ts +72 -0
  478. package/src/routing/turn-context.ts +108 -0
  479. package/src/routing/types.ts +15 -1
  480. package/src/runtime/answer-contracts.ts +672 -0
  481. package/src/runtime/artifact-contracts.ts +77 -0
  482. package/src/runtime/planning-evidence.ts +682 -0
  483. package/src/runtime/prompt-step.ts +1 -16
  484. package/src/runtime/run-context.ts +12 -2
  485. package/src/runtime/session-coordinator.ts +297 -56
  486. package/src/runtime/session-title.ts +60 -0
  487. package/src/runtime/tool-defaults-wrapper.ts +1 -3
  488. package/src/runtime/validation.ts +1 -4
  489. package/src/runtime/workflow-events.ts +7 -7
  490. package/src/runtime/workflow-runner.ts +5 -11
  491. package/src/sentiment/adapters/finnhub.ts +7 -2
  492. package/src/sentiment/adapters/reddit.ts +2 -2
  493. package/src/sentiment/adapters/twitter.ts +1 -1
  494. package/src/sentiment/adapters/web.ts +1 -1
  495. package/src/sentiment/index.ts +16 -26
  496. package/src/sentiment/keywords.ts +26 -4
  497. package/src/sentiment/pipeline.ts +15 -4
  498. package/src/sentiment/scorer.ts +1 -1
  499. package/src/sentiment/store.ts +2 -2
  500. package/src/sentiment/trends.ts +9 -3
  501. package/src/sentiment/types.ts +5 -4
  502. package/src/system-prompt.ts +7 -3
  503. package/src/tool-kit.ts +10 -9
  504. package/src/tools/fundamentals/company-overview.ts +20 -10
  505. package/src/tools/fundamentals/comps.ts +69 -56
  506. package/src/tools/fundamentals/dcf.ts +146 -96
  507. package/src/tools/fundamentals/earnings.ts +17 -7
  508. package/src/tools/fundamentals/financials.ts +17 -8
  509. package/src/tools/fundamentals/sec-filings.ts +52 -8
  510. package/src/tools/index.ts +53 -38
  511. package/src/tools/interaction/ask-user.ts +22 -10
  512. package/src/tools/interaction/twitter-login.ts +17 -5
  513. package/src/tools/macro/fear-greed.ts +2 -2
  514. package/src/tools/macro/fred-data.ts +80 -42
  515. package/src/tools/market/crypto-history.ts +25 -4
  516. package/src/tools/market/crypto-price.ts +7 -7
  517. package/src/tools/market/screen-stocks.ts +279 -0
  518. package/src/tools/market/search-ticker.ts +219 -18
  519. package/src/tools/market/stock-history.ts +38 -13
  520. package/src/tools/market/stock-quote.ts +11 -8
  521. package/src/tools/options/greeks.ts +5 -6
  522. package/src/tools/options/option-chain.ts +47 -18
  523. package/src/tools/portfolio/alerts.ts +457 -0
  524. package/src/tools/portfolio/correlation.ts +48 -21
  525. package/src/tools/portfolio/daily-report.ts +101 -0
  526. package/src/tools/portfolio/holdings-overlap.ts +139 -0
  527. package/src/tools/portfolio/notifications.ts +45 -0
  528. package/src/tools/portfolio/predictions.ts +407 -107
  529. package/src/tools/portfolio/risk-analysis.ts +47 -8
  530. package/src/tools/portfolio/tracker.ts +271 -110
  531. package/src/tools/portfolio/watchlist.ts +251 -116
  532. package/src/tools/sentiment/reddit-sentiment.ts +51 -25
  533. package/src/tools/sentiment/sentiment-summary.ts +116 -35
  534. package/src/tools/sentiment/sentiment-trend.ts +24 -7
  535. package/src/tools/sentiment/twitter-sentiment.ts +23 -16
  536. package/src/tools/sentiment/untrusted-text.ts +21 -0
  537. package/src/tools/sentiment/web-search.ts +52 -16
  538. package/src/tools/sentiment/web-sentiment.ts +27 -11
  539. package/src/tools/technical/backtest.ts +78 -47
  540. package/src/tools/technical/indicators.ts +40 -17
  541. package/src/types/index.ts +8 -3
  542. package/src/types/market.ts +1 -0
  543. package/src/types/options.ts +17 -0
  544. package/src/types/portfolio.ts +46 -4
  545. package/src/types/sentiment.ts +2 -2
  546. package/src/workflows/compare-assets.ts +67 -19
  547. package/src/workflows/index.ts +3 -4
  548. package/src/workflows/options-screener.ts +98 -22
  549. package/src/workflows/portfolio-builder.ts +40 -29
  550. package/dist/runtime/index.d.ts +0 -16
  551. package/dist/runtime/index.js +0 -10
  552. package/dist/runtime/index.js.map +0 -1
  553. package/dist/runtime/provider-ids.d.ts +0 -14
  554. package/dist/runtime/provider-ids.js +0 -14
  555. package/dist/runtime/provider-ids.js.map +0 -1
  556. package/dist/workflows/types.d.ts +0 -4
  557. package/dist/workflows/types.js +0 -2
  558. package/dist/workflows/types.js.map +0 -1
  559. package/gui/web/dist/assets/CatalogOverlay-D1ImSJTe.js +0 -1
  560. package/gui/web/dist/assets/index-DBrWq43L.css +0 -1
  561. package/gui/web/dist/assets/index-RflHaj0y.js +0 -67
  562. package/src/runtime/index.ts +0 -55
  563. package/src/runtime/provider-ids.ts +0 -15
  564. package/src/workflows/types.ts +0 -4
@@ -1,13 +1,31 @@
1
- import { extractEntities } from "./entity-extractor.js";
1
+ import {
2
+ extractEntities,
3
+ isAmbiguousConceptUsage,
4
+ isCurrencyCodeUsage,
5
+ } from "./entity-extractor.js";
6
+ import { classifyWithLegacyRules } from "./legacy-rule-router.js";
7
+ import {
8
+ computeMissingRequiredSlots,
9
+ isDispatchableWorkflow,
10
+ isRouteKind,
11
+ isToolBundleName,
12
+ legacyRouteForRouteKind,
13
+ routeKindFromLegacyRoute,
14
+ selectToolBundles,
15
+ } from "./route-manifest.js";
2
16
  import { buildRouterPrompt } from "./router-prompt.js";
3
17
  import type {
18
+ RouterDiagnostic,
4
19
  RouterInputContext,
5
20
  RouterLlmClient,
6
21
  RouterOutput,
7
22
  RouterPreferenceUpdate,
8
23
  RouterRoute,
24
+ RouterRouteKind,
9
25
  RouterSlot,
26
+ ToolBundleName,
10
27
  } from "./router-types.js";
28
+ import { disambiguateSymbols } from "./symbol-disambiguator.js";
11
29
  import type { ExtractedEntities, WorkflowType } from "./types.js";
12
30
 
13
31
  const VALID_ROUTES: readonly RouterRoute[] = ["workflow", "fallback"];
@@ -19,7 +37,7 @@ const VALID_WORKFLOWS: ReadonlyArray<Exclude<WorkflowType, "unclassified">> = [
19
37
  "watchlist_or_tracking",
20
38
  "general_finance_qa",
21
39
  ];
22
- const VALID_SOURCES = new Set(["user", "preference", "default"]);
40
+ const VALID_SOURCES = new Set(["user", "preference", "default", "prior_context", "memory"]);
23
41
  const VALID_CONFIDENCE = new Set(["high", "medium", "low"]);
24
42
 
25
43
  /**
@@ -38,7 +56,7 @@ export async function route(
38
56
  let firstError: string | undefined;
39
57
  try {
40
58
  const raw = await client.complete(prompt);
41
- return validateRouterOutput(raw);
59
+ return postProcessRouterOutput(input.text, validateRouterOutput(raw));
42
60
  } catch (err) {
43
61
  firstError = err instanceof Error ? err.message : String(err);
44
62
  }
@@ -47,10 +65,10 @@ export async function route(
47
65
  try {
48
66
  const retryPrompt = `${prompt}\n\n(Your previous response failed validation: ${firstError}. Return a valid JSON object conforming to RouterOutput. Nothing else.)`;
49
67
  const raw = await client.complete(retryPrompt);
50
- return validateRouterOutput(raw);
68
+ return postProcessRouterOutput(input.text, validateRouterOutput(raw));
51
69
  } catch {
52
70
  // Persistent failure — return a minimal fallback with regex-extracted symbols.
53
- return minimalFallback(input.text);
71
+ return postProcessRouterOutput(input.text, minimalFallback(input.text));
54
72
  }
55
73
  }
56
74
 
@@ -62,33 +80,70 @@ export function validateRouterOutput(raw: string): RouterOutput {
62
80
  }
63
81
  const obj = parsed as Record<string, unknown>;
64
82
 
65
- const route = obj.route;
66
- if (typeof route !== "string" || !VALID_ROUTES.includes(route as RouterRoute)) {
67
- throw new Error(`invalid route: ${JSON.stringify(route)}`);
83
+ const rawMissingRequired = validateStringArray(obj.missing_required, "missing_required");
84
+
85
+ const explicitRouteKind = obj.routeKind;
86
+ if (
87
+ explicitRouteKind !== undefined &&
88
+ (typeof explicitRouteKind !== "string" || !isRouteKind(explicitRouteKind))
89
+ ) {
90
+ throw new Error(`invalid routeKind: ${JSON.stringify(explicitRouteKind)}`);
91
+ }
92
+
93
+ const rawRoute = obj.route;
94
+ let route: RouterRoute;
95
+ if (typeof rawRoute === "string") {
96
+ if (!VALID_ROUTES.includes(rawRoute as RouterRoute)) {
97
+ throw new Error(`invalid route: ${JSON.stringify(rawRoute)}`);
98
+ }
99
+ route = rawRoute as RouterRoute;
100
+ } else if (typeof explicitRouteKind === "string" && isRouteKind(explicitRouteKind)) {
101
+ route = legacyRouteForRouteKind(explicitRouteKind);
102
+ } else {
103
+ throw new Error(`invalid route: ${JSON.stringify(rawRoute)}`);
68
104
  }
69
105
 
70
106
  let workflow: RouterOutput["workflow"];
71
- if (route === "workflow") {
72
- if (typeof obj.workflow !== "string" || !VALID_WORKFLOWS.includes(obj.workflow as Exclude<WorkflowType, "unclassified">)) {
73
- throw new Error(`workflow route requires a valid workflow; got ${JSON.stringify(obj.workflow)}`);
107
+ const routeKind: RouterRouteKind =
108
+ typeof explicitRouteKind === "string" && isRouteKind(explicitRouteKind)
109
+ ? explicitRouteKind
110
+ : routeKindFromLegacyRoute(route, rawMissingRequired);
111
+
112
+ if (route === "workflow" || routeKind === "workflow_dispatch") {
113
+ if (
114
+ typeof obj.workflow !== "string" ||
115
+ !VALID_WORKFLOWS.includes(obj.workflow as Exclude<WorkflowType, "unclassified">)
116
+ ) {
117
+ throw new Error(
118
+ `workflow route requires a valid workflow; got ${JSON.stringify(obj.workflow)}`,
119
+ );
74
120
  }
75
121
  workflow = obj.workflow as Exclude<WorkflowType, "unclassified">;
122
+ } else if (
123
+ typeof obj.workflow === "string" &&
124
+ VALID_WORKFLOWS.includes(obj.workflow as Exclude<WorkflowType, "unclassified">)
125
+ ) {
126
+ workflow = obj.workflow as Exclude<WorkflowType, "unclassified">;
76
127
  }
77
128
 
78
129
  const entities = validateEntities(obj.entities);
79
130
  const slots = validateSlots(obj.slots);
80
131
  const preference_updates = validatePreferenceUpdates(obj.preference_updates);
81
- const missing_required = validateStringArray(obj.missing_required, "missing_required");
82
- const reasoning =
83
- typeof obj.reasoning === "string" ? obj.reasoning : "";
132
+ const missing_required = rawMissingRequired;
133
+ const tool_bundles = validateToolBundles(obj.tool_bundles);
134
+ const diagnostics = validateDiagnostics(obj.diagnostics);
135
+ const reasoning = typeof obj.reasoning === "string" ? obj.reasoning : "";
84
136
 
85
137
  return {
86
- route: route as RouterRoute,
138
+ routeKind,
139
+ route: legacyRouteForRouteKind(routeKind),
87
140
  workflow,
88
141
  entities,
89
142
  slots,
90
143
  preference_updates,
91
144
  missing_required,
145
+ tool_bundles,
146
+ diagnostics,
92
147
  reasoning,
93
148
  };
94
149
  }
@@ -113,20 +168,586 @@ function validateEntities(raw: unknown): ExtractedEntities {
113
168
  throw new Error("entities must be an object");
114
169
  }
115
170
  const e = raw as Record<string, unknown>;
116
- const symbols = validateStringArray(e.symbols, "entities.symbols").map((s) =>
117
- s.toUpperCase(),
118
- );
171
+ const symbols = validateStringArray(e.symbols, "entities.symbols").map((s) => s.toUpperCase());
119
172
 
120
173
  const out: ExtractedEntities = { symbols };
121
174
  if (typeof e.budget === "number") out.budget = e.budget;
122
175
  if (typeof e.maxPremium === "number") out.maxPremium = e.maxPremium;
176
+ if (typeof e.costBasis === "number") out.costBasis = e.costBasis;
177
+ if (typeof e.shareQuantity === "number") out.shareQuantity = e.shareQuantity;
123
178
  if (typeof e.timeHorizon === "string") out.timeHorizon = e.timeHorizon;
124
179
  if (typeof e.riskProfile === "string") out.riskProfile = e.riskProfile;
125
180
  if (e.direction === "bullish" || e.direction === "bearish") out.direction = e.direction;
126
181
  if (typeof e.dteHint === "string") out.dteHint = e.dteHint;
182
+ if (e.optionStrategy === "covered_call" || e.optionStrategy === "protective_put")
183
+ out.optionStrategy = e.optionStrategy;
184
+ if (typeof e.heldSymbol === "string") out.heldSymbol = e.heldSymbol.toUpperCase();
185
+ const catalystSymbols = validateStringArray(e.catalystSymbols, "entities.catalystSymbols").map(
186
+ (s) => s.toUpperCase(),
187
+ );
188
+ if (catalystSymbols.length > 0) out.catalystSymbols = catalystSymbols;
189
+ const compareMetrics = validateStringArray(e.compareMetrics, "entities.compareMetrics");
190
+ if (compareMetrics.length > 0) out.compareMetrics = compareMetrics;
127
191
  return out;
128
192
  }
129
193
 
194
+ export function postProcessRouterOutput(text: string, output: RouterOutput): RouterOutput {
195
+ const extracted = extractEntities(text);
196
+ const deterministic = classifyWithLegacyRules(text);
197
+ let diagnostics: RouterDiagnostic[] = [...output.diagnostics];
198
+ let next: RouterOutput = {
199
+ ...output,
200
+ entities: {
201
+ ...output.entities,
202
+ symbols: output.entities.symbols.filter((symbol) => !isAmbiguousConceptUsage(text, symbol)),
203
+ budget: output.entities.budget ?? extracted.budget,
204
+ maxPremium: output.entities.maxPremium ?? extracted.maxPremium,
205
+ timeHorizon: output.entities.timeHorizon ?? extracted.timeHorizon,
206
+ riskProfile: output.entities.riskProfile ?? extracted.riskProfile,
207
+ assetScope: output.entities.assetScope ?? extracted.assetScope,
208
+ compareMetrics: mergeStringArrays(output.entities.compareMetrics, extracted.compareMetrics),
209
+ direction: output.entities.direction ?? extracted.direction,
210
+ optionStrategy: output.entities.optionStrategy ?? extracted.optionStrategy,
211
+ costBasis: output.entities.costBasis ?? extracted.costBasis,
212
+ shareQuantity: output.entities.shareQuantity ?? extracted.shareQuantity,
213
+ heldSymbol: output.entities.heldSymbol ?? extracted.heldSymbol,
214
+ catalystSymbols: output.entities.catalystSymbols ?? extracted.catalystSymbols,
215
+ dteHint:
216
+ output.entities.dteHint ??
217
+ (output.workflow === "options_screener" ? extracted.dteHint : undefined),
218
+ },
219
+ diagnostics,
220
+ };
221
+
222
+ if (
223
+ next.workflow === "options_screener" &&
224
+ isExistingPositionOptionRequest(text, extracted) &&
225
+ extracted.heldSymbol
226
+ ) {
227
+ const reorderedSymbols = [
228
+ extracted.heldSymbol,
229
+ ...mergeSymbols(next.entities.symbols, extracted.symbols).filter(
230
+ (symbol) => symbol !== extracted.heldSymbol,
231
+ ),
232
+ ];
233
+ if (next.entities.symbols[0] !== extracted.heldSymbol) {
234
+ diagnostics.push({
235
+ code:
236
+ extracted.optionStrategy === "protective_put"
237
+ ? "existing_position_underlying_corrected"
238
+ : "covered_call_underlying_corrected",
239
+ message: `using owned position ${extracted.heldSymbol} as the option-chain underlying`,
240
+ });
241
+ }
242
+ next = {
243
+ ...next,
244
+ entities: {
245
+ ...next.entities,
246
+ symbols: reorderedSymbols,
247
+ optionStrategy: extracted.optionStrategy ?? next.entities.optionStrategy,
248
+ direction: extracted.direction ?? next.entities.direction,
249
+ heldSymbol: extracted.heldSymbol,
250
+ catalystSymbols: reorderedSymbols.filter((symbol) => symbol !== extracted.heldSymbol),
251
+ costBasis: extracted.costBasis ?? next.entities.costBasis,
252
+ shareQuantity: extracted.shareQuantity ?? next.entities.shareQuantity,
253
+ dteHint: extracted.dteHint ?? next.entities.dteHint,
254
+ },
255
+ diagnostics,
256
+ };
257
+ }
258
+
259
+ if (
260
+ next.workflow === "options_screener" &&
261
+ isOptionsEducationOrSuitabilityRequest(text) &&
262
+ !isSpecificOptionContractSelectionRequest(text)
263
+ ) {
264
+ diagnostics.push({
265
+ code: "options_workflow_corrected_to_policy_task",
266
+ message:
267
+ "options education or suitability prompt should use policy-card synthesis, not contract-screen workflow dispatch",
268
+ });
269
+ next = {
270
+ ...next,
271
+ routeKind: "agent_task",
272
+ route: "fallback",
273
+ workflow: "general_finance_qa",
274
+ missing_required: [],
275
+ diagnostics,
276
+ };
277
+ }
278
+
279
+ // Legacy rules may recover a primary route only when the LLM router path has
280
+ // already failed validation. Otherwise they are limited to enrichment and
281
+ // narrow corrections below.
282
+ if (
283
+ next.diagnostics.some((d) => d.code === "router_validation_failed") &&
284
+ deterministic.workflow !== "unclassified"
285
+ ) {
286
+ next = {
287
+ ...next,
288
+ routeKind: isDispatchableWorkflow(deterministic.workflow)
289
+ ? "workflow_dispatch"
290
+ : "agent_task",
291
+ route: isDispatchableWorkflow(deterministic.workflow) ? "workflow" : "fallback",
292
+ workflow: deterministic.workflow,
293
+ entities: {
294
+ ...deterministic.entities,
295
+ budget: deterministic.entities.budget ?? extracted.budget,
296
+ maxPremium: deterministic.entities.maxPremium ?? extracted.maxPremium,
297
+ timeHorizon: deterministic.entities.timeHorizon ?? extracted.timeHorizon,
298
+ riskProfile: deterministic.entities.riskProfile ?? extracted.riskProfile,
299
+ assetScope: deterministic.entities.assetScope ?? extracted.assetScope,
300
+ compareMetrics: mergeStringArrays(
301
+ deterministic.entities.compareMetrics,
302
+ extracted.compareMetrics,
303
+ ),
304
+ direction: deterministic.entities.direction ?? extracted.direction,
305
+ costBasis: deterministic.entities.costBasis ?? extracted.costBasis,
306
+ shareQuantity: deterministic.entities.shareQuantity ?? extracted.shareQuantity,
307
+ heldSymbol: deterministic.entities.heldSymbol ?? extracted.heldSymbol,
308
+ catalystSymbols: deterministic.entities.catalystSymbols ?? extracted.catalystSymbols,
309
+ },
310
+ diagnostics: [
311
+ ...diagnostics,
312
+ {
313
+ code: "deterministic_failure_recovery",
314
+ message: `deterministic classifier selected ${deterministic.workflow} after router validation failure`,
315
+ },
316
+ ],
317
+ reasoning: next.reasoning
318
+ ? `${next.reasoning}; deterministic classifier selected ${deterministic.workflow}`
319
+ : `deterministic classifier selected ${deterministic.workflow}`,
320
+ };
321
+ diagnostics = next.diagnostics;
322
+ }
323
+
324
+ if (next.routeKind === "workflow_dispatch" && !isDispatchableWorkflow(next.workflow)) {
325
+ diagnostics.push({
326
+ code: "route_kind_corrected_to_agent_task",
327
+ message: next.workflow
328
+ ? `${next.workflow} is not a dispatchable workflow`
329
+ : "workflow_dispatch requires a dispatchable workflow",
330
+ });
331
+ next = {
332
+ ...next,
333
+ routeKind: "agent_task",
334
+ route: "fallback",
335
+ diagnostics,
336
+ };
337
+ }
338
+
339
+ if (
340
+ (next.workflow === "compare_assets" || next.workflow === "portfolio_builder") &&
341
+ isStatefulTrackingRequest(text)
342
+ ) {
343
+ diagnostics.push({
344
+ code: "stateful_tracking_corrected_to_agent_task",
345
+ message:
346
+ "portfolio/watchlist tracking mutation should use stateful tracking tools, not compare or construction workflow dispatch",
347
+ });
348
+ next = {
349
+ ...next,
350
+ routeKind: "agent_task",
351
+ route: "fallback",
352
+ workflow: "watchlist_or_tracking",
353
+ missing_required: [],
354
+ entities: {
355
+ ...next.entities,
356
+ symbols: filterCurrencyUnitSymbols(text, next.entities.symbols),
357
+ },
358
+ slots: removeCurrencyUnitSymbolSlots(text, next.slots),
359
+ diagnostics,
360
+ };
361
+ }
362
+
363
+ if (next.routeKind === "agent_task" && isDispatchableWorkflow(next.workflow)) {
364
+ diagnostics.push({
365
+ code: "dispatchable_workflow_corrected_to_workflow_dispatch",
366
+ message: `${next.workflow} is a dispatchable workflow`,
367
+ });
368
+ next = {
369
+ ...next,
370
+ routeKind: "workflow_dispatch",
371
+ route: "workflow",
372
+ diagnostics,
373
+ };
374
+ }
375
+
376
+ if (
377
+ next.workflow === "compare_assets" &&
378
+ next.entities.symbols.length === 0 &&
379
+ isExplicitMacroDataRequest(text)
380
+ ) {
381
+ diagnostics.push({
382
+ code: "compare_route_corrected_to_macro_task",
383
+ message: "macro/source acronyms were not explicit tickers",
384
+ });
385
+ next = {
386
+ ...next,
387
+ routeKind: "agent_task",
388
+ route: "fallback",
389
+ workflow: "general_finance_qa",
390
+ missing_required: [],
391
+ diagnostics,
392
+ };
393
+ }
394
+
395
+ if (next.workflow === "compare_assets" && isPortfolioEvaluationRequest(text)) {
396
+ diagnostics.push({
397
+ code: "portfolio_evaluation_corrected_to_agent_task",
398
+ message:
399
+ "existing portfolio/allocation risk review should not be reduced to asset comparison",
400
+ });
401
+ next = {
402
+ ...next,
403
+ routeKind: "agent_task",
404
+ route: "fallback",
405
+ workflow: "general_finance_qa",
406
+ missing_required: [],
407
+ diagnostics,
408
+ };
409
+ }
410
+
411
+ if (
412
+ next.routeKind === "agent_task" &&
413
+ !next.workflow &&
414
+ next.entities.symbols.length === 0 &&
415
+ isExplicitMacroDataRequest(text)
416
+ ) {
417
+ diagnostics.push({
418
+ code: "macro_task_inferred_from_prompt",
419
+ message: "macro data terms were present without explicit tickers",
420
+ });
421
+ next = {
422
+ ...next,
423
+ workflow: "general_finance_qa",
424
+ diagnostics,
425
+ };
426
+ }
427
+
428
+ if (next.workflow === "portfolio_builder" && isCryptoSizingRequest(text)) {
429
+ diagnostics.push({
430
+ code: "crypto_sizing_corrected_to_agent_task",
431
+ message:
432
+ "crypto allocation-range and drawdown questions are advisory tradeoffs, not portfolio construction",
433
+ });
434
+ next = {
435
+ ...next,
436
+ routeKind: "agent_task",
437
+ route: "fallback",
438
+ workflow: "general_finance_qa",
439
+ missing_required: [],
440
+ diagnostics,
441
+ };
442
+ }
443
+
444
+ if (next.workflow === "portfolio_builder" && isPortfolioEvaluationRequest(text)) {
445
+ diagnostics.push({
446
+ code: "portfolio_evaluation_corrected_to_agent_task",
447
+ message:
448
+ "existing portfolio/allocation evaluation does not require portfolio-construction budget",
449
+ });
450
+ next = {
451
+ ...next,
452
+ routeKind: "agent_task",
453
+ route: "fallback",
454
+ workflow: "general_finance_qa",
455
+ missing_required: [],
456
+ diagnostics,
457
+ };
458
+ }
459
+
460
+ if (
461
+ next.workflow === "portfolio_builder" &&
462
+ next.entities.symbols.length >= 2 &&
463
+ isPortfolioTradeoffComparisonRequest(text)
464
+ ) {
465
+ diagnostics.push({
466
+ code: "portfolio_tradeoff_corrected_to_compare_assets",
467
+ message:
468
+ "explicit multi-asset tradeoff question should compare the requested assets before constructing a portfolio",
469
+ });
470
+ next = {
471
+ ...next,
472
+ routeKind: "workflow_dispatch",
473
+ route: "workflow",
474
+ workflow: "compare_assets",
475
+ missing_required: [],
476
+ diagnostics,
477
+ };
478
+ }
479
+
480
+ if (next.workflow === "single_asset_analysis" && isSpecializedSingleAssetPolicyRequest(text)) {
481
+ diagnostics.push({
482
+ code: "single_asset_workflow_corrected_to_general_policy_task",
483
+ message: "prompt asks for policy-card planning outside a single-asset buy/sell analysis",
484
+ });
485
+ next = {
486
+ ...next,
487
+ workflow: "general_finance_qa",
488
+ diagnostics,
489
+ };
490
+ }
491
+
492
+ const disambiguated = disambiguateSymbols(next.entities.symbols, text);
493
+ if (disambiguated.dropped.length > 0) {
494
+ for (const drop of disambiguated.dropped) {
495
+ diagnostics.push({
496
+ code: "symbol_dropped",
497
+ message: `${drop.token} dropped: ${drop.reason}`,
498
+ details: {
499
+ token: drop.token,
500
+ reason: drop.reason,
501
+ signalsChecked: drop.signalsChecked,
502
+ source: "llm",
503
+ },
504
+ });
505
+ }
506
+ next = {
507
+ ...next,
508
+ entities: {
509
+ ...next.entities,
510
+ symbols: disambiguated.kept,
511
+ },
512
+ slots: removeDroppedSymbolSlots(
513
+ next.slots,
514
+ disambiguated.dropped.map((drop) => drop.token),
515
+ ),
516
+ diagnostics,
517
+ };
518
+ }
519
+
520
+ const missingRequired = computeMissingRequiredSlots(
521
+ next.workflow,
522
+ next.entities,
523
+ next.slots,
524
+ next.missing_required,
525
+ );
526
+ if (missingRequired.length > 0 && next.routeKind !== "pass_through") {
527
+ if (next.routeKind !== "clarification") {
528
+ diagnostics.push({
529
+ code: "route_kind_corrected_to_clarification",
530
+ message: `missing required slots: ${missingRequired.join(", ")}`,
531
+ });
532
+ }
533
+ next = {
534
+ ...next,
535
+ routeKind: "clarification",
536
+ route: "fallback",
537
+ missing_required: missingRequired,
538
+ diagnostics,
539
+ };
540
+ }
541
+
542
+ const selectedToolBundles = isConceptualEducationRequest(text, next)
543
+ ? []
544
+ : selectToolBundles(next);
545
+ if (selectedToolBundles.length === 0 && isConceptualEducationRequest(text, next)) {
546
+ diagnostics.push({
547
+ code: "conceptual_education_no_tools",
548
+ message: "conceptual education prompt does not need live finance tools",
549
+ });
550
+ }
551
+ const emittedUnsupported = next.tool_bundles.filter(
552
+ (bundle) => !selectedToolBundles.includes(bundle),
553
+ );
554
+ if (emittedUnsupported.length > 0) {
555
+ diagnostics.push({
556
+ code: "tool_bundles_corrected",
557
+ message: `unsupported emitted bundles dropped: ${emittedUnsupported.join(", ")}`,
558
+ });
559
+ }
560
+
561
+ return omitUndefined({
562
+ ...next,
563
+ route: legacyRouteForRouteKind(next.routeKind),
564
+ tool_bundles: selectedToolBundles,
565
+ diagnostics,
566
+ });
567
+ }
568
+
569
+ function isExplicitMacroDataRequest(text: string): boolean {
570
+ return /\b(?:get_economic_data|fred|cpi|inflation|fed\s+funds?|unemployment|gdp|macro)\b/i.test(
571
+ text,
572
+ );
573
+ }
574
+
575
+ function isConceptualEducationRequest(text: string, output: RouterOutput): boolean {
576
+ if (output.routeKind !== "agent_task") return false;
577
+ if (output.entities.symbols.length > 0) return false;
578
+ if (isForwardLookingMacroContextRequest(text)) return false;
579
+ if (
580
+ /\b(?:current|recent|today|right now|latest|news|sentiment|build|portfolio|buy|sell|allocate|compare)\b/i.test(
581
+ text,
582
+ )
583
+ ) {
584
+ return false;
585
+ }
586
+ return /\b(?:explain|what is|define|how (?:do|should|to)|teach me|help me understand)\b/i.test(
587
+ text,
588
+ );
589
+ }
590
+
591
+ function isForwardLookingMacroContextRequest(text: string): boolean {
592
+ return (
593
+ /\b(?:rates?|rate\s*cuts?|fed|inflation|macro)\b/i.test(text) &&
594
+ /\b(?:next\s+(?:year|12\s*months?)|over\s+the\s+next|outlook|affect|impact|falling|rising)\b/i.test(
595
+ text,
596
+ )
597
+ );
598
+ }
599
+
600
+ function isCoveredCallRequest(text: string): boolean {
601
+ return /\bcovered\s+calls?\b/i.test(text);
602
+ }
603
+
604
+ function isPortfolioEvaluationRequest(text: string): boolean {
605
+ const lower = text.toLowerCase();
606
+ const hasEvaluationIntent =
607
+ /\b(?:evaluat(?:e|ion)|review|assess|analy[sz]e|prospects?|risks?|risky|opportunities?|mitigat(?:e|ion)|adjustment|rebalance|diversify|concentration|overweight|underweight|target\s+bands?|drift|worried|crash|protect|protection|missing\s+out\s+on\s+growth)\b/.test(
608
+ lower,
609
+ );
610
+ const hasPortfolioObject =
611
+ /\b(?:portfolio|allocation|asset\s+allocation|60\/40|equity|fixed\s+income|bonds?)\b/.test(
612
+ lower,
613
+ );
614
+ const hasConstructionIntent =
615
+ /\b(?:build|create|construct|put\s+together|invest|allocate)\b/.test(lower) &&
616
+ /\$\s*\d|\b\d+(?:\.\d+)?\s*k\b|\bbudget\b|\bcapital\b/.test(lower);
617
+ return hasEvaluationIntent && hasPortfolioObject && !hasConstructionIntent;
618
+ }
619
+
620
+ function isStatefulTrackingRequest(text: string): boolean {
621
+ const lower = text.toLowerCase();
622
+ const hasPortfolioConstructionIntent =
623
+ /\b(?:build|create|construct|put\s+together)\b/.test(lower) &&
624
+ /\bportfolio\b/.test(lower) &&
625
+ /\$\s*\d|\b\d+(?:\.\d+)?\s*k\b|\bbudget\b|\bcapital\b/.test(lower);
626
+ const hasStateVerb =
627
+ /\b(?:add|remove|update|record|track|create|configure|check|show|list|view|cancel)\b/.test(
628
+ lower,
629
+ );
630
+ const hasStateObject =
631
+ /\b(?:watchlist|portfolio|holding|holdings|position|positions|prediction|predictions|alert|alerts|daily\s+report|watchlist\s+report|report\s+history)\b/.test(
632
+ lower,
633
+ );
634
+ const hasPortfolioLotShape =
635
+ /\b(?:add|record|track)\b/.test(lower) &&
636
+ /\b\d+(?:\.\d+)?\s+shares?\b/.test(lower) &&
637
+ /\b(?:portfolio|holding|holdings|position|positions)\b/.test(lower);
638
+ if (hasPortfolioConstructionIntent) return false;
639
+ return (hasStateVerb && hasStateObject) || hasPortfolioLotShape;
640
+ }
641
+
642
+ function filterCurrencyUnitSymbols(text: string, symbols: string[]): string[] {
643
+ return symbols.filter((symbol) => !isCurrencyCodeUsage(text, symbol));
644
+ }
645
+
646
+ function removeCurrencyUnitSymbolSlots(
647
+ text: string,
648
+ slots: Record<string, RouterSlot>,
649
+ ): Record<string, RouterSlot> {
650
+ const next = { ...slots };
651
+ for (const key of ["symbol", "symbols"]) {
652
+ const slot = next[key];
653
+ if (!slot) continue;
654
+ if (Array.isArray(slot.value)) {
655
+ const value = slot.value.filter(
656
+ (item) => typeof item !== "string" || !isCurrencyCodeUsage(text, item.toUpperCase()),
657
+ );
658
+ if (value.length === 0) delete next[key];
659
+ else next[key] = { ...slot, value };
660
+ continue;
661
+ }
662
+ if (typeof slot.value === "string" && isCurrencyCodeUsage(text, slot.value.toUpperCase())) {
663
+ delete next[key];
664
+ }
665
+ }
666
+ return next;
667
+ }
668
+
669
+ function isPortfolioTradeoffComparisonRequest(text: string): boolean {
670
+ const lower = text.toLowerCase();
671
+ return (
672
+ /\b(?:prioritize|tradeoffs?|growth[-\s]?oriented|dividend|income|which\s+(?:one|is)\s+better|should\s+i)\b/.test(
673
+ lower,
674
+ ) && /\b(?:or|vs\.?|versus|compare)\b/.test(lower)
675
+ );
676
+ }
677
+
678
+ function isCryptoSizingRequest(text: string): boolean {
679
+ const lower = text.toLowerCase();
680
+ const hasPortfolioConstructionIntent =
681
+ /\b(?:build|create|construct|put\s+together)\b/.test(lower) &&
682
+ /\b(?:portfolio|allocation)\b/.test(lower);
683
+ if (hasPortfolioConstructionIntent) return false;
684
+ return (
685
+ /\b(?:btc|bitcoin|crypto)\b/.test(lower) &&
686
+ /\b(?:allocation|range|position\s+size|sizing|exposure|drawdown)\b/.test(lower)
687
+ );
688
+ }
689
+
690
+ function isSpecializedSingleAssetPolicyRequest(text: string): boolean {
691
+ const lower = text.toLowerCase();
692
+ return (
693
+ /\b(?:ticker|symbol|formerly|old ticker|earnings are|earnings tonight)\b/.test(lower) ||
694
+ /\b(?:today|right now|this morning|after close|moved|catalyst)\b/.test(lower) ||
695
+ /\b(?:sentiment|mood|reddit|twitter|x\/twitter)\b/.test(lower) ||
696
+ /\b(?:filing|10-k|10-q|8-k|sec)\b/.test(lower)
697
+ );
698
+ }
699
+
700
+ function isExistingPositionOptionRequest(text: string, extracted: ExtractedEntities): boolean {
701
+ return isCoveredCallRequest(text) || extracted.optionStrategy === "protective_put";
702
+ }
703
+
704
+ function isOptionsEducationOrSuitabilityRequest(text: string): boolean {
705
+ const lower = text.toLowerCase();
706
+ return (
707
+ /\b(?:how\s+does|how\s+do|explain|what\s+is|good\s+idea|make\s+sense|suitable|suitability|is\s+it\s+(?:good|worth|smart))\b/.test(
708
+ lower,
709
+ ) &&
710
+ /\b(?:covered\s+calls?|protective\s+puts?|options?|selling\s+calls?|option\s+income)\b/.test(
711
+ lower,
712
+ )
713
+ );
714
+ }
715
+
716
+ function isSpecificOptionContractSelectionRequest(text: string): boolean {
717
+ const lower = text.toLowerCase();
718
+ return (
719
+ /\b(?:best|which|what\s+(?:strike|contract|option)|rank|screen|specific|right\s+now|today|around\s+earnings|expiration|dte|premium\s+under)\b/.test(
720
+ lower,
721
+ ) && /\b(?:sell|buy|trade|contract|strike|expiration|premium|call|put)\b/.test(lower)
722
+ );
723
+ }
724
+
725
+ function mergeSymbols(primary: string[], secondary: string[]): string[] {
726
+ const merged: string[] = [];
727
+ for (const symbol of [...primary, ...secondary]) {
728
+ if (!merged.includes(symbol)) merged.push(symbol);
729
+ }
730
+ return merged;
731
+ }
732
+
733
+ function mergeStringArrays(primary?: string[], secondary?: string[]): string[] | undefined {
734
+ const merged: string[] = [];
735
+ for (const value of [...(primary ?? []), ...(secondary ?? [])]) {
736
+ if (!merged.includes(value)) merged.push(value);
737
+ }
738
+ return merged.length > 0 ? merged : undefined;
739
+ }
740
+
741
+ function omitUndefined<T>(value: T): T {
742
+ if (Array.isArray(value)) return value.map(omitUndefined) as T;
743
+ if (!value || typeof value !== "object") return value;
744
+ const out: Record<string, unknown> = {};
745
+ for (const [key, entry] of Object.entries(value)) {
746
+ if (entry !== undefined) out[key] = omitUndefined(entry);
747
+ }
748
+ return out as T;
749
+ }
750
+
130
751
  function validateSlots(raw: unknown): Record<string, RouterSlot> {
131
752
  if (raw === undefined || raw === null) return {};
132
753
  if (typeof raw !== "object") {
@@ -176,7 +797,9 @@ function validatePreferenceUpdates(raw: unknown): RouterPreferenceUpdate[] {
176
797
  // (normalized), but any explicit non-"inferred" value is an invariant
177
798
  // violation the caller should see rather than silently lose.
178
799
  if (p.source !== undefined && p.source !== "inferred") {
179
- throw new Error(`preference_updates[${idx}].source must be "inferred" (got ${JSON.stringify(p.source)})`);
800
+ throw new Error(
801
+ `preference_updates[${idx}].source must be "inferred" (got ${JSON.stringify(p.source)})`,
802
+ );
180
803
  }
181
804
  return {
182
805
  key: p.key,
@@ -187,6 +810,72 @@ function validatePreferenceUpdates(raw: unknown): RouterPreferenceUpdate[] {
187
810
  });
188
811
  }
189
812
 
813
+ function validateToolBundles(raw: unknown): ToolBundleName[] {
814
+ const bundles = validateStringArray(raw, "tool_bundles");
815
+ return bundles.filter(isToolBundleName);
816
+ }
817
+
818
+ function removeDroppedSymbolSlots(
819
+ slots: Record<string, RouterSlot>,
820
+ droppedTokens: string[],
821
+ ): Record<string, RouterSlot> {
822
+ if (droppedTokens.length === 0) return slots;
823
+ const dropped = new Set(droppedTokens.map((token) => token.toUpperCase()));
824
+ const next = { ...slots };
825
+
826
+ for (const key of ["symbol", "symbols"]) {
827
+ const slot = next[key];
828
+ if (!slot) continue;
829
+ if (Array.isArray(slot.value)) {
830
+ const value = slot.value.filter(
831
+ (item) => typeof item !== "string" || !dropped.has(item.toUpperCase()),
832
+ );
833
+ if (value.length === 0) {
834
+ delete next[key];
835
+ } else {
836
+ next[key] = { ...slot, value };
837
+ }
838
+ continue;
839
+ }
840
+ if (typeof slot.value === "string" && dropped.has(slot.value.toUpperCase())) {
841
+ delete next[key];
842
+ }
843
+ }
844
+
845
+ return next;
846
+ }
847
+
848
+ function validateDiagnostics(raw: unknown): RouterDiagnostic[] {
849
+ if (raw === undefined || raw === null) return [];
850
+ if (!Array.isArray(raw)) {
851
+ throw new Error("diagnostics must be an array");
852
+ }
853
+ return raw.map((item, idx) => {
854
+ if (!item || typeof item !== "object") {
855
+ throw new Error(`diagnostics[${idx}] must be an object`);
856
+ }
857
+ const diagnostic = item as Record<string, unknown>;
858
+ if (typeof diagnostic.code !== "string" || diagnostic.code.length === 0) {
859
+ throw new Error(`diagnostics[${idx}].code must be a non-empty string`);
860
+ }
861
+ if (typeof diagnostic.message !== "string") {
862
+ throw new Error(`diagnostics[${idx}].message must be a string`);
863
+ }
864
+ const out: RouterDiagnostic = {
865
+ code: diagnostic.code,
866
+ message: diagnostic.message,
867
+ };
868
+ if (
869
+ diagnostic.details &&
870
+ typeof diagnostic.details === "object" &&
871
+ !Array.isArray(diagnostic.details)
872
+ ) {
873
+ out.details = diagnostic.details as Record<string, unknown>;
874
+ }
875
+ return out;
876
+ });
877
+ }
878
+
190
879
  function validateStringArray(raw: unknown, field: string): string[] {
191
880
  if (raw === undefined || raw === null) return [];
192
881
  if (!Array.isArray(raw)) {
@@ -203,11 +892,19 @@ function validateStringArray(raw: unknown, field: string): string[] {
203
892
  function minimalFallback(text: string): RouterOutput {
204
893
  const entities = extractEntities(text);
205
894
  return {
895
+ routeKind: "agent_task",
206
896
  route: "fallback",
207
- entities: { symbols: entities.symbols },
897
+ entities,
208
898
  slots: {},
209
899
  preference_updates: [],
210
900
  missing_required: [],
901
+ tool_bundles: [],
902
+ diagnostics: [
903
+ {
904
+ code: "router_validation_failed",
905
+ message: "router validation failed persistently; emitted minimal fallback",
906
+ },
907
+ ],
211
908
  reasoning: "router validation failed; emitted minimal fallback",
212
909
  };
213
910
  }