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
@@ -0,0 +1,823 @@
1
+ import {
2
+ type ArtifactContractId,
3
+ artifactContractIdsForPlanning,
4
+ } from "../runtime/artifact-contracts.js";
5
+ import type {
6
+ RouterDiagnostic,
7
+ RouterInputContext,
8
+ RouterOutput,
9
+ RouterRouteKind,
10
+ ToolBundleName,
11
+ } from "./router-types.js";
12
+ import type { WorkflowType } from "./types.js";
13
+
14
+ export const PLANNING_VERSION = "planning-v1" as const;
15
+
16
+ export type TaskFamily =
17
+ | "single_asset_decision"
18
+ | "asset_compare"
19
+ | "portfolio_build"
20
+ | "portfolio_review"
21
+ | "macro_allocation_review"
22
+ | "options_strategy"
23
+ | "current_event_explanation"
24
+ | "ticker_disambiguation"
25
+ | "filing_thesis_review"
26
+ | "sentiment_snapshot"
27
+ | "concept_explainer"
28
+ | "retail_finance_tradeoff"
29
+ | "stateful_tracking_update"
30
+ | "backtest_review"
31
+ | "general_fallback";
32
+
33
+ export type CommitmentMode =
34
+ | "decision"
35
+ | "compare_tradeoffs"
36
+ | "framework"
37
+ | "construct"
38
+ | "update_state"
39
+ | "clarify";
40
+
41
+ export type PolicyCardId =
42
+ | "single_asset_decision"
43
+ | "asset_compare"
44
+ | "portfolio_build"
45
+ | "portfolio_review"
46
+ | "portfolio_rebalance_review"
47
+ | "macro_allocation_review"
48
+ | "options_strategy"
49
+ | "current_event_explanation"
50
+ | "ticker_disambiguation"
51
+ | "sentiment_snapshot"
52
+ | "filing_thesis_review"
53
+ | "retail_finance_tradeoff"
54
+ | "concept_explainer"
55
+ | "concept_options_education"
56
+ | "concept_inflation_cash_education"
57
+ | "concept_valuation_metric_education"
58
+ | "backtest_review"
59
+ | "stateful_tracking_update"
60
+ | "general_fallback";
61
+
62
+ export type EvidencePlanId =
63
+ | "market_status"
64
+ | "ticker_disambiguation"
65
+ | "placeholder_single_asset_decision"
66
+ | "placeholder_asset_compare"
67
+ | "placeholder_portfolio_build"
68
+ | "placeholder_portfolio_review"
69
+ | "placeholder_options_strategy"
70
+ | "placeholder_current_event_explanation"
71
+ | "placeholder_sentiment_snapshot"
72
+ | "placeholder_filing_thesis_review"
73
+ | "placeholder_retail_finance_tradeoff"
74
+ | "placeholder_concept_explainer"
75
+ | "placeholder_backtest_review"
76
+ | "placeholder_stateful_tracking_update"
77
+ | "placeholder_general_fallback";
78
+
79
+ export type AnswerContractId =
80
+ | "single_asset_decision"
81
+ | "asset_compare_tradeoff"
82
+ | "portfolio_build"
83
+ | "portfolio_review"
84
+ | "macro_allocation_review"
85
+ | "options_strategy"
86
+ | "current_event_explanation"
87
+ | "ticker_disambiguation"
88
+ | "sentiment_snapshot"
89
+ | "filing_thesis_review"
90
+ | "retail_tradeoff_framework"
91
+ | "concept_explainer"
92
+ | "backtest_review"
93
+ | "stateful_tracking_update"
94
+ | "general_fallback";
95
+
96
+ export type StructuredCheckId =
97
+ | "required_evidence_present"
98
+ | "freshness_disclosed"
99
+ | "data_gap_disclosed"
100
+ | "commitment_mode_respected"
101
+ | "source_coverage_disclosed"
102
+ | "capability_gap_disclosure"
103
+ | "assumption_disclosed"
104
+ | "tax_caveat_present"
105
+ | "target_bands_present"
106
+ | "when_not_ideal_present";
107
+
108
+ export type CapabilityGapId =
109
+ | "market_calendar"
110
+ | "etf_holdings_overlap"
111
+ | "brokerage_comparison"
112
+ | "cash_yield_products"
113
+ | "earnings_event_risk"
114
+ | "fund_tax_efficiency"
115
+ | "forward_rate_probabilities"
116
+ | "sentiment_sample_depth";
117
+
118
+ export interface CapabilityGapDefinition {
119
+ id: CapabilityGapId;
120
+ label: string;
121
+ description: string;
122
+ v1Status: "classified_gap";
123
+ specialistCompetitive: boolean;
124
+ }
125
+
126
+ export const CAPABILITY_GAP_REGISTRY: Record<CapabilityGapId, CapabilityGapDefinition> = {
127
+ market_calendar: {
128
+ id: "market_calendar",
129
+ label: "Market calendar",
130
+ description:
131
+ "Exchange holiday and session-state data beyond deterministic weekday/known-holiday grounding.",
132
+ v1Status: "classified_gap",
133
+ specialistCompetitive: false,
134
+ },
135
+ etf_holdings_overlap: {
136
+ id: "etf_holdings_overlap",
137
+ label: "ETF holdings overlap",
138
+ description: "Exact fund holding overlap, weights, and issuer-level exposure calculations.",
139
+ v1Status: "classified_gap",
140
+ specialistCompetitive: false,
141
+ },
142
+ brokerage_comparison: {
143
+ id: "brokerage_comparison",
144
+ label: "Brokerage comparison",
145
+ description:
146
+ "Live brokerage fees, platform features, account support, and execution-quality comparison data.",
147
+ v1Status: "classified_gap",
148
+ specialistCompetitive: false,
149
+ },
150
+ cash_yield_products: {
151
+ id: "cash_yield_products",
152
+ label: "Cash-yield products",
153
+ description: "Live HYSA, money-market, CD, T-bill, and sweep-rate comparison data.",
154
+ v1Status: "classified_gap",
155
+ specialistCompetitive: false,
156
+ },
157
+ earnings_event_risk: {
158
+ id: "earnings_event_risk",
159
+ label: "Earnings-event risk",
160
+ description:
161
+ "Upcoming earnings timing, transcript, implied move, and event-specific risk coverage.",
162
+ v1Status: "classified_gap",
163
+ specialistCompetitive: false,
164
+ },
165
+ fund_tax_efficiency: {
166
+ id: "fund_tax_efficiency",
167
+ label: "Fund tax efficiency",
168
+ description: "Distribution, turnover, asset-location, and after-tax fund comparison data.",
169
+ v1Status: "classified_gap",
170
+ specialistCompetitive: false,
171
+ },
172
+ forward_rate_probabilities: {
173
+ id: "forward_rate_probabilities",
174
+ label: "Forward-rate probabilities",
175
+ description: "Forward policy-rate probability and curve-derived expectation data.",
176
+ v1Status: "classified_gap",
177
+ specialistCompetitive: false,
178
+ },
179
+ sentiment_sample_depth: {
180
+ id: "sentiment_sample_depth",
181
+ label: "Sentiment sample depth",
182
+ description:
183
+ "Coverage, sample-size, source-depth, and low-volume confidence metadata for sentiment evidence.",
184
+ v1Status: "classified_gap",
185
+ specialistCompetitive: false,
186
+ },
187
+ };
188
+
189
+ export type PlanningBehaviorMode = "observe_only" | "dual_run" | "replacement_active";
190
+
191
+ export interface PlanningSelection {
192
+ taskFamily: TaskFamily;
193
+ commitmentMode: CommitmentMode;
194
+ policyCardId: PolicyCardId;
195
+ evidencePlanId: EvidencePlanId;
196
+ answerContractId: AnswerContractId;
197
+ structuredCheckIds: StructuredCheckId[];
198
+ capabilityGapIds: CapabilityGapId[];
199
+ }
200
+
201
+ export interface PlanningEnvelope extends PlanningSelection {
202
+ version: typeof PLANNING_VERSION;
203
+ behaviorMode: PlanningBehaviorMode;
204
+ workspacePlaceholderIds: string[];
205
+ artifactPlaceholderIds: string[];
206
+ artifactContractIds?: ArtifactContractId[];
207
+ diagnostics: RouterDiagnostic[];
208
+ }
209
+
210
+ export interface PlanningBuildOptions {
211
+ migrationStatuses?: Partial<Record<TaskFamily, PlanningBehaviorMode>>;
212
+ }
213
+
214
+ interface PlanningManifestEntry extends PlanningSelection {
215
+ routeKinds: RouterRouteKind[];
216
+ workflows: Array<Exclude<WorkflowType, "unclassified"> | undefined>;
217
+ compatibleToolBundles: ToolBundleName[];
218
+ migrated: boolean;
219
+ migrationStatus?: PlanningBehaviorMode;
220
+ }
221
+
222
+ export const PLANNING_MANIFEST: Record<TaskFamily, PlanningManifestEntry> = {
223
+ single_asset_decision: {
224
+ routeKinds: ["agent_task"],
225
+ workflows: ["single_asset_analysis", "general_finance_qa", undefined],
226
+ taskFamily: "single_asset_decision",
227
+ commitmentMode: "decision",
228
+ policyCardId: "single_asset_decision",
229
+ evidencePlanId: "placeholder_single_asset_decision",
230
+ answerContractId: "single_asset_decision",
231
+ structuredCheckIds: ["required_evidence_present", "freshness_disclosed", "data_gap_disclosed"],
232
+ capabilityGapIds: [],
233
+ compatibleToolBundles: ["core_market", "options", "sentiment", "sec", "clarification"],
234
+ migrated: false,
235
+ migrationStatus: "replacement_active",
236
+ },
237
+ asset_compare: {
238
+ routeKinds: ["workflow_dispatch", "agent_task"],
239
+ workflows: ["compare_assets"],
240
+ taskFamily: "asset_compare",
241
+ commitmentMode: "compare_tradeoffs",
242
+ policyCardId: "asset_compare",
243
+ evidencePlanId: "placeholder_asset_compare",
244
+ answerContractId: "asset_compare_tradeoff",
245
+ structuredCheckIds: [
246
+ "required_evidence_present",
247
+ "data_gap_disclosed",
248
+ "capability_gap_disclosure",
249
+ ],
250
+ capabilityGapIds: ["etf_holdings_overlap"],
251
+ compatibleToolBundles: ["core_market", "macro", "sentiment", "clarification"],
252
+ migrated: false,
253
+ migrationStatus: "replacement_active",
254
+ },
255
+ portfolio_build: {
256
+ routeKinds: ["workflow_dispatch"],
257
+ workflows: ["portfolio_builder"],
258
+ taskFamily: "portfolio_build",
259
+ commitmentMode: "construct",
260
+ policyCardId: "portfolio_build",
261
+ evidencePlanId: "placeholder_portfolio_build",
262
+ answerContractId: "portfolio_build",
263
+ structuredCheckIds: ["required_evidence_present", "commitment_mode_respected"],
264
+ capabilityGapIds: [],
265
+ compatibleToolBundles: ["core_market", "macro", "sentiment", "clarification"],
266
+ migrated: false,
267
+ },
268
+ portfolio_review: {
269
+ routeKinds: ["agent_task", "workflow_dispatch"],
270
+ workflows: ["general_finance_qa", "portfolio_builder"],
271
+ taskFamily: "portfolio_review",
272
+ commitmentMode: "decision",
273
+ policyCardId: "portfolio_review",
274
+ evidencePlanId: "placeholder_portfolio_review",
275
+ answerContractId: "portfolio_review",
276
+ structuredCheckIds: [
277
+ "required_evidence_present",
278
+ "data_gap_disclosed",
279
+ "commitment_mode_respected",
280
+ ],
281
+ capabilityGapIds: [],
282
+ compatibleToolBundles: ["core_market", "macro", "sentiment", "clarification"],
283
+ migrated: false,
284
+ migrationStatus: "replacement_active",
285
+ },
286
+ options_strategy: {
287
+ routeKinds: ["workflow_dispatch", "agent_task"],
288
+ workflows: ["options_screener", "general_finance_qa"],
289
+ taskFamily: "options_strategy",
290
+ commitmentMode: "decision",
291
+ policyCardId: "options_strategy",
292
+ evidencePlanId: "placeholder_options_strategy",
293
+ answerContractId: "options_strategy",
294
+ structuredCheckIds: ["required_evidence_present", "freshness_disclosed"],
295
+ capabilityGapIds: [],
296
+ compatibleToolBundles: ["core_market", "options", "sentiment", "clarification"],
297
+ migrated: false,
298
+ migrationStatus: "replacement_active",
299
+ },
300
+ current_event_explanation: {
301
+ routeKinds: ["agent_task"],
302
+ workflows: ["general_finance_qa", "single_asset_analysis", undefined],
303
+ taskFamily: "current_event_explanation",
304
+ commitmentMode: "framework",
305
+ policyCardId: "current_event_explanation",
306
+ evidencePlanId: "market_status",
307
+ answerContractId: "current_event_explanation",
308
+ structuredCheckIds: ["required_evidence_present", "freshness_disclosed", "data_gap_disclosed"],
309
+ capabilityGapIds: ["market_calendar"],
310
+ compatibleToolBundles: ["core_market", "macro", "sentiment", "sec", "clarification"],
311
+ migrated: false,
312
+ migrationStatus: "replacement_active",
313
+ },
314
+ ticker_disambiguation: {
315
+ routeKinds: ["agent_task"],
316
+ workflows: ["general_finance_qa", "single_asset_analysis", undefined],
317
+ taskFamily: "ticker_disambiguation",
318
+ commitmentMode: "framework",
319
+ policyCardId: "ticker_disambiguation",
320
+ evidencePlanId: "ticker_disambiguation",
321
+ answerContractId: "ticker_disambiguation",
322
+ structuredCheckIds: ["required_evidence_present", "data_gap_disclosed"],
323
+ capabilityGapIds: ["earnings_event_risk"],
324
+ compatibleToolBundles: ["core_market", "clarification"],
325
+ migrated: true,
326
+ },
327
+ filing_thesis_review: {
328
+ routeKinds: ["agent_task"],
329
+ workflows: ["general_finance_qa", "single_asset_analysis"],
330
+ taskFamily: "filing_thesis_review",
331
+ commitmentMode: "framework",
332
+ policyCardId: "filing_thesis_review",
333
+ evidencePlanId: "placeholder_filing_thesis_review",
334
+ answerContractId: "filing_thesis_review",
335
+ structuredCheckIds: ["required_evidence_present", "data_gap_disclosed"],
336
+ capabilityGapIds: [],
337
+ compatibleToolBundles: ["core_market", "sec"],
338
+ migrated: false,
339
+ migrationStatus: "replacement_active",
340
+ },
341
+ sentiment_snapshot: {
342
+ routeKinds: ["agent_task"],
343
+ workflows: ["general_finance_qa", "single_asset_analysis"],
344
+ taskFamily: "sentiment_snapshot",
345
+ commitmentMode: "framework",
346
+ policyCardId: "sentiment_snapshot",
347
+ evidencePlanId: "placeholder_sentiment_snapshot",
348
+ answerContractId: "sentiment_snapshot",
349
+ structuredCheckIds: [
350
+ "required_evidence_present",
351
+ "source_coverage_disclosed",
352
+ "data_gap_disclosed",
353
+ ],
354
+ capabilityGapIds: ["sentiment_sample_depth"],
355
+ compatibleToolBundles: ["core_market", "sentiment"],
356
+ migrated: false,
357
+ migrationStatus: "replacement_active",
358
+ },
359
+ concept_explainer: {
360
+ routeKinds: ["agent_task"],
361
+ workflows: ["general_finance_qa", undefined],
362
+ taskFamily: "concept_explainer",
363
+ commitmentMode: "framework",
364
+ policyCardId: "concept_explainer",
365
+ evidencePlanId: "placeholder_concept_explainer",
366
+ answerContractId: "concept_explainer",
367
+ structuredCheckIds: ["commitment_mode_respected"],
368
+ capabilityGapIds: [],
369
+ compatibleToolBundles: [],
370
+ migrated: false,
371
+ migrationStatus: "replacement_active",
372
+ },
373
+ retail_finance_tradeoff: {
374
+ routeKinds: ["agent_task"],
375
+ workflows: ["general_finance_qa", undefined],
376
+ taskFamily: "retail_finance_tradeoff",
377
+ commitmentMode: "framework",
378
+ policyCardId: "retail_finance_tradeoff",
379
+ evidencePlanId: "placeholder_retail_finance_tradeoff",
380
+ answerContractId: "retail_tradeoff_framework",
381
+ structuredCheckIds: ["data_gap_disclosed", "capability_gap_disclosure"],
382
+ capabilityGapIds: ["brokerage_comparison", "cash_yield_products", "fund_tax_efficiency"],
383
+ compatibleToolBundles: [],
384
+ migrated: false,
385
+ migrationStatus: "replacement_active",
386
+ },
387
+ stateful_tracking_update: {
388
+ routeKinds: ["agent_task"],
389
+ workflows: ["watchlist_or_tracking"],
390
+ taskFamily: "stateful_tracking_update",
391
+ commitmentMode: "update_state",
392
+ policyCardId: "stateful_tracking_update",
393
+ evidencePlanId: "placeholder_stateful_tracking_update",
394
+ answerContractId: "stateful_tracking_update",
395
+ structuredCheckIds: ["commitment_mode_respected"],
396
+ capabilityGapIds: [],
397
+ compatibleToolBundles: ["core_market", "clarification"],
398
+ migrated: false,
399
+ migrationStatus: "replacement_active",
400
+ },
401
+ backtest_review: {
402
+ routeKinds: ["agent_task"],
403
+ workflows: ["general_finance_qa", "single_asset_analysis"],
404
+ taskFamily: "backtest_review",
405
+ commitmentMode: "framework",
406
+ policyCardId: "backtest_review",
407
+ evidencePlanId: "placeholder_backtest_review",
408
+ answerContractId: "backtest_review",
409
+ structuredCheckIds: [
410
+ "required_evidence_present",
411
+ "data_gap_disclosed",
412
+ "source_coverage_disclosed",
413
+ ],
414
+ capabilityGapIds: [],
415
+ compatibleToolBundles: ["core_market", "options", "sentiment", "sec", "clarification"],
416
+ migrated: false,
417
+ migrationStatus: "replacement_active",
418
+ },
419
+ macro_allocation_review: {
420
+ routeKinds: ["agent_task"],
421
+ workflows: ["general_finance_qa"],
422
+ taskFamily: "macro_allocation_review",
423
+ commitmentMode: "decision",
424
+ policyCardId: "macro_allocation_review",
425
+ evidencePlanId: "market_status",
426
+ answerContractId: "macro_allocation_review",
427
+ structuredCheckIds: ["required_evidence_present", "freshness_disclosed", "data_gap_disclosed"],
428
+ capabilityGapIds: ["market_calendar", "forward_rate_probabilities"],
429
+ compatibleToolBundles: ["core_market", "macro", "sentiment", "clarification"],
430
+ migrated: false,
431
+ migrationStatus: "replacement_active",
432
+ },
433
+ general_fallback: {
434
+ routeKinds: ["agent_task", "clarification", "pass_through"],
435
+ workflows: [undefined, "general_finance_qa"],
436
+ taskFamily: "general_fallback",
437
+ commitmentMode: "framework",
438
+ policyCardId: "general_fallback",
439
+ evidencePlanId: "placeholder_general_fallback",
440
+ answerContractId: "general_fallback",
441
+ structuredCheckIds: ["data_gap_disclosed"],
442
+ capabilityGapIds: [],
443
+ compatibleToolBundles: ["core_market", "clarification"],
444
+ migrated: false,
445
+ },
446
+ };
447
+
448
+ export function buildPlanningEnvelope(
449
+ input: RouterInputContext,
450
+ output: RouterOutput,
451
+ options: PlanningBuildOptions = {},
452
+ ): PlanningEnvelope {
453
+ const proposed = defaultPlanningSelection(input, output);
454
+ const { selection, diagnostics } = validatePlanningSelection(output, proposed);
455
+ const manifestEntry = PLANNING_MANIFEST[selection.taskFamily];
456
+ const behaviorMode: PlanningBehaviorMode =
457
+ options.migrationStatuses?.[selection.taskFamily] ??
458
+ manifestEntry.migrationStatus ??
459
+ (manifestEntry.migrated ? "replacement_active" : "observe_only");
460
+
461
+ return {
462
+ version: PLANNING_VERSION,
463
+ ...selection,
464
+ behaviorMode,
465
+ workspacePlaceholderIds: [],
466
+ artifactPlaceholderIds: [],
467
+ artifactContractIds: artifactContractIdsForPlanning(selection),
468
+ diagnostics: [
469
+ ...diagnostics,
470
+ ...(output.diagnostics.length > 0
471
+ ? [
472
+ {
473
+ code: "planning_after_router_corrections",
474
+ message: "planning selected after router diagnostics were applied",
475
+ },
476
+ ]
477
+ : []),
478
+ ...(behaviorMode === "observe_only"
479
+ ? [
480
+ {
481
+ code: "planning_observe_only",
482
+ message:
483
+ "planning metadata is recorded without changing active prompt, route, workflow, tools, or answer behavior",
484
+ },
485
+ ]
486
+ : []),
487
+ ],
488
+ };
489
+ }
490
+
491
+ export function validatePlanningSelection(
492
+ output: RouterOutput,
493
+ proposed: PlanningSelection,
494
+ ): { selection: PlanningSelection; diagnostics: RouterDiagnostic[] } {
495
+ const manifestEntry = PLANNING_MANIFEST[proposed.taskFamily];
496
+ if (isAllowedForOutput(manifestEntry, output)) {
497
+ return { selection: proposed, diagnostics: [] };
498
+ }
499
+
500
+ const fallback = defaultTaskFamilyForOutput(output, "");
501
+ return {
502
+ selection: selectionForTaskFamily(fallback),
503
+ diagnostics: [
504
+ {
505
+ code: "planning_task_family_corrected",
506
+ message: `${proposed.taskFamily} is not supported for ${output.routeKind}${output.workflow ? `/${output.workflow}` : ""}; using ${fallback}`,
507
+ },
508
+ ],
509
+ };
510
+ }
511
+
512
+ function defaultPlanningSelection(
513
+ input: RouterInputContext,
514
+ output: RouterOutput,
515
+ ): PlanningSelection {
516
+ const selection = selectionForTaskFamily(defaultTaskFamilyForOutput(output, input.text));
517
+ return refinePlanningSelectionForPrompt(selection, input.text);
518
+ }
519
+
520
+ function selectionForTaskFamily(taskFamily: TaskFamily): PlanningSelection {
521
+ const entry = PLANNING_MANIFEST[taskFamily];
522
+ return {
523
+ taskFamily: entry.taskFamily,
524
+ commitmentMode: entry.commitmentMode,
525
+ policyCardId: entry.policyCardId,
526
+ evidencePlanId: entry.evidencePlanId,
527
+ answerContractId: entry.answerContractId,
528
+ structuredCheckIds: [...entry.structuredCheckIds],
529
+ capabilityGapIds: [...entry.capabilityGapIds],
530
+ };
531
+ }
532
+
533
+ function defaultTaskFamilyForOutput(output: RouterOutput, text: string): TaskFamily {
534
+ const lower = text.toLowerCase();
535
+ if (output.routeKind === "clarification") return "general_fallback";
536
+ if (output.routeKind === "pass_through") return "general_fallback";
537
+ if (
538
+ output.workflow === "portfolio_builder" &&
539
+ isExistingRetirementAllocationReviewPrompt(lower)
540
+ ) {
541
+ return "portfolio_review";
542
+ }
543
+ if (output.workflow === "portfolio_builder") return "portfolio_build";
544
+ if (output.workflow === "options_screener") return "options_strategy";
545
+ if (output.workflow === "compare_assets") return "asset_compare";
546
+ if (output.workflow === "watchlist_or_tracking") return "stateful_tracking_update";
547
+
548
+ if (output.entities.symbols.length === 0 && isNoToolConceptEducationPrompt(lower)) {
549
+ return "concept_explainer";
550
+ }
551
+ if (isTickerDisambiguationPrompt(lower)) {
552
+ return "ticker_disambiguation";
553
+ }
554
+ if (isOptionsStrategyPrompt(lower)) return "options_strategy";
555
+ if (/\bbacktest(?:ing|ed)?\b/.test(lower)) {
556
+ return "backtest_review";
557
+ }
558
+ if (/\b(?:macro|inflation|fed|rates?|duration|recession)\b/.test(lower)) {
559
+ return "macro_allocation_review";
560
+ }
561
+ if (/\b(?:sentiment|mood|reddit|twitter|x\/twitter)\b/.test(lower)) {
562
+ return "sentiment_snapshot";
563
+ }
564
+ if (
565
+ isPortfolioRebalancePrompt(lower) ||
566
+ isAddToExistingHoldingsPrompt(lower, output.entities.symbols.length)
567
+ ) {
568
+ return "portfolio_review";
569
+ }
570
+ if (/\b(?:today|right now|this morning|after close|moved|catalyst)\b/.test(lower)) {
571
+ return "current_event_explanation";
572
+ }
573
+ if (/\b(?:filing|10-k|10-q|8-k|sec)\b/.test(lower)) {
574
+ return "filing_thesis_review";
575
+ }
576
+ if (/\b(?:brokerage|hysa|money-market|t-bills?|cds?|mortgage|taxable account)\b/.test(lower)) {
577
+ return "retail_finance_tradeoff";
578
+ }
579
+ if (
580
+ /\b(?:btc|bitcoin|crypto)\b/.test(lower) &&
581
+ /\b(?:allocation|range|position\s+size|sizing|drawdown)\b/.test(lower)
582
+ ) {
583
+ return "retail_finance_tradeoff";
584
+ }
585
+ if (
586
+ /\b(?:60\/40|portfolio|allocation)\b/.test(lower) &&
587
+ /\b(?:evaluate|evaluation|review|risk|prospects)\b/.test(lower)
588
+ ) {
589
+ return "portfolio_review";
590
+ }
591
+ if (
592
+ output.entities.symbols.length === 1 &&
593
+ /\b(?:analyze|buy|sell|wait|avoid|recommendation|attractive)\b/.test(lower)
594
+ ) {
595
+ return "single_asset_decision";
596
+ }
597
+ if (output.workflow === "single_asset_analysis") return "single_asset_decision";
598
+ if (
599
+ /\b(?:explain|what is|what does|how to|define)\b/.test(lower) &&
600
+ output.entities.symbols.length === 0
601
+ ) {
602
+ return "concept_explainer";
603
+ }
604
+ return "general_fallback";
605
+ }
606
+
607
+ function refinePlanningSelectionForPrompt(
608
+ selection: PlanningSelection,
609
+ text: string,
610
+ ): PlanningSelection {
611
+ const lower = text.toLowerCase();
612
+
613
+ if (selection.taskFamily === "concept_explainer") {
614
+ if (isOptionsEducationPrompt(lower)) {
615
+ return {
616
+ ...selection,
617
+ policyCardId: "concept_options_education",
618
+ structuredCheckIds: mergeStructuredChecks(selection.structuredCheckIds, [
619
+ "tax_caveat_present",
620
+ "when_not_ideal_present",
621
+ ]),
622
+ };
623
+ }
624
+ if (isInflationCashEducationPrompt(lower)) {
625
+ return {
626
+ ...selection,
627
+ policyCardId: "concept_inflation_cash_education",
628
+ structuredCheckIds: mergeStructuredChecks(selection.structuredCheckIds, [
629
+ "tax_caveat_present",
630
+ ]),
631
+ };
632
+ }
633
+ if (isValuationMetricEducationPrompt(lower)) {
634
+ return {
635
+ ...selection,
636
+ policyCardId: "concept_valuation_metric_education",
637
+ };
638
+ }
639
+ }
640
+
641
+ if (
642
+ selection.taskFamily === "portfolio_review" &&
643
+ (isPortfolioRebalancePrompt(lower) || isAddToExistingHoldingsPrompt(lower, 2)) &&
644
+ !isExistingRetirementAllocationReviewPrompt(lower)
645
+ ) {
646
+ return {
647
+ ...selection,
648
+ policyCardId: "portfolio_rebalance_review",
649
+ structuredCheckIds: mergeStructuredChecks(selection.structuredCheckIds, [
650
+ "assumption_disclosed",
651
+ "target_bands_present",
652
+ ]),
653
+ capabilityGapIds: mergeCapabilityGaps(selection.capabilityGapIds, ["etf_holdings_overlap"]),
654
+ };
655
+ }
656
+
657
+ if (
658
+ selection.taskFamily === "retail_finance_tradeoff" &&
659
+ /\b(?:where should|hysa|money-market|t-bills?|cds?|mortgage|versus|vs\.?|or)\b/.test(lower)
660
+ ) {
661
+ return {
662
+ ...selection,
663
+ commitmentMode: "compare_tradeoffs",
664
+ };
665
+ }
666
+
667
+ if (
668
+ selection.taskFamily === "retail_finance_tradeoff" &&
669
+ /\b(?:btc|bitcoin|crypto)\b/.test(lower) &&
670
+ /\b(?:allocation|range|position\s+size|sizing|drawdown)\b/.test(lower)
671
+ ) {
672
+ return {
673
+ ...selection,
674
+ commitmentMode: "decision",
675
+ };
676
+ }
677
+
678
+ if (
679
+ selection.taskFamily === "ticker_disambiguation" &&
680
+ /\b(?:earnings|trim|hedge|hold|event[-\s]?risk|position\s+size)\b/.test(lower)
681
+ ) {
682
+ return {
683
+ ...selection,
684
+ commitmentMode: "decision",
685
+ };
686
+ }
687
+
688
+ if (
689
+ selection.taskFamily === "macro_allocation_review" &&
690
+ /\b(?:sentiment|mood|reddit|twitter|x\/twitter)\b/.test(lower)
691
+ ) {
692
+ return {
693
+ ...selection,
694
+ commitmentMode: "framework",
695
+ capabilityGapIds: mergeCapabilityGaps(selection.capabilityGapIds, ["sentiment_sample_depth"]),
696
+ };
697
+ }
698
+
699
+ return selection;
700
+ }
701
+
702
+ function isNoToolConceptEducationPrompt(lower: string): boolean {
703
+ if (
704
+ !/\b(?:explain|what\s+is|what\s+are|what\s+does|how\s+does|how\s+do|how\s+to|define)\b/.test(
705
+ lower,
706
+ )
707
+ ) {
708
+ return false;
709
+ }
710
+ return (
711
+ isOptionsEducationPrompt(lower) ||
712
+ isInflationCashEducationPrompt(lower) ||
713
+ isValuationMetricEducationPrompt(lower)
714
+ );
715
+ }
716
+
717
+ function isOptionsEducationPrompt(lower: string): boolean {
718
+ return /\b(?:covered\s+calls?|protective\s+puts?|options?|option\s+premium|assignment\s+risk|strike|expiration|delta|theta|greeks?)\b/.test(
719
+ lower,
720
+ );
721
+ }
722
+
723
+ function isOptionsStrategyPrompt(lower: string): boolean {
724
+ const explicitOptionsStrategy =
725
+ /\b(?:covered\s+calls?|protective\s+puts?|buy(?:ing)?\s+puts?|sell(?:ing)?\s+calls?|sell(?:ing)?\s+puts?|options?\s+income|option\s+strategy|options?\s+hedge|collar)\b/.test(
726
+ lower,
727
+ );
728
+ const optionsHedge =
729
+ /\bhedg(?:e|ing)\b.{0,80}\b(?:puts|put\s+options?|options?)\b/.test(lower) ||
730
+ /\b(?:puts|put\s+options?|options?)\b.{0,80}\bhedg(?:e|ing)\b/.test(lower);
731
+ return (
732
+ (explicitOptionsStrategy || optionsHedge) &&
733
+ /\b(?:own|have|shares?|position|cost\s+basis|good\s+idea|make\s+sense|income|premium|strike|expiration|assignment|stable|flat|protect|hedge|sell)\b/.test(
734
+ lower,
735
+ )
736
+ );
737
+ }
738
+
739
+ function isTickerDisambiguationPrompt(lower: string): boolean {
740
+ if (/\b(?:ticker|symbol|formerly|old ticker)\b/.test(lower)) return true;
741
+ return (
742
+ /\b(?:earnings are|earnings tonight)\b/.test(lower) &&
743
+ /\b(?:trim|hedge|hold|event[-\s]?risk|position\s+size)\b/.test(lower) &&
744
+ !/\b(?:covered\s+calls?|protective\s+puts?|sell(?:ing)?\s+calls?|sell(?:ing)?\s+puts?|option\s+chain|strike|expiration|premium)\b/.test(
745
+ lower,
746
+ )
747
+ );
748
+ }
749
+
750
+ function isInflationCashEducationPrompt(lower: string): boolean {
751
+ return (
752
+ /\b(?:inflation|purchasing\s+power|real\s+returns?|cash\s+savings?|cash\s+drag|tips|short(?:er)?[-\s]?duration)\b/.test(
753
+ lower,
754
+ ) &&
755
+ /\b(?:cash|savings?|purchasing\s+power|real\s+returns?|bonds?|tips|protect|protection|affect)\b/.test(
756
+ lower,
757
+ )
758
+ );
759
+ }
760
+
761
+ function isValuationMetricEducationPrompt(lower: string): boolean {
762
+ return /\b(?:p\/e|pe\s+ratio|price[-\s]?to[-\s]?earnings|ev\/ebitda|p\/s|price[-\s]?to[-\s]?sales|valuation\s+metric|trailing|forward\s+earnings|normalized\s+earnings|cyclically\s+adjusted)\b/.test(
763
+ lower,
764
+ );
765
+ }
766
+
767
+ function isPortfolioRebalancePrompt(lower: string): boolean {
768
+ return (
769
+ /\b(?:portfolio|allocation|holdings?|sleeves?|ira|etfs?|funds?|s&p\s*500|index|equity|bonds?|cash)\b/.test(
770
+ lower,
771
+ ) &&
772
+ /\b(?:rebalance|diversify|diversifying|concentration|overweight|underweight|target\s+bands?|drift|reduce\s+concentration|adjust|adjustment|more\s+aggressive|higher\s+growth|too\s+risky|riskier|worried|crash|hedge|protect|protection|missing\s+out\s+on\s+growth)\b/.test(
773
+ lower,
774
+ )
775
+ );
776
+ }
777
+
778
+ function isAddToExistingHoldingsPrompt(lower: string, symbolCount: number): boolean {
779
+ return (
780
+ symbolCount >= 2 &&
781
+ /\b(?:already\s+own|already\s+hold|current(?:ly)?\s+own|existing\s+(?:holdings?|portfolio|position)|my\s+portfolio)\b/.test(
782
+ lower,
783
+ ) &&
784
+ /\b(?:add(?:ing)?|buy(?:ing)?|make\s+sense|fit|long[-\s]?term|growth)\b/.test(lower)
785
+ );
786
+ }
787
+
788
+ function isExistingRetirementAllocationReviewPrompt(lower: string): boolean {
789
+ return (
790
+ /\b(?:401k|401\(k\)|target[-\s]?date|tdf)\b/.test(lower) &&
791
+ /\b(?:mostly|current(?:ly)?|already|have|holding|invested|allocation)\b/.test(lower) &&
792
+ /\b(?:review|solid|worried|inflation|diversify|boost returns?|other options?|without taking crazy risks?|risk)\b/.test(
793
+ lower,
794
+ )
795
+ );
796
+ }
797
+
798
+ function mergeCapabilityGaps(
799
+ primary: readonly CapabilityGapId[],
800
+ secondary: readonly CapabilityGapId[],
801
+ ): CapabilityGapId[] {
802
+ const merged: CapabilityGapId[] = [];
803
+ for (const gap of [...primary, ...secondary]) {
804
+ if (!merged.includes(gap)) merged.push(gap);
805
+ }
806
+ return merged;
807
+ }
808
+
809
+ function mergeStructuredChecks(
810
+ primary: readonly StructuredCheckId[],
811
+ secondary: readonly StructuredCheckId[],
812
+ ): StructuredCheckId[] {
813
+ const merged: StructuredCheckId[] = [];
814
+ for (const check of [...primary, ...secondary]) {
815
+ if (!merged.includes(check)) merged.push(check);
816
+ }
817
+ return merged;
818
+ }
819
+
820
+ function isAllowedForOutput(entry: PlanningManifestEntry, output: RouterOutput): boolean {
821
+ if (!entry.routeKinds.includes(output.routeKind)) return false;
822
+ return entry.workflows.includes(output.workflow);
823
+ }