opencandle 0.5.0 → 0.7.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 (574) hide show
  1. package/README.md +170 -186
  2. package/dist/analysts/contracts.d.ts +1 -3
  3. package/dist/analysts/contracts.js +1 -11
  4. package/dist/analysts/contracts.js.map +1 -1
  5. package/dist/analysts/orchestrator.d.ts +1 -3
  6. package/dist/analysts/orchestrator.js +1 -26
  7. package/dist/analysts/orchestrator.js.map +1 -1
  8. package/dist/cli.js +66 -7
  9. package/dist/cli.js.map +1 -1
  10. package/dist/config.d.ts +13 -3
  11. package/dist/config.js +25 -5
  12. package/dist/config.js.map +1 -1
  13. package/dist/index.d.ts +1 -1
  14. package/dist/index.js +1 -1
  15. package/dist/index.js.map +1 -1
  16. package/dist/infra/cache.d.ts +8 -11
  17. package/dist/infra/cache.js +17 -15
  18. package/dist/infra/cache.js.map +1 -1
  19. package/dist/infra/http-client.d.ts +4 -1
  20. package/dist/infra/http-client.js +59 -6
  21. package/dist/infra/http-client.js.map +1 -1
  22. package/dist/infra/index.d.ts +2 -3
  23. package/dist/infra/index.js +2 -3
  24. package/dist/infra/index.js.map +1 -1
  25. package/dist/infra/native-dependencies.js +2 -2
  26. package/dist/infra/native-dependencies.js.map +1 -1
  27. package/dist/infra/node-version.js.map +1 -1
  28. package/dist/infra/opencandle-paths.d.ts +0 -3
  29. package/dist/infra/opencandle-paths.js +4 -11
  30. package/dist/infra/opencandle-paths.js.map +1 -1
  31. package/dist/infra/rate-limiter.js +12 -9
  32. package/dist/infra/rate-limiter.js.map +1 -1
  33. package/dist/market-state/alert-conditions.d.ts +34 -0
  34. package/dist/market-state/alert-conditions.js +23 -0
  35. package/dist/market-state/alert-conditions.js.map +1 -0
  36. package/dist/market-state/alert-runner.d.ts +55 -0
  37. package/dist/market-state/alert-runner.js +634 -0
  38. package/dist/market-state/alert-runner.js.map +1 -0
  39. package/dist/market-state/daily-report.d.ts +26 -0
  40. package/dist/market-state/daily-report.js +179 -0
  41. package/dist/market-state/daily-report.js.map +1 -0
  42. package/dist/market-state/local-automation-service.d.ts +25 -0
  43. package/dist/market-state/local-automation-service.js +119 -0
  44. package/dist/market-state/local-automation-service.js.map +1 -0
  45. package/dist/market-state/notification-delivery.d.ts +14 -0
  46. package/dist/market-state/notification-delivery.js +139 -0
  47. package/dist/market-state/notification-delivery.js.map +1 -0
  48. package/dist/market-state/resolve-for-mutation.d.ts +10 -0
  49. package/dist/market-state/resolve-for-mutation.js +15 -0
  50. package/dist/market-state/resolve-for-mutation.js.map +1 -0
  51. package/dist/market-state/resolve.d.ts +14 -0
  52. package/dist/market-state/resolve.js +89 -0
  53. package/dist/market-state/resolve.js.map +1 -0
  54. package/dist/market-state/service.d.ts +527 -0
  55. package/dist/market-state/service.js +1099 -0
  56. package/dist/market-state/service.js.map +1 -0
  57. package/dist/memory/index.d.ts +7 -7
  58. package/dist/memory/index.js +6 -6
  59. package/dist/memory/index.js.map +1 -1
  60. package/dist/memory/manager.js +11 -11
  61. package/dist/memory/manager.js.map +1 -1
  62. package/dist/memory/retrieval.js +7 -4
  63. package/dist/memory/retrieval.js.map +1 -1
  64. package/dist/memory/sqlite.js +385 -3
  65. package/dist/memory/sqlite.js.map +1 -1
  66. package/dist/memory/storage.js +1 -2
  67. package/dist/memory/storage.js.map +1 -1
  68. package/dist/memory/tool-defaults.js +64 -28
  69. package/dist/memory/tool-defaults.js.map +1 -1
  70. package/dist/memory/types.js.map +1 -1
  71. package/dist/monitor.d.ts +2 -0
  72. package/dist/monitor.js +104 -0
  73. package/dist/monitor.js.map +1 -0
  74. package/dist/onboarding/connect.d.ts +2 -2
  75. package/dist/onboarding/connect.js +13 -8
  76. package/dist/onboarding/connect.js.map +1 -1
  77. package/dist/onboarding/credential-interceptor.js +1 -1
  78. package/dist/onboarding/credential-interceptor.js.map +1 -1
  79. package/dist/onboarding/degradation-accumulator.js +1 -3
  80. package/dist/onboarding/degradation-accumulator.js.map +1 -1
  81. package/dist/onboarding/provider-status.d.ts +48 -0
  82. package/dist/onboarding/provider-status.js +285 -0
  83. package/dist/onboarding/provider-status.js.map +1 -0
  84. package/dist/onboarding/providers.d.ts +85 -8
  85. package/dist/onboarding/providers.js +83 -18
  86. package/dist/onboarding/providers.js.map +1 -1
  87. package/dist/onboarding/state.d.ts +1 -0
  88. package/dist/onboarding/state.js +5 -0
  89. package/dist/onboarding/state.js.map +1 -1
  90. package/dist/onboarding/tool-helpers.js +1 -1
  91. package/dist/onboarding/tool-helpers.js.map +1 -1
  92. package/dist/onboarding/tool-tags.d.ts +12 -1
  93. package/dist/onboarding/tool-tags.js +37 -5
  94. package/dist/onboarding/tool-tags.js.map +1 -1
  95. package/dist/onboarding/validation.d.ts +2 -2
  96. package/dist/onboarding/validation.js +1 -1
  97. package/dist/onboarding/validation.js.map +1 -1
  98. package/dist/pi/opencandle-extension.d.ts +8 -0
  99. package/dist/pi/opencandle-extension.js +502 -42
  100. package/dist/pi/opencandle-extension.js.map +1 -1
  101. package/dist/pi/session.d.ts +1 -1
  102. package/dist/pi/session.js +3 -1
  103. package/dist/pi/session.js.map +1 -1
  104. package/dist/pi/setup.js +8 -3
  105. package/dist/pi/setup.js.map +1 -1
  106. package/dist/pi/tool-adapter.d.ts +4 -1
  107. package/dist/pi/tool-adapter.js +10 -6
  108. package/dist/pi/tool-adapter.js.map +1 -1
  109. package/dist/prompts/context-builder.d.ts +1 -1
  110. package/dist/prompts/context-builder.js +20 -7
  111. package/dist/prompts/context-builder.js.map +1 -1
  112. package/dist/prompts/policy-cards.d.ts +1 -1
  113. package/dist/prompts/policy-cards.js +2 -2
  114. package/dist/prompts/policy-cards.js.map +1 -1
  115. package/dist/prompts/sections.d.ts +1 -1
  116. package/dist/prompts/symbol-preflight.d.ts +20 -0
  117. package/dist/prompts/symbol-preflight.js +49 -0
  118. package/dist/prompts/symbol-preflight.js.map +1 -0
  119. package/dist/prompts/workflow-prompts.d.ts +1 -1
  120. package/dist/prompts/workflow-prompts.js +54 -16
  121. package/dist/prompts/workflow-prompts.js.map +1 -1
  122. package/dist/providers/alpha-vantage.d.ts +1 -1
  123. package/dist/providers/alpha-vantage.js +26 -7
  124. package/dist/providers/alpha-vantage.js.map +1 -1
  125. package/dist/providers/coingecko.js +1 -1
  126. package/dist/providers/coingecko.js.map +1 -1
  127. package/dist/providers/errors.d.ts +5 -0
  128. package/dist/providers/errors.js +11 -0
  129. package/dist/providers/errors.js.map +1 -0
  130. package/dist/providers/exa-search.d.ts +2 -2
  131. package/dist/providers/exa-search.js +19 -11
  132. package/dist/providers/exa-search.js.map +1 -1
  133. package/dist/providers/external-tool-error.d.ts +10 -0
  134. package/dist/providers/external-tool-error.js +21 -0
  135. package/dist/providers/external-tool-error.js.map +1 -0
  136. package/dist/providers/fear-greed.js +1 -1
  137. package/dist/providers/fear-greed.js.map +1 -1
  138. package/dist/providers/finnhub.js +3 -5
  139. package/dist/providers/finnhub.js.map +1 -1
  140. package/dist/providers/fred.js +2 -2
  141. package/dist/providers/fred.js.map +1 -1
  142. package/dist/providers/index.d.ts +7 -6
  143. package/dist/providers/index.js +6 -5
  144. package/dist/providers/index.js.map +1 -1
  145. package/dist/providers/reddit-cli.d.ts +36 -0
  146. package/dist/providers/reddit-cli.js +201 -0
  147. package/dist/providers/reddit-cli.js.map +1 -0
  148. package/dist/providers/reddit.d.ts +1 -1
  149. package/dist/providers/reddit.js +9 -37
  150. package/dist/providers/reddit.js.map +1 -1
  151. package/dist/providers/sec-edgar.d.ts +1 -0
  152. package/dist/providers/sec-edgar.js +12 -4
  153. package/dist/providers/sec-edgar.js.map +1 -1
  154. package/dist/providers/tradingview.d.ts +47 -0
  155. package/dist/providers/tradingview.js +275 -0
  156. package/dist/providers/tradingview.js.map +1 -0
  157. package/dist/providers/twitter-cli.d.ts +40 -0
  158. package/dist/providers/twitter-cli.js +153 -0
  159. package/dist/providers/twitter-cli.js.map +1 -0
  160. package/dist/providers/twitter.d.ts +0 -8
  161. package/dist/providers/twitter.js +8 -60
  162. package/dist/providers/twitter.js.map +1 -1
  163. package/dist/providers/web-search.js +26 -12
  164. package/dist/providers/web-search.js.map +1 -1
  165. package/dist/providers/with-fallback.js +4 -2
  166. package/dist/providers/with-fallback.js.map +1 -1
  167. package/dist/providers/wrap-provider.d.ts +2 -3
  168. package/dist/providers/wrap-provider.js +44 -8
  169. package/dist/providers/wrap-provider.js.map +1 -1
  170. package/dist/providers/yahoo-finance.d.ts +1 -1
  171. package/dist/providers/yahoo-finance.js +153 -48
  172. package/dist/providers/yahoo-finance.js.map +1 -1
  173. package/dist/routing/classify-intent.d.ts +6 -0
  174. package/dist/routing/classify-intent.js +78 -7
  175. package/dist/routing/classify-intent.js.map +1 -1
  176. package/dist/routing/defaults.d.ts +1 -1
  177. package/dist/routing/entity-extractor.d.ts +1 -0
  178. package/dist/routing/entity-extractor.js +234 -29
  179. package/dist/routing/entity-extractor.js.map +1 -1
  180. package/dist/routing/fund-symbols.d.ts +2 -0
  181. package/dist/routing/fund-symbols.js +55 -0
  182. package/dist/routing/fund-symbols.js.map +1 -0
  183. package/dist/routing/horizon.d.ts +1 -0
  184. package/dist/routing/horizon.js +10 -0
  185. package/dist/routing/horizon.js.map +1 -0
  186. package/dist/routing/index.d.ts +10 -10
  187. package/dist/routing/index.js +6 -6
  188. package/dist/routing/index.js.map +1 -1
  189. package/dist/routing/planning.d.ts +2 -2
  190. package/dist/routing/planning.js +65 -34
  191. package/dist/routing/planning.js.map +1 -1
  192. package/dist/routing/route-manifest.d.ts +2 -2
  193. package/dist/routing/route-manifest.js +25 -4
  194. package/dist/routing/route-manifest.js.map +1 -1
  195. package/dist/routing/router-llm-client.js.map +1 -1
  196. package/dist/routing/router-prompt.js +7 -9
  197. package/dist/routing/router-prompt.js.map +1 -1
  198. package/dist/routing/router-types.d.ts +1 -0
  199. package/dist/routing/router.js +137 -22
  200. package/dist/routing/router.js.map +1 -1
  201. package/dist/routing/slot-resolver.d.ts +1 -1
  202. package/dist/routing/slot-resolver.js +2 -4
  203. package/dist/routing/slot-resolver.js.map +1 -1
  204. package/dist/routing/symbol-disambiguator.d.ts +11 -0
  205. package/dist/routing/symbol-disambiguator.js +52 -0
  206. package/dist/routing/symbol-disambiguator.js.map +1 -0
  207. package/dist/routing/turn-context.d.ts +1 -1
  208. package/dist/routing/turn-context.js +1 -1
  209. package/dist/routing/turn-context.js.map +1 -1
  210. package/dist/routing/types.d.ts +2 -0
  211. package/dist/runtime/answer-contracts.d.ts +1 -1
  212. package/dist/runtime/answer-contracts.js +48 -9
  213. package/dist/runtime/answer-contracts.js.map +1 -1
  214. package/dist/runtime/artifact-contracts.js.map +1 -1
  215. package/dist/runtime/planning-evidence.js +47 -26
  216. package/dist/runtime/planning-evidence.js.map +1 -1
  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 +13 -5
  224. package/dist/runtime/session-coordinator.js +160 -20
  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 +7 -5
  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 +10 -11
  243. package/dist/sentiment/index.js +10 -20
  244. package/dist/sentiment/index.js.map +1 -1
  245. package/dist/sentiment/insights.d.ts +17 -0
  246. package/dist/sentiment/insights.js +206 -0
  247. package/dist/sentiment/insights.js.map +1 -0
  248. package/dist/sentiment/keywords.js +26 -4
  249. package/dist/sentiment/keywords.js.map +1 -1
  250. package/dist/sentiment/pipeline.d.ts +2 -2
  251. package/dist/sentiment/pipeline.js +14 -2
  252. package/dist/sentiment/pipeline.js.map +1 -1
  253. package/dist/sentiment/scorer.d.ts +2 -0
  254. package/dist/sentiment/scorer.js +11 -2
  255. package/dist/sentiment/scorer.js.map +1 -1
  256. package/dist/sentiment/store.d.ts +1 -1
  257. package/dist/sentiment/store.js +1 -1
  258. package/dist/sentiment/store.js.map +1 -1
  259. package/dist/sentiment/trends.d.ts +1 -1
  260. package/dist/sentiment/trends.js.map +1 -1
  261. package/dist/sentiment/types.d.ts +2 -0
  262. package/dist/sentiment/types.js.map +1 -1
  263. package/dist/system-prompt.js +6 -9
  264. package/dist/system-prompt.js.map +1 -1
  265. package/dist/tool-kit.d.ts +7 -7
  266. package/dist/tool-kit.js +4 -4
  267. package/dist/tool-kit.js.map +1 -1
  268. package/dist/tools/fundamentals/company-overview.js +11 -6
  269. package/dist/tools/fundamentals/company-overview.js.map +1 -1
  270. package/dist/tools/fundamentals/comps.js +18 -9
  271. package/dist/tools/fundamentals/comps.js.map +1 -1
  272. package/dist/tools/fundamentals/dcf.js +23 -11
  273. package/dist/tools/fundamentals/dcf.js.map +1 -1
  274. package/dist/tools/fundamentals/earnings.js +8 -3
  275. package/dist/tools/fundamentals/earnings.js.map +1 -1
  276. package/dist/tools/fundamentals/financials.js +8 -3
  277. package/dist/tools/fundamentals/financials.js.map +1 -1
  278. package/dist/tools/fundamentals/sec-filings.js +21 -6
  279. package/dist/tools/fundamentals/sec-filings.js.map +1 -1
  280. package/dist/tools/index.d.ts +27 -20
  281. package/dist/tools/index.js +55 -43
  282. package/dist/tools/index.js.map +1 -1
  283. package/dist/tools/interaction/ask-user.js +15 -3
  284. package/dist/tools/interaction/ask-user.js.map +1 -1
  285. package/dist/tools/macro/fear-greed.js.map +1 -1
  286. package/dist/tools/macro/fred-data.d.ts +1 -1
  287. package/dist/tools/macro/fred-data.js +17 -6
  288. package/dist/tools/macro/fred-data.js.map +1 -1
  289. package/dist/tools/market/crypto-history.js +3 -1
  290. package/dist/tools/market/crypto-history.js.map +1 -1
  291. package/dist/tools/market/crypto-price.js +3 -1
  292. package/dist/tools/market/crypto-price.js.map +1 -1
  293. package/dist/tools/market/screen-stocks.d.ts +18 -0
  294. package/dist/tools/market/screen-stocks.js +252 -0
  295. package/dist/tools/market/screen-stocks.js.map +1 -0
  296. package/dist/tools/market/search-ticker.js +160 -8
  297. package/dist/tools/market/search-ticker.js.map +1 -1
  298. package/dist/tools/market/stock-history.d.ts +2 -2
  299. package/dist/tools/market/stock-history.js +26 -7
  300. package/dist/tools/market/stock-history.js.map +1 -1
  301. package/dist/tools/market/stock-quote.js +5 -3
  302. package/dist/tools/market/stock-quote.js.map +1 -1
  303. package/dist/tools/options/greeks.js +1 -1
  304. package/dist/tools/options/greeks.js.map +1 -1
  305. package/dist/tools/options/option-chain.js +19 -6
  306. package/dist/tools/options/option-chain.js.map +1 -1
  307. package/dist/tools/portfolio/alerts.d.ts +15 -0
  308. package/dist/tools/portfolio/alerts.js +357 -0
  309. package/dist/tools/portfolio/alerts.js.map +1 -0
  310. package/dist/tools/portfolio/correlation.d.ts +1 -1
  311. package/dist/tools/portfolio/correlation.js +33 -13
  312. package/dist/tools/portfolio/correlation.js.map +1 -1
  313. package/dist/tools/portfolio/daily-report.d.ts +8 -0
  314. package/dist/tools/portfolio/daily-report.js +83 -0
  315. package/dist/tools/portfolio/daily-report.js.map +1 -0
  316. package/dist/tools/portfolio/holdings-overlap.js +10 -3
  317. package/dist/tools/portfolio/holdings-overlap.js.map +1 -1
  318. package/dist/tools/portfolio/notifications.d.ts +7 -0
  319. package/dist/tools/portfolio/notifications.js +43 -0
  320. package/dist/tools/portfolio/notifications.js.map +1 -0
  321. package/dist/tools/portfolio/predictions.d.ts +12 -6
  322. package/dist/tools/portfolio/predictions.js +337 -87
  323. package/dist/tools/portfolio/predictions.js.map +1 -1
  324. package/dist/tools/portfolio/risk-analysis.d.ts +1 -1
  325. package/dist/tools/portfolio/risk-analysis.js +45 -6
  326. package/dist/tools/portfolio/risk-analysis.js.map +1 -1
  327. package/dist/tools/portfolio/tracker.d.ts +4 -3
  328. package/dist/tools/portfolio/tracker.js +246 -101
  329. package/dist/tools/portfolio/tracker.js.map +1 -1
  330. package/dist/tools/portfolio/watchlist.d.ts +6 -4
  331. package/dist/tools/portfolio/watchlist.js +208 -108
  332. package/dist/tools/portfolio/watchlist.js.map +1 -1
  333. package/dist/tools/sentiment/insight-format.d.ts +2 -0
  334. package/dist/tools/sentiment/insight-format.js +36 -0
  335. package/dist/tools/sentiment/insight-format.js.map +1 -0
  336. package/dist/tools/sentiment/query-match.d.ts +3 -0
  337. package/dist/tools/sentiment/query-match.js +113 -0
  338. package/dist/tools/sentiment/query-match.js.map +1 -0
  339. package/dist/tools/sentiment/reddit-sentiment.d.ts +12 -1
  340. package/dist/tools/sentiment/reddit-sentiment.js +266 -107
  341. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  342. package/dist/tools/sentiment/sentiment-summary.d.ts +9 -1
  343. package/dist/tools/sentiment/sentiment-summary.js +223 -205
  344. package/dist/tools/sentiment/sentiment-summary.js.map +1 -1
  345. package/dist/tools/sentiment/sentiment-trend.d.ts +1 -1
  346. package/dist/tools/sentiment/sentiment-trend.js +12 -2
  347. package/dist/tools/sentiment/sentiment-trend.js.map +1 -1
  348. package/dist/tools/sentiment/twitter-sentiment.d.ts +11 -1
  349. package/dist/tools/sentiment/twitter-sentiment.js +188 -58
  350. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
  351. package/dist/tools/sentiment/untrusted-text.d.ts +2 -0
  352. package/dist/tools/sentiment/untrusted-text.js +17 -0
  353. package/dist/tools/sentiment/untrusted-text.js.map +1 -0
  354. package/dist/tools/sentiment/web-search.js +9 -13
  355. package/dist/tools/sentiment/web-search.js.map +1 -1
  356. package/dist/tools/sentiment/web-sentiment.js +19 -3
  357. package/dist/tools/sentiment/web-sentiment.js.map +1 -1
  358. package/dist/tools/technical/backtest.d.ts +1 -1
  359. package/dist/tools/technical/backtest.js +27 -20
  360. package/dist/tools/technical/backtest.js.map +1 -1
  361. package/dist/tools/technical/indicators.js +23 -5
  362. package/dist/tools/technical/indicators.js.map +1 -1
  363. package/dist/types/index.d.ts +3 -3
  364. package/dist/types/index.js.map +1 -1
  365. package/dist/types/market.d.ts +1 -0
  366. package/dist/types/portfolio.d.ts +14 -4
  367. package/dist/types/sentiment.d.ts +52 -0
  368. package/dist/workflows/compare-assets.d.ts +0 -3
  369. package/dist/workflows/compare-assets.js +20 -11
  370. package/dist/workflows/compare-assets.js.map +1 -1
  371. package/dist/workflows/index.d.ts +3 -4
  372. package/dist/workflows/index.js +3 -3
  373. package/dist/workflows/index.js.map +1 -1
  374. package/dist/workflows/options-screener.d.ts +0 -3
  375. package/dist/workflows/options-screener.js +4 -11
  376. package/dist/workflows/options-screener.js.map +1 -1
  377. package/dist/workflows/portfolio-builder.d.ts +0 -3
  378. package/dist/workflows/portfolio-builder.js +0 -8
  379. package/dist/workflows/portfolio-builder.js.map +1 -1
  380. package/gui/server/ask-user-bridge.ts +1 -1
  381. package/gui/server/automation-heartbeat.ts +97 -0
  382. package/gui/server/background-quotes.ts +97 -1
  383. package/gui/server/chat-event-adapter.ts +32 -10
  384. package/gui/server/chat-run-session.ts +16 -0
  385. package/gui/server/invoke-tool.ts +160 -3
  386. package/gui/server/live-chat-event-adapter.ts +21 -6
  387. package/gui/server/market-state-api.ts +315 -0
  388. package/gui/server/model-setup.ts +156 -2
  389. package/gui/server/private-api-access.ts +62 -0
  390. package/gui/server/projector.ts +18 -9
  391. package/gui/server/prompt-observation.ts +4 -7
  392. package/gui/server/quote-snapshot-store.ts +50 -0
  393. package/gui/server/server.ts +218 -451
  394. package/gui/server/session-actions.ts +186 -1
  395. package/gui/server/shutdown.ts +47 -0
  396. package/gui/server/tool-invoke-ack.ts +49 -0
  397. package/gui/server/tool-metadata.ts +101 -24
  398. package/gui/server/websocket.ts +13 -3
  399. package/gui/server/writer-lock.ts +6 -2
  400. package/gui/server/ws-hub.ts +311 -0
  401. package/gui/shared/chat-events.ts +16 -1
  402. package/gui/shared/event-reducer.ts +24 -6
  403. package/gui/web/dist/assets/CatalogOverlay-CgeY5Pkp.js +1 -0
  404. package/gui/web/dist/assets/index-C6W_2eAn.js +69 -0
  405. package/gui/web/dist/assets/index-hwbx24a5.css +1 -0
  406. package/gui/web/dist/index.html +2 -2
  407. package/package.json +9 -6
  408. package/src/analysts/contracts.ts +10 -23
  409. package/src/analysts/orchestrator.ts +8 -43
  410. package/src/cli.ts +76 -12
  411. package/src/config.ts +44 -9
  412. package/src/index.ts +1 -1
  413. package/src/infra/cache.ts +41 -30
  414. package/src/infra/http-client.ts +72 -6
  415. package/src/infra/index.ts +6 -10
  416. package/src/infra/native-dependencies.ts +8 -3
  417. package/src/infra/node-version.ts +3 -1
  418. package/src/infra/opencandle-paths.ts +3 -14
  419. package/src/infra/rate-limiter.ts +22 -19
  420. package/src/market-state/alert-conditions.ts +82 -0
  421. package/src/market-state/alert-runner.ts +863 -0
  422. package/src/market-state/daily-report.ts +247 -0
  423. package/src/market-state/local-automation-service.ts +162 -0
  424. package/src/market-state/notification-delivery.ts +158 -0
  425. package/src/market-state/resolve-for-mutation.ts +24 -0
  426. package/src/market-state/resolve.ts +112 -0
  427. package/src/market-state/service.ts +2344 -0
  428. package/src/memory/index.ts +7 -7
  429. package/src/memory/manager.ts +14 -16
  430. package/src/memory/retrieval.ts +8 -7
  431. package/src/memory/sqlite.ts +407 -6
  432. package/src/memory/storage.ts +5 -15
  433. package/src/memory/tool-defaults.ts +60 -39
  434. package/src/memory/types.ts +3 -3
  435. package/src/monitor.ts +121 -0
  436. package/src/onboarding/connect.ts +24 -31
  437. package/src/onboarding/credential-interceptor.ts +3 -15
  438. package/src/onboarding/degradation-accumulator.ts +1 -3
  439. package/src/onboarding/provider-status.ts +410 -0
  440. package/src/onboarding/providers.ts +144 -45
  441. package/src/onboarding/state.ts +13 -15
  442. package/src/onboarding/tool-helpers.ts +2 -9
  443. package/src/onboarding/tool-tags.ts +51 -8
  444. package/src/onboarding/validation.ts +16 -22
  445. package/src/pi/opencandle-extension.ts +643 -101
  446. package/src/pi/session.ts +7 -5
  447. package/src/pi/setup.ts +61 -43
  448. package/src/pi/tool-adapter.ts +19 -6
  449. package/src/prompts/context-builder.ts +24 -13
  450. package/src/prompts/policy-cards.ts +3 -3
  451. package/src/prompts/sections.ts +1 -1
  452. package/src/prompts/symbol-preflight.ts +80 -0
  453. package/src/prompts/workflow-prompts.ts +77 -28
  454. package/src/providers/alpha-vantage.ts +58 -39
  455. package/src/providers/coingecko.ts +2 -5
  456. package/src/providers/errors.ts +9 -0
  457. package/src/providers/exa-search.ts +24 -22
  458. package/src/providers/external-tool-error.ts +20 -0
  459. package/src/providers/fear-greed.ts +1 -1
  460. package/src/providers/finnhub.ts +7 -6
  461. package/src/providers/fred.ts +3 -3
  462. package/src/providers/index.ts +14 -6
  463. package/src/providers/reddit-cli.ts +317 -0
  464. package/src/providers/reddit.ts +14 -59
  465. package/src/providers/sec-edgar.ts +20 -6
  466. package/src/providers/tradingview.ts +399 -0
  467. package/src/providers/twitter-cli.ts +233 -0
  468. package/src/providers/twitter.ts +8 -79
  469. package/src/providers/web-search.ts +30 -20
  470. package/src/providers/with-fallback.ts +8 -7
  471. package/src/providers/wrap-provider.ts +49 -10
  472. package/src/providers/yahoo-finance.ts +204 -66
  473. package/src/routing/classify-intent.ts +101 -10
  474. package/src/routing/defaults.ts +1 -1
  475. package/src/routing/entity-extractor.ts +287 -38
  476. package/src/routing/fund-symbols.ts +58 -0
  477. package/src/routing/horizon.ts +7 -0
  478. package/src/routing/index.ts +48 -48
  479. package/src/routing/planning.ts +145 -53
  480. package/src/routing/route-manifest.ts +37 -15
  481. package/src/routing/router-llm-client.ts +4 -4
  482. package/src/routing/router-prompt.ts +15 -19
  483. package/src/routing/router-types.ts +2 -5
  484. package/src/routing/router.ts +251 -53
  485. package/src/routing/slot-resolver.ts +34 -11
  486. package/src/routing/symbol-disambiguator.ts +72 -0
  487. package/src/routing/turn-context.ts +6 -9
  488. package/src/routing/types.ts +2 -0
  489. package/src/runtime/answer-contracts.ts +105 -45
  490. package/src/runtime/artifact-contracts.ts +2 -1
  491. package/src/runtime/planning-evidence.ts +157 -66
  492. package/src/runtime/prompt-step.ts +1 -16
  493. package/src/runtime/run-context.ts +12 -2
  494. package/src/runtime/session-coordinator.ts +238 -63
  495. package/src/runtime/session-title.ts +60 -0
  496. package/src/runtime/tool-defaults-wrapper.ts +13 -5
  497. package/src/runtime/validation.ts +1 -4
  498. package/src/runtime/workflow-events.ts +7 -7
  499. package/src/runtime/workflow-runner.ts +5 -11
  500. package/src/sentiment/adapters/finnhub.ts +7 -2
  501. package/src/sentiment/adapters/reddit.ts +2 -2
  502. package/src/sentiment/adapters/twitter.ts +1 -1
  503. package/src/sentiment/adapters/web.ts +1 -1
  504. package/src/sentiment/index.ts +17 -26
  505. package/src/sentiment/insights.ts +269 -0
  506. package/src/sentiment/keywords.ts +26 -4
  507. package/src/sentiment/pipeline.ts +28 -5
  508. package/src/sentiment/scorer.ts +13 -2
  509. package/src/sentiment/store.ts +2 -2
  510. package/src/sentiment/trends.ts +9 -3
  511. package/src/sentiment/types.ts +8 -4
  512. package/src/system-prompt.ts +6 -9
  513. package/src/tool-kit.ts +10 -9
  514. package/src/tools/fundamentals/company-overview.ts +19 -9
  515. package/src/tools/fundamentals/comps.ts +68 -55
  516. package/src/tools/fundamentals/dcf.ts +145 -95
  517. package/src/tools/fundamentals/earnings.ts +16 -6
  518. package/src/tools/fundamentals/financials.ts +16 -7
  519. package/src/tools/fundamentals/sec-filings.ts +37 -16
  520. package/src/tools/index.ts +56 -43
  521. package/src/tools/interaction/ask-user.ts +22 -10
  522. package/src/tools/macro/fear-greed.ts +1 -1
  523. package/src/tools/macro/fred-data.ts +58 -46
  524. package/src/tools/market/crypto-history.ts +8 -3
  525. package/src/tools/market/crypto-price.ts +6 -6
  526. package/src/tools/market/screen-stocks.ts +279 -0
  527. package/src/tools/market/search-ticker.ts +218 -17
  528. package/src/tools/market/stock-history.ts +37 -12
  529. package/src/tools/market/stock-quote.ts +10 -7
  530. package/src/tools/options/greeks.ts +5 -5
  531. package/src/tools/options/option-chain.ts +41 -17
  532. package/src/tools/portfolio/alerts.ts +457 -0
  533. package/src/tools/portfolio/correlation.ts +47 -20
  534. package/src/tools/portfolio/daily-report.ts +101 -0
  535. package/src/tools/portfolio/holdings-overlap.ts +31 -15
  536. package/src/tools/portfolio/notifications.ts +45 -0
  537. package/src/tools/portfolio/predictions.ts +406 -106
  538. package/src/tools/portfolio/risk-analysis.ts +46 -7
  539. package/src/tools/portfolio/tracker.ts +270 -109
  540. package/src/tools/portfolio/watchlist.ts +250 -121
  541. package/src/tools/sentiment/insight-format.ts +50 -0
  542. package/src/tools/sentiment/query-match.ts +117 -0
  543. package/src/tools/sentiment/reddit-sentiment.ts +360 -121
  544. package/src/tools/sentiment/sentiment-summary.ts +302 -235
  545. package/src/tools/sentiment/sentiment-trend.ts +24 -7
  546. package/src/tools/sentiment/twitter-sentiment.ts +264 -73
  547. package/src/tools/sentiment/untrusted-text.ts +21 -0
  548. package/src/tools/sentiment/web-search.ts +21 -18
  549. package/src/tools/sentiment/web-sentiment.ts +30 -10
  550. package/src/tools/technical/backtest.ts +32 -22
  551. package/src/tools/technical/indicators.ts +39 -14
  552. package/src/types/index.ts +8 -3
  553. package/src/types/market.ts +1 -0
  554. package/src/types/portfolio.ts +14 -4
  555. package/src/types/sentiment.ts +61 -2
  556. package/src/workflows/compare-assets.ts +33 -21
  557. package/src/workflows/index.ts +3 -4
  558. package/src/workflows/options-screener.ts +27 -29
  559. package/src/workflows/portfolio-builder.ts +34 -27
  560. package/dist/infra/browser.d.ts +0 -35
  561. package/dist/infra/browser.js +0 -103
  562. package/dist/infra/browser.js.map +0 -1
  563. package/dist/tools/interaction/twitter-login.d.ts +0 -8
  564. package/dist/tools/interaction/twitter-login.js +0 -77
  565. package/dist/tools/interaction/twitter-login.js.map +0 -1
  566. package/dist/workflows/types.d.ts +0 -4
  567. package/dist/workflows/types.js +0 -2
  568. package/dist/workflows/types.js.map +0 -1
  569. package/gui/web/dist/assets/CatalogOverlay-Bmp6Knu7.js +0 -1
  570. package/gui/web/dist/assets/index-Bxt9QpLX.css +0 -1
  571. package/gui/web/dist/assets/index-CZ9DHZYy.js +0 -67
  572. package/src/infra/browser.ts +0 -111
  573. package/src/tools/interaction/twitter-login.ts +0 -93
  574. package/src/workflows/types.ts +0 -4
@@ -1,50 +1,53 @@
1
- import { Type } from "@sinclair/typebox";
2
1
  import type { AgentTool } from "@earendil-works/pi-agent-core";
3
- import { readFileSync, writeFileSync, existsSync } from "node:fs";
4
- import { getQuote } from "../../providers/yahoo-finance.js";
2
+ import { Type } from "@sinclair/typebox";
3
+ import { isZeroFilledQuote } from "../../market-state/resolve.js";
4
+ import { resolveInstrumentForMutation } from "../../market-state/resolve-for-mutation.js";
5
+ import { MarketStateService, type WatchlistItemRecord } from "../../market-state/service.js";
6
+ import { initDefaultDatabase } from "../../memory/sqlite.js";
7
+ import { getQuotes, type TradingViewQuote } from "../../providers/tradingview.js";
5
8
  import { wrapProvider } from "../../providers/wrap-provider.js";
6
- import { ensureParentDir, getWatchlistPath } from "../../infra/opencandle-paths.js";
7
-
8
- interface WatchlistItem {
9
- symbol: string;
10
- addedAt: string;
11
- targetPrice?: number;
12
- stopPrice?: number;
13
- notes?: string;
14
- }
15
-
16
- function loadWatchlist(): WatchlistItem[] {
17
- const watchlistPath = getWatchlistPath();
18
- if (!existsSync(watchlistPath)) return [];
19
- try {
20
- return JSON.parse(readFileSync(watchlistPath, "utf-8"));
21
- } catch {
22
- return [];
23
- }
24
- }
9
+ import { getQuote } from "../../providers/yahoo-finance.js";
25
10
 
26
- function saveWatchlist(items: WatchlistItem[]): void {
27
- const watchlistPath = getWatchlistPath();
28
- ensureParentDir(watchlistPath);
29
- writeFileSync(watchlistPath, JSON.stringify(items, null, 2));
11
+ interface WatchlistCheck extends WatchlistItemRecord {
12
+ currentPrice: number | null;
13
+ alerts: string[];
14
+ statuses: string[];
15
+ sourceProvider?: "tradingview" | "yahoo";
16
+ dataCaveat?: string;
30
17
  }
31
18
 
32
19
  const params = Type.Object({
33
20
  action: Type.Union(
34
- [Type.Literal("add"), Type.Literal("remove"), Type.Literal("check")],
35
- { description: "One of: 'add', 'remove', or 'check'" },
36
- ),
37
- symbol: Type.Optional(
38
- Type.String({ description: "Ticker symbol (required for add/remove)" }),
21
+ [Type.Literal("add"), Type.Literal("update"), Type.Literal("remove"), Type.Literal("check")],
22
+ { description: "One of: 'add', 'update', 'remove', or 'check'" },
39
23
  ),
24
+ symbol: Type.Optional(Type.String({ description: "Ticker symbol (required for add/remove)" })),
40
25
  target_price: Type.Optional(
41
- Type.Number({ description: "Alert when price rises above this level" }),
26
+ Type.Union([
27
+ Type.Number({ description: "Alert when price rises above this level" }),
28
+ Type.Null({ description: "Clear the existing target price during update" }),
29
+ ]),
42
30
  ),
43
31
  stop_price: Type.Optional(
44
- Type.Number({ description: "Alert when price falls below this level" }),
32
+ Type.Union([
33
+ Type.Number({ description: "Alert when price falls below this level" }),
34
+ Type.Null({ description: "Clear the existing stop price during update" }),
35
+ ]),
45
36
  ),
46
37
  notes: Type.Optional(
47
- Type.String({ description: "Optional notes for why you're watching this" }),
38
+ Type.Union([
39
+ Type.String({ description: "Optional notes for why you're watching this" }),
40
+ Type.Null({ description: "Clear existing notes during update" }),
41
+ ]),
42
+ ),
43
+ thesis: Type.Optional(
44
+ Type.Union([
45
+ Type.String({ description: "Optional thesis for why you're watching this" }),
46
+ Type.Null({ description: "Clear the existing thesis during update" }),
47
+ ]),
48
+ ),
49
+ tags: Type.Optional(
50
+ Type.Array(Type.String(), { description: "Optional tags for organizing the watchlist item" }),
48
51
  ),
49
52
  });
50
53
 
@@ -52,108 +55,234 @@ export const watchlistTool: AgentTool<typeof params> = {
52
55
  name: "manage_watchlist",
53
56
  label: "Watchlist",
54
57
  description:
55
- "Manage your watchlist of stocks and crypto. Add symbols with optional target and stop prices, remove symbols, or check current prices against your alert levels. Data persisted to ~/.opencandle/watchlist.json.",
58
+ "Manage your watchlist of stocks and crypto. Add symbols with optional target and stop prices, remove symbols, or check current prices against your watchlist levels.",
56
59
  parameters: params,
57
60
  async execute(_toolCallId, args) {
58
- const items = loadWatchlist();
61
+ const db = initDefaultDatabase();
62
+ const service = new MarketStateService(db);
59
63
 
60
- if (args.action === "add") {
61
- if (!args.symbol) {
62
- throw new Error("symbol is required for add action.");
64
+ try {
65
+ if (args.action === "add") {
66
+ if (!args.symbol) {
67
+ throw new Error("symbol is required for add action.");
68
+ }
69
+ const instrument = await resolveInstrumentForMutation(args.symbol);
70
+ if (instrument.status === "needs_selection") {
71
+ return {
72
+ content: [
73
+ {
74
+ type: "text",
75
+ text: `Could not verify ${instrument.query}. Choose one of the returned candidates before adding it to the watchlist.`,
76
+ },
77
+ ],
78
+ details: instrument,
79
+ };
80
+ }
81
+ const item = service.addWatchlistItem({
82
+ instrument: instrument.instrument,
83
+ targetPrice: args.target_price,
84
+ stopPrice: args.stop_price,
85
+ thesis: args.thesis,
86
+ notes: args.notes,
87
+ tags: args.tags,
88
+ });
89
+ const alerts = [];
90
+ if (args.target_price) alerts.push(`target: $${args.target_price}`);
91
+ if (args.stop_price) alerts.push(`stop: $${args.stop_price}`);
92
+ const alertStr = alerts.length > 0 ? ` (${alerts.join(", ")})` : "";
93
+ return {
94
+ content: [{ type: "text", text: `Added ${item.symbol} to watchlist${alertStr}` }],
95
+ details: item,
96
+ };
63
97
  }
64
- const symbol = args.symbol.toUpperCase();
65
- const existing = items.findIndex((i) => i.symbol === symbol);
66
- const item: WatchlistItem = {
67
- symbol,
68
- addedAt: new Date().toISOString(),
69
- ...(args.target_price != null && { targetPrice: args.target_price }),
70
- ...(args.stop_price != null && { stopPrice: args.stop_price }),
71
- ...(args.notes != null && { notes: args.notes }),
72
- };
73
- if (existing >= 0) {
74
- items[existing] = item;
75
- } else {
76
- items.push(item);
98
+
99
+ if (args.action === "remove") {
100
+ if (!args.symbol) {
101
+ throw new Error("symbol is required for remove action.");
102
+ }
103
+ const symbol = args.symbol.toUpperCase();
104
+ if (!service.removeWatchlistItemBySymbol(symbol)) {
105
+ return {
106
+ content: [{ type: "text", text: `${symbol} not found in watchlist` }],
107
+ details: null,
108
+ };
109
+ }
110
+ return {
111
+ content: [{ type: "text", text: `Removed ${symbol} from watchlist` }],
112
+ details: null,
113
+ };
77
114
  }
78
- saveWatchlist(items);
79
- const alerts = [];
80
- if (args.target_price) alerts.push(`target: $${args.target_price}`);
81
- if (args.stop_price) alerts.push(`stop: $${args.stop_price}`);
82
- const alertStr = alerts.length > 0 ? ` (${alerts.join(", ")})` : "";
83
- return {
84
- content: [{ type: "text", text: `Added ${symbol} to watchlist${alertStr}` }],
85
- details: null,
86
- };
87
- }
88
115
 
89
- if (args.action === "remove") {
90
- if (!args.symbol) {
91
- throw new Error("symbol is required for remove action.");
116
+ if (args.action === "update") {
117
+ if (!args.symbol) {
118
+ throw new Error("symbol is required for update action.");
119
+ }
120
+ const symbol = args.symbol.toUpperCase();
121
+ const item = service.updateWatchlistItemBySymbol(symbol, {
122
+ targetPrice: args.target_price,
123
+ stopPrice: args.stop_price,
124
+ notes: args.notes,
125
+ thesis: args.thesis,
126
+ tags: args.tags,
127
+ });
128
+ if (item == null) {
129
+ return {
130
+ content: [{ type: "text", text: `${symbol} not found in watchlist` }],
131
+ details: null,
132
+ };
133
+ }
134
+ return {
135
+ content: [{ type: "text", text: `Updated ${item.symbol} in watchlist` }],
136
+ details: item,
137
+ };
92
138
  }
93
- const symbol = args.symbol.toUpperCase();
94
- const idx = items.findIndex((i) => i.symbol === symbol);
95
- if (idx === -1) {
139
+
140
+ const items = service.listWatchlistItems();
141
+ if (items.length === 0) {
96
142
  return {
97
- content: [{ type: "text", text: `${symbol} not found in watchlist` }],
143
+ content: [{ type: "text", text: "Watchlist is empty. Use add action to add symbols." }],
98
144
  details: null,
99
145
  };
100
146
  }
101
- items.splice(idx, 1);
102
- saveWatchlist(items);
103
- return {
104
- content: [{ type: "text", text: `Removed ${symbol} from watchlist` }],
105
- details: null,
106
- };
107
- }
108
147
 
109
- // Check action
110
- if (items.length === 0) {
148
+ const checks = await checkWatchlistPrices(items);
149
+ const alertItems = checks.filter((c) => c.alerts.length > 0);
150
+ const lines = [
151
+ `**Watchlist** — ${items.length} symbols${alertItems.length > 0 ? ` | ${alertItems.length} ALERT(S)` : ""}`,
152
+ "",
153
+ ];
154
+
155
+ const tradingViewCaveats = Array.from(
156
+ new Set(
157
+ checks
158
+ .filter((c) => c.sourceProvider === "tradingview")
159
+ .map(
160
+ (c) =>
161
+ c.dataCaveat ??
162
+ "TradingView scanner data may be delayed about 15 minutes and comes from an unofficial endpoint.",
163
+ ),
164
+ ),
165
+ );
166
+ if (tradingViewCaveats.length > 0) {
167
+ lines.push(...tradingViewCaveats.map((caveat) => `Data caveat: ${caveat}`), "");
168
+ }
169
+
170
+ for (const c of checks) {
171
+ const alertStr = c.alerts.length > 0 ? ` ** ${c.alerts.join(" | ")} **` : "";
172
+ const statusStr = c.statuses.length > 0 ? ` | ${c.statuses.join(" | ")}` : "";
173
+ const targetStr = c.targetPrice ? ` | Target: $${c.targetPrice}` : "";
174
+ const stopStr = c.stopPrice ? ` | Stop: $${c.stopPrice}` : "";
175
+ const sourceStr = c.sourceProvider
176
+ ? ` | Source: ${c.sourceProvider === "tradingview" ? "TradingView" : "Yahoo"}`
177
+ : "";
178
+ const priceStr =
179
+ typeof c.currentPrice === "number" ? `$${c.currentPrice.toFixed(2)}` : "Unavailable";
180
+ lines.push(
181
+ ` ${c.symbol}: ${priceStr}${targetStr}${stopStr}${sourceStr}${statusStr}${alertStr}`,
182
+ );
183
+ }
184
+
111
185
  return {
112
- content: [{ type: "text", text: "Watchlist is empty. Use add action to add symbols." }],
113
- details: null,
186
+ content: [{ type: "text", text: lines.join("\n") }],
187
+ details: { items: checks },
114
188
  };
189
+ } finally {
190
+ db.close();
115
191
  }
192
+ },
193
+ };
116
194
 
117
- const checks = await Promise.all(
118
- items.map(async (item) => {
119
- const result = await wrapProvider("yahoo", () => getQuote(item.symbol));
120
- if (result.status === "unavailable") {
121
- return { ...item, currentPrice: 0, alerts: [`UNAVAILABLE: ${result.reason}`], statuses: [] };
122
- }
123
- const quote = result.data;
124
- const alerts: string[] = [];
125
- const statuses: string[] = [];
126
- if (item.targetPrice && quote.price >= item.targetPrice) {
127
- alerts.push(`TARGET HIT: $${quote.price.toFixed(2)} >= $${item.targetPrice}`);
128
- } else if (item.targetPrice) {
129
- statuses.push(`Target pending: $${quote.price.toFixed(2)} < $${item.targetPrice}`);
130
- }
131
- if (item.stopPrice && quote.price <= item.stopPrice) {
132
- alerts.push(`STOP ALERT: $${quote.price.toFixed(2)} fell below $${item.stopPrice}`);
133
- } else if (item.stopPrice) {
134
- statuses.push(`Stop OK: $${quote.price.toFixed(2)} > $${item.stopPrice}`);
135
- }
136
- return { ...item, currentPrice: quote.price, alerts, statuses };
137
- }),
138
- );
139
-
140
- const alertItems = checks.filter((c) => c.alerts.length > 0);
141
- const lines = [
142
- `**Watchlist** — ${items.length} symbols${alertItems.length > 0 ? ` | ${alertItems.length} ALERT(S)` : ""}`,
143
- "",
144
- ];
145
-
146
- for (const c of checks) {
147
- const alertStr = c.alerts.length > 0 ? ` ** ${c.alerts.join(" | ")} **` : "";
148
- const statusStr = c.statuses.length > 0 ? ` | ${c.statuses.join(" | ")}` : "";
149
- const targetStr = c.targetPrice ? ` | Target: $${c.targetPrice}` : "";
150
- const stopStr = c.stopPrice ? ` | Stop: $${c.stopPrice}` : "";
151
- lines.push(` ${c.symbol}: $${c.currentPrice.toFixed(2)}${targetStr}${stopStr}${statusStr}${alertStr}`);
195
+ async function checkWatchlistPrices(items: WatchlistItemRecord[]): Promise<WatchlistCheck[]> {
196
+ const tradingViewSymbols = items
197
+ .map((item) => item.symbol)
198
+ .filter((symbol) => !shouldSkipTradingView(symbol));
199
+ const tradingViewQuotes = new Map<string, TradingViewQuote>();
200
+
201
+ if (tradingViewSymbols.length > 0) {
202
+ const result = await wrapProvider("tradingview", () => getQuotes(tradingViewSymbols));
203
+ if (result.status === "ok" && result.data.length > 0) {
204
+ for (const quote of result.data) {
205
+ tradingViewQuotes.set(quote.requestedSymbol.toUpperCase(), {
206
+ ...quote,
207
+ dataCaveat: result.stale
208
+ ? `cached TradingView data from ${result.timestamp}; ${quote.dataCaveat}`
209
+ : quote.dataCaveat,
210
+ });
211
+ }
152
212
  }
213
+ }
153
214
 
154
- return {
155
- content: [{ type: "text", text: lines.join("\n") }],
156
- details: { items: checks },
157
- };
158
- },
159
- };
215
+ return Promise.all(
216
+ items.map(async (item) => {
217
+ const tradingViewQuote = tradingViewQuotes.get(item.symbol.toUpperCase());
218
+ if (tradingViewQuote) {
219
+ return buildCheckResult(
220
+ item,
221
+ tradingViewQuote.price,
222
+ "tradingview",
223
+ tradingViewQuote.dataCaveat,
224
+ );
225
+ }
226
+
227
+ const result = await wrapProvider("yahoo", () => getQuote(item.symbol));
228
+ if (result.status === "unavailable") {
229
+ return {
230
+ ...item,
231
+ currentPrice: null,
232
+ alerts: [`UNAVAILABLE: ${result.reason}`],
233
+ statuses: [],
234
+ };
235
+ }
236
+ if (result.stale) {
237
+ return {
238
+ ...item,
239
+ currentPrice: null,
240
+ alerts: ["UNAVAILABLE: provider returned stale market data"],
241
+ statuses: [],
242
+ };
243
+ }
244
+ const quote = result.data;
245
+ if (isZeroFilledQuote(quote)) {
246
+ return {
247
+ ...item,
248
+ currentPrice: null,
249
+ alerts: ["UNAVAILABLE: Yahoo returned no valid market data."],
250
+ statuses: [],
251
+ };
252
+ }
253
+ return buildCheckResult(item, quote.price, "yahoo");
254
+ }),
255
+ );
256
+ }
257
+
258
+ function buildCheckResult(
259
+ item: WatchlistItemRecord,
260
+ price: number,
261
+ sourceProvider: "tradingview" | "yahoo",
262
+ dataCaveat?: string,
263
+ ): WatchlistCheck {
264
+ const alerts: string[] = [];
265
+ const statuses: string[] = [];
266
+ if (item.targetPrice && price >= item.targetPrice) {
267
+ alerts.push(`TARGET HIT: $${price.toFixed(2)} >= $${item.targetPrice}`);
268
+ } else if (item.targetPrice) {
269
+ statuses.push(`Target pending: $${price.toFixed(2)} < $${item.targetPrice}`);
270
+ }
271
+ if (item.stopPrice && price <= item.stopPrice) {
272
+ alerts.push(`STOP ALERT: $${price.toFixed(2)} fell below $${item.stopPrice}`);
273
+ } else if (item.stopPrice) {
274
+ statuses.push(`Stop OK: $${price.toFixed(2)} > $${item.stopPrice}`);
275
+ }
276
+ return {
277
+ ...item,
278
+ currentPrice: price,
279
+ alerts,
280
+ statuses,
281
+ sourceProvider,
282
+ dataCaveat,
283
+ };
284
+ }
285
+
286
+ function shouldSkipTradingView(symbol: string): boolean {
287
+ return /(?:-USD|\.(?:TO|DE|T|L|HK))$/i.test(symbol.trim());
288
+ }
@@ -0,0 +1,50 @@
1
+ import type { SentimentInsight } from "../../types/sentiment.js";
2
+ import { renderUntrustedText } from "./untrusted-text.js";
3
+
4
+ export function formatInsightSection(insight: SentimentInsight): string[] {
5
+ const lines: string[] = [];
6
+ lines.push("");
7
+ lines.push("Findings:");
8
+ lines.push(
9
+ `- Overall: ${insight.label} (${formatSigned(insight.score)}) from ${insight.sampleSize} records; ${insight.scoredSampleSize} had keyword sentiment evidence.`,
10
+ );
11
+ lines.push(
12
+ `- Confidence: ${insight.confidence.level} (${insight.confidence.score.toFixed(2)})${
13
+ insight.confidence.reasons.length > 0 ? ` — ${insight.confidence.reasons.join("; ")}` : ""
14
+ }`,
15
+ );
16
+ appendDrivers(lines, "Positive drivers", insight.positiveDrivers);
17
+ appendDrivers(lines, "Negative drivers", insight.negativeDrivers);
18
+ appendDrivers(lines, "Mixed drivers", insight.mixedDrivers);
19
+ if (insight.caveats.length > 0) {
20
+ lines.push(`- Caveats: ${insight.caveats.join("; ")}`);
21
+ }
22
+ if (insight.notableClaims.length > 0) {
23
+ lines.push("- Notable source claims:");
24
+ for (const claim of insight.notableClaims.slice(0, 3)) {
25
+ lines.push(` - ${renderUntrustedText(claim, 140)}`);
26
+ }
27
+ }
28
+ if (insight.representativeItems.length > 0) {
29
+ lines.push(
30
+ `- Representative evidence preview: ${insight.representativeItems.length} shown from ${insight.scoredSampleSize} scored records (${insight.sampleSize} total records).`,
31
+ );
32
+ }
33
+ return lines;
34
+ }
35
+
36
+ function appendDrivers(
37
+ lines: string[],
38
+ label: string,
39
+ drivers: SentimentInsight["positiveDrivers"],
40
+ ): void {
41
+ if (drivers.length === 0) return;
42
+ const rendered = drivers
43
+ .map((driver) => `${renderUntrustedText(driver.label, 60)} (${driver.count})`)
44
+ .join(", ");
45
+ lines.push(`- ${label}: ${rendered}`);
46
+ }
47
+
48
+ function formatSigned(value: number): string {
49
+ return `${value >= 0 ? "+" : ""}${value.toFixed(2)}`;
50
+ }
@@ -0,0 +1,117 @@
1
+ import { extractEntities } from "../../routing/entity-extractor.js";
2
+ import type { SentinelRecord } from "../../sentiment/types.js";
3
+
4
+ const STOP_TERMS = new Set([
5
+ "about",
6
+ "around",
7
+ "are",
8
+ "from",
9
+ "how",
10
+ "is",
11
+ "latest",
12
+ "mood",
13
+ "now",
14
+ "on",
15
+ "or",
16
+ "people",
17
+ "reddit",
18
+ "retail",
19
+ "right",
20
+ "say",
21
+ "saying",
22
+ "says",
23
+ "sentiment",
24
+ "social",
25
+ "stock",
26
+ "stocks",
27
+ "talking",
28
+ "the",
29
+ "think",
30
+ "thinking",
31
+ "to",
32
+ "tweet",
33
+ "tweets",
34
+ "twitter",
35
+ "what",
36
+ "with",
37
+ ]);
38
+
39
+ export function sentimentQueryTerms(query: string): string[] {
40
+ const hasSp500 = /\bs\s*&\s*p\s*500\b/i.test(query) || /\bspx\b/i.test(query);
41
+
42
+ const symbols = hasSp500
43
+ ? ["sp500"]
44
+ : extractEntities(query).symbols.map((symbol) => symbol.toLowerCase());
45
+ const punctuationTerms = [...query.toLowerCase().matchAll(/\b([a-z])\s*[&/]\s*([a-z])\b/g)].map(
46
+ (match) => `${match[1]}${match[2]}`,
47
+ );
48
+ const terms = query.toLowerCase().match(/[a-z0-9]{2,}/g);
49
+ if (!terms) {
50
+ if (punctuationTerms.length > 0) return [...new Set(punctuationTerms)];
51
+ const compact = query.toLowerCase().replace(/[^a-z0-9]/g, "");
52
+ if (compact.length >= 2) return [compact];
53
+ return query.trim().length > 0 ? [query.trim().toLowerCase()] : [];
54
+ }
55
+
56
+ const stopTerms = hasSp500 ? new Set([...STOP_TERMS, "sp", "spx", "500"]) : STOP_TERMS;
57
+ const filtered = [
58
+ ...new Set([...punctuationTerms, ...terms.filter((term) => !stopTerms.has(term))]),
59
+ ].slice(0, 6);
60
+ if (symbols.length > 0) {
61
+ return [
62
+ ...new Set([
63
+ ...symbols,
64
+ ...filtered.filter((term) => !symbols.some((symbol) => symbol.startsWith(term))),
65
+ ]),
66
+ ].slice(0, 6);
67
+ }
68
+ if (filtered.length > 0) return filtered;
69
+ return query.trim().length > 0 ? [query.trim().toLowerCase()] : [];
70
+ }
71
+
72
+ export function recordMatchesSentimentQuery(
73
+ record: SentinelRecord,
74
+ terms: readonly string[],
75
+ ): boolean {
76
+ if (terms.length === 0) return true;
77
+ const text = `${record.title ?? ""}\n${record.text}`.toLowerCase();
78
+ const textTokens = new Set(text.match(/[a-z0-9]+/g) ?? []);
79
+ const metadataTokens = new Set<string>();
80
+ const subreddit = record.metadata.subreddit;
81
+ if (typeof subreddit === "string") {
82
+ metadataTokens.add(subreddit.toLowerCase());
83
+ metadataTokens.add(subreddit.toLowerCase().replace(/^r\//, ""));
84
+ }
85
+ if (/\bs\s*&\s*p\s*500\b/i.test(text) || /\bspx\b/i.test(text)) textTokens.add("sp500");
86
+ for (const match of text.matchAll(/\b([a-z])\s*[&/]\s*([a-z])\b/g)) {
87
+ textTokens.add(`${match[1]}${match[2]}`);
88
+ }
89
+ for (const match of text.matchAll(/\b([a-z]{1,5})[.-]([a-z])\b/g)) {
90
+ textTokens.add(`${match[1]}${match[2]}`);
91
+ }
92
+ const tickerTokens = new Set<string>();
93
+ for (const ticker of record.sentiment.tickers) {
94
+ const normalized = ticker.toLowerCase();
95
+ tickerTokens.add(normalized);
96
+ tickerTokens.add(normalized.replace(/[^a-z0-9]/g, ""));
97
+ }
98
+ const matchedTerms = terms.filter(
99
+ (term) => tickerTokens.has(term) || textTokens.has(term) || metadataTokens.has(term),
100
+ );
101
+ for (const term of terms) {
102
+ if (matchedTerms.includes(term) || !term.includes(" ")) continue;
103
+ if (text.includes(term)) matchedTerms.push(term);
104
+ }
105
+ for (const term of terms) {
106
+ if (matchedTerms.includes(term) || term.length < 4) continue;
107
+ if (textTokens.has(`${term}s`)) {
108
+ matchedTerms.push(term);
109
+ continue;
110
+ }
111
+ if (term.endsWith("s") && textTokens.has(term.slice(0, -1))) {
112
+ matchedTerms.push(term);
113
+ }
114
+ }
115
+ const requiredMatches = terms.length > 1 ? 2 : 1;
116
+ return matchedTerms.length >= requiredMatches;
117
+ }