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
@@ -0,0 +1,399 @@
1
+ import { cache, STALE_LIMIT, TTL } from "../infra/cache.js";
2
+ import { httpPost } from "../infra/http-client.js";
3
+ import { rateLimiter } from "../infra/rate-limiter.js";
4
+
5
+ const BASE_URL = "https://scanner.tradingview.com";
6
+ const DATA_CAVEAT =
7
+ "TradingView scanner data may be delayed about 15 minutes and comes from an unofficial endpoint.";
8
+ const BROWSER_UA =
9
+ "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/131.0.0.0 Safari/537.36";
10
+
11
+ const QUOTE_COLUMNS = [
12
+ "name",
13
+ "close",
14
+ "change",
15
+ "change_abs",
16
+ "volume",
17
+ "exchange",
18
+ "market",
19
+ "description",
20
+ "type",
21
+ "typespecs",
22
+ ] as const;
23
+
24
+ export const DEFAULT_COLUMNS = [
25
+ "name",
26
+ "close",
27
+ "change",
28
+ "volume",
29
+ "market_cap_basic",
30
+ "price_earnings_ttm",
31
+ ] as const;
32
+
33
+ export type ScreenFilterOp =
34
+ | "greater"
35
+ | "egreater"
36
+ | "less"
37
+ | "eless"
38
+ | "equal"
39
+ | "nequal"
40
+ | "in_range"
41
+ | "not_in_range"
42
+ | "crosses"
43
+ | "crosses_above"
44
+ | "crosses_below"
45
+ | "above%"
46
+ | "below%"
47
+ | "match"
48
+ | "nmatch"
49
+ | "has"
50
+ | "has_none_of"
51
+ | "empty"
52
+ | "nempty";
53
+
54
+ export interface ScreenFilterClause {
55
+ field: string;
56
+ op: ScreenFilterOp;
57
+ value?: unknown;
58
+ }
59
+
60
+ export interface ScreenSort {
61
+ field: string;
62
+ direction?: "asc" | "desc";
63
+ }
64
+
65
+ export interface ScreenStocksOpts {
66
+ market?: string;
67
+ columns?: string[];
68
+ filter?: ScreenFilterClause[];
69
+ sort?: ScreenSort;
70
+ limit?: number;
71
+ }
72
+
73
+ export interface ScreenerRow {
74
+ tvSymbol: string;
75
+ symbol: string;
76
+ values: Record<string, unknown | null>;
77
+ sourceProvider: "tradingview";
78
+ dataCaveat: string;
79
+ }
80
+
81
+ export interface TradingViewQuote {
82
+ requestedSymbol: string;
83
+ tvSymbol: string;
84
+ symbol: string;
85
+ price: number;
86
+ change: number;
87
+ changePercent: number;
88
+ volume: number;
89
+ exchange?: string;
90
+ market?: string;
91
+ name?: string;
92
+ type?: string;
93
+ typespecs?: unknown;
94
+ sourceProvider: "tradingview";
95
+ dataCaveat: string;
96
+ }
97
+
98
+ interface TradingViewResponse {
99
+ fields?: string[];
100
+ symbols?: Array<{
101
+ s: string;
102
+ f: unknown[];
103
+ }>;
104
+ }
105
+
106
+ interface ScannerBody {
107
+ markets?: string[];
108
+ symbols?: { tickers?: string[]; query?: { types: string[] } };
109
+ columns: readonly string[];
110
+ filter?: Array<{ left: string; operation: string; right?: unknown }>;
111
+ sort?: { sortBy: string; sortOrder: "asc" | "desc" };
112
+ range: [number, number];
113
+ options: { lang: "en" };
114
+ }
115
+
116
+ interface DecodedRow {
117
+ tvSymbol: string;
118
+ symbol: string;
119
+ values: Record<string, unknown | null>;
120
+ }
121
+
122
+ export function buildTvSymbol(exchange: string, ticker: string): string {
123
+ const trimmedTicker = ticker.trim().toUpperCase();
124
+ if (trimmedTicker.includes(":")) return trimmedTicker;
125
+ const trimmedExchange = exchange.trim().toUpperCase();
126
+ return trimmedExchange ? `${trimmedExchange}:${trimmedTicker}` : trimmedTicker;
127
+ }
128
+
129
+ export async function screenStocks(opts: ScreenStocksOpts = {}): Promise<ScreenerRow[]> {
130
+ const market = opts.market?.trim().toLowerCase() || "america";
131
+ const columns = opts.columns?.length ? opts.columns : [...DEFAULT_COLUMNS];
132
+ const body = buildScannerBody({
133
+ market,
134
+ columns,
135
+ filter: opts.filter,
136
+ sort: opts.sort,
137
+ limit: opts.limit,
138
+ });
139
+ const rows = decodeScannerRows(await scannerFetch(market, body), columns);
140
+ return rows.map((row) => ({
141
+ ...row,
142
+ sourceProvider: "tradingview" as const,
143
+ dataCaveat: DATA_CAVEAT,
144
+ }));
145
+ }
146
+
147
+ export async function getQuotes(symbols: string[]): Promise<TradingViewQuote[]> {
148
+ const requested = symbols.map(normalizeRequestedSymbol).filter(Boolean);
149
+ const qualified = requested.filter(isTradingViewQualified);
150
+ const bare = requested.filter(
151
+ (symbol) => !isTradingViewQualified(symbol) && !shouldSkipTradingView(symbol),
152
+ );
153
+
154
+ const resolved = new Map<string, TradingViewQuote>();
155
+
156
+ if (qualified.length > 0) {
157
+ const body = buildQualifiedQuoteBody(qualified);
158
+ const rows = decodeScannerRows(await scannerFetch("global", body), [...QUOTE_COLUMNS]);
159
+ const byTvSymbol = new Map(rows.map((row) => [row.tvSymbol.toUpperCase(), row]));
160
+ for (const symbol of qualified) {
161
+ const row = byTvSymbol.get(symbol);
162
+ const quote = row ? rowToQuote(symbol, row) : undefined;
163
+ if (quote) resolved.set(symbol, quote);
164
+ }
165
+ }
166
+
167
+ if (bare.length > 0) {
168
+ const body = buildBareQuoteBody(bare);
169
+ const rows = decodeScannerRows(await scannerFetch("america", body), [...QUOTE_COLUMNS]);
170
+ const byName = groupRowsByName(rows);
171
+ for (const symbol of bare) {
172
+ const row = pickPrimaryListing(byName.get(symbol) ?? []);
173
+ const quote = row ? rowToQuote(symbol, row) : undefined;
174
+ if (quote) resolved.set(symbol, quote);
175
+ }
176
+ }
177
+
178
+ return requested.flatMap((symbol) => {
179
+ const quote = resolved.get(symbol);
180
+ return quote ? [quote] : [];
181
+ });
182
+ }
183
+
184
+ function buildScannerBody(
185
+ opts: Required<Pick<ScreenStocksOpts, "market" | "columns">> &
186
+ Pick<ScreenStocksOpts, "filter" | "sort" | "limit">,
187
+ ): ScannerBody {
188
+ const limit = clampLimit(opts.limit);
189
+ return {
190
+ markets: [opts.market],
191
+ columns: opts.columns,
192
+ ...(opts.filter?.length && { filter: opts.filter.map(mapFilterClause) }),
193
+ ...(opts.sort && {
194
+ sort: {
195
+ sortBy: opts.sort.field,
196
+ sortOrder: opts.sort.direction ?? "desc",
197
+ },
198
+ }),
199
+ range: [0, limit],
200
+ options: { lang: "en" },
201
+ };
202
+ }
203
+
204
+ function buildQualifiedQuoteBody(tickers: string[]): ScannerBody {
205
+ return {
206
+ symbols: { tickers },
207
+ columns: QUOTE_COLUMNS,
208
+ range: [0, clampLimit(tickers.length)],
209
+ options: { lang: "en" },
210
+ };
211
+ }
212
+
213
+ function buildBareQuoteBody(symbols: string[]): ScannerBody {
214
+ return {
215
+ markets: ["america"],
216
+ symbols: { query: { types: [] } },
217
+ columns: QUOTE_COLUMNS,
218
+ filter: [
219
+ { left: "name", operation: "in_range", right: symbols },
220
+ { left: "is_primary", operation: "equal", right: true },
221
+ { left: "type", operation: "in_range", right: ["stock", "fund", "dr"] },
222
+ ],
223
+ range: [0, 500],
224
+ options: { lang: "en" },
225
+ };
226
+ }
227
+
228
+ function mapFilterClause(clause: ScreenFilterClause): {
229
+ left: string;
230
+ operation: string;
231
+ right?: unknown;
232
+ } {
233
+ return {
234
+ left: clause.field,
235
+ operation: clause.op,
236
+ ...(clause.value !== undefined && { right: clause.value }),
237
+ };
238
+ }
239
+
240
+ async function scannerFetch(market: string, body: ScannerBody): Promise<TradingViewResponse> {
241
+ const endpoint = `${BASE_URL}/${market}/scan2?label-product=screener-stock`;
242
+ const cacheKey = `tradingview:scan2:${market}:${stableStringify(body)}`;
243
+ const cached = cache.get<TradingViewResponse>(cacheKey);
244
+ if (cached) return cached;
245
+
246
+ try {
247
+ await rateLimiter.acquire("tradingview");
248
+ const response = await httpPost<TradingViewResponse>(endpoint, body, {
249
+ headers: {
250
+ Origin: "https://www.tradingview.com",
251
+ Referer: "https://www.tradingview.com/",
252
+ "User-Agent": BROWSER_UA,
253
+ },
254
+ });
255
+ cache.set(cacheKey, response, TTL.SCREENER);
256
+ return response;
257
+ } catch (error) {
258
+ const stale = cache.getStale<TradingViewResponse>(cacheKey, STALE_LIMIT.SCREENER);
259
+ if (stale) return stale.value;
260
+ throw error;
261
+ }
262
+ }
263
+
264
+ function decodeScannerRows(
265
+ response: TradingViewResponse,
266
+ requestedColumns: readonly string[],
267
+ ): DecodedRow[] {
268
+ const fields = response.fields ?? [];
269
+ return (response.symbols ?? []).map((row) => {
270
+ const values: Record<string, unknown | null> = {};
271
+ for (const column of requestedColumns) {
272
+ const index = fields.indexOf(column);
273
+ values[column] = index >= 0 ? (row.f[index] ?? null) : null;
274
+ }
275
+ return {
276
+ tvSymbol: row.s,
277
+ symbol: stringValue(values.name) ?? symbolFromTvSymbol(row.s),
278
+ values,
279
+ };
280
+ });
281
+ }
282
+
283
+ function rowToQuote(requestedSymbol: string, row: DecodedRow): TradingViewQuote | undefined {
284
+ const price = numberValue(row.values.close);
285
+ if (price === undefined) return undefined;
286
+ const changePercent = numberValue(row.values.change) ?? 0;
287
+ const changeAbs = numberValue(row.values.change_abs) ?? 0;
288
+ return {
289
+ requestedSymbol,
290
+ tvSymbol: row.tvSymbol,
291
+ symbol: row.symbol,
292
+ price,
293
+ change: changeAbs,
294
+ changePercent,
295
+ volume: numberValue(row.values.volume) ?? 0,
296
+ exchange: stringValue(row.values.exchange),
297
+ market: stringValue(row.values.market),
298
+ name: stringValue(row.values.description),
299
+ type: stringValue(row.values.type),
300
+ typespecs: row.values.typespecs ?? undefined,
301
+ sourceProvider: "tradingview",
302
+ dataCaveat: DATA_CAVEAT,
303
+ };
304
+ }
305
+
306
+ function groupRowsByName(rows: DecodedRow[]): Map<string, DecodedRow[]> {
307
+ const grouped = new Map<string, DecodedRow[]>();
308
+ for (const row of rows) {
309
+ const name = row.symbol.toUpperCase();
310
+ grouped.set(name, [...(grouped.get(name) ?? []), row]);
311
+ }
312
+ return grouped;
313
+ }
314
+
315
+ function pickPrimaryListing(rows: DecodedRow[]): DecodedRow | undefined {
316
+ if (rows.length === 0) return undefined;
317
+ return [...rows].sort(compareListings)[0];
318
+ }
319
+
320
+ function compareListings(a: DecodedRow, b: DecodedRow): number {
321
+ return (
322
+ scoreMarket(a) - scoreMarket(b) ||
323
+ scoreType(a) - scoreType(b) ||
324
+ scoreExchange(a) - scoreExchange(b) ||
325
+ a.tvSymbol.localeCompare(b.tvSymbol)
326
+ );
327
+ }
328
+
329
+ function scoreMarket(row: DecodedRow): number {
330
+ return stringValue(row.values.market)?.toLowerCase() === "america" ? 0 : 1;
331
+ }
332
+
333
+ function scoreType(row: DecodedRow): number {
334
+ const type = stringValue(row.values.type)?.toLowerCase();
335
+ if (type === "stock") return 0;
336
+ if (type === "fund") return 1;
337
+ if (type === "dr") return 2;
338
+ return 3;
339
+ }
340
+
341
+ function scoreExchange(row: DecodedRow): number {
342
+ const exchange = stringValue(row.values.exchange)?.toUpperCase();
343
+ if (exchange === "NASDAQ") return 0;
344
+ if (exchange === "NYSE") return 1;
345
+ if (exchange === "AMEX") return 2;
346
+ return 3;
347
+ }
348
+
349
+ function clampLimit(limit: number | undefined): number {
350
+ if (limit == null) return 50;
351
+ if (limit < 1) return 50;
352
+ return Math.min(Math.floor(limit), 500);
353
+ }
354
+
355
+ function normalizeRequestedSymbol(symbol: string): string {
356
+ return symbol.trim().toUpperCase();
357
+ }
358
+
359
+ export function isTradingViewQualified(symbol: string): boolean {
360
+ return /^[A-Z0-9_]+:[A-Z0-9._-]+$/.test(symbol);
361
+ }
362
+
363
+ export function shouldSkipTradingView(symbol: string): boolean {
364
+ return /(?:-USD|\.(?:TO|DE|T|L|HK))$/i.test(symbol);
365
+ }
366
+
367
+ export function canUseTradingViewQuote(symbol: string): boolean {
368
+ const normalized = normalizeRequestedSymbol(symbol);
369
+ return (
370
+ normalized.length > 0 &&
371
+ (isTradingViewQualified(normalized) || !shouldSkipTradingView(normalized))
372
+ );
373
+ }
374
+
375
+ function symbolFromTvSymbol(tvSymbol: string): string {
376
+ return tvSymbol.includes(":") ? tvSymbol.split(":").at(-1)! : tvSymbol;
377
+ }
378
+
379
+ function stringValue(value: unknown): string | undefined {
380
+ return typeof value === "string" ? value : undefined;
381
+ }
382
+
383
+ function numberValue(value: unknown): number | undefined {
384
+ return typeof value === "number" && Number.isFinite(value) ? value : undefined;
385
+ }
386
+
387
+ function stableStringify(value: unknown): string {
388
+ return JSON.stringify(sortObjectKeys(value));
389
+ }
390
+
391
+ function sortObjectKeys(value: unknown): unknown {
392
+ if (Array.isArray(value)) return value.map(sortObjectKeys);
393
+ if (!value || typeof value !== "object") return value;
394
+ return Object.fromEntries(
395
+ Object.entries(value)
396
+ .sort(([a], [b]) => a.localeCompare(b))
397
+ .map(([key, nested]) => [key, sortObjectKeys(nested)]),
398
+ );
399
+ }
@@ -0,0 +1,233 @@
1
+ import { spawn } from "node:child_process";
2
+ import type { TwitterTweet } from "../types/sentiment.js";
3
+ import { ExternalToolError, ExternalToolNotInstalled } from "./external-tool-error.js";
4
+
5
+ const TWITTER_CLI_BINARY = "twitter";
6
+ const TWITTER_CLI_TOOL_NAME = "twitter-cli";
7
+ const TWITTER_CLI_INSTALL_CMD = "uv tool install twitter-cli";
8
+ const COMMAND_TIMEOUT_MS = 20_000;
9
+ const MAX_OUTPUT_CHARS = 2_000_000;
10
+
11
+ export interface RawTweet {
12
+ readonly id?: string;
13
+ readonly text?: string;
14
+ readonly author?: {
15
+ readonly username?: string;
16
+ readonly screenName?: string;
17
+ readonly name?: string;
18
+ };
19
+ readonly username?: string;
20
+ readonly url?: string;
21
+ readonly permanentUrl?: string;
22
+ readonly createdAt?: string | number;
23
+ readonly created_at?: string | number;
24
+ readonly likeCount?: number;
25
+ readonly likes?: number;
26
+ readonly retweetCount?: number;
27
+ readonly retweets?: number;
28
+ readonly replyCount?: number;
29
+ readonly replies?: number;
30
+ readonly viewCount?: number | null;
31
+ readonly views?: number | null;
32
+ readonly metrics?: {
33
+ readonly likes?: number;
34
+ readonly retweets?: number;
35
+ readonly replies?: number;
36
+ readonly views?: number | null;
37
+ };
38
+ }
39
+
40
+ interface TwitterCliEnvelope<T> {
41
+ readonly ok: boolean;
42
+ readonly schema_version: string;
43
+ readonly data: T;
44
+ readonly error?: {
45
+ readonly code?: string;
46
+ readonly message?: string;
47
+ };
48
+ }
49
+
50
+ interface CommandResult {
51
+ readonly code: number | null;
52
+ readonly stdout: string;
53
+ readonly stderr: string;
54
+ }
55
+
56
+ type TwitterCliCommandRunner = (command: string, args: readonly string[]) => Promise<CommandResult>;
57
+
58
+ let commandRunner: TwitterCliCommandRunner = runCommand;
59
+
60
+ export function setTwitterCliCommandRunnerForTests(runner: TwitterCliCommandRunner): void {
61
+ commandRunner = runner;
62
+ }
63
+
64
+ export function resetTwitterCliCommandRunnerForTests(): void {
65
+ commandRunner = runCommand;
66
+ }
67
+
68
+ export async function searchTweets(query: string, max = 20): Promise<TwitterTweet[]> {
69
+ const envelope = await runTwitterCli<TwitterCliEnvelope<RawTweet[]>>([
70
+ "search",
71
+ query,
72
+ "--max",
73
+ String(max),
74
+ "-t",
75
+ "Latest",
76
+ "--json",
77
+ ]);
78
+
79
+ if (!envelope.ok) {
80
+ throw new ExternalToolError(
81
+ TWITTER_CLI_TOOL_NAME,
82
+ redactSensitiveOutput(envelope.error?.message ?? "twitter-cli returned an error"),
83
+ envelope.error?.code,
84
+ );
85
+ }
86
+ if (!Array.isArray(envelope.data)) {
87
+ throw new ExternalToolError(TWITTER_CLI_TOOL_NAME, "twitter-cli returned invalid tweet data");
88
+ }
89
+ return envelope.data.map(adaptRawTweet);
90
+ }
91
+
92
+ async function runTwitterCli<T>(args: readonly string[]): Promise<T> {
93
+ let result: CommandResult;
94
+ try {
95
+ result = await commandRunner(TWITTER_CLI_BINARY, args);
96
+ } catch (err) {
97
+ const nodeError = err as NodeJS.ErrnoException;
98
+ if (nodeError.code === "ENOENT") {
99
+ throw new ExternalToolNotInstalled(TWITTER_CLI_TOOL_NAME, TWITTER_CLI_INSTALL_CMD);
100
+ }
101
+ throw new ExternalToolError(
102
+ TWITTER_CLI_TOOL_NAME,
103
+ redactSensitiveOutput(err instanceof Error ? err.message : String(err)),
104
+ );
105
+ }
106
+
107
+ if (result.code !== 0) {
108
+ const envelopeError = parseCliErrorEnvelope(result.stdout);
109
+ if (envelopeError) {
110
+ throw new ExternalToolError(
111
+ TWITTER_CLI_TOOL_NAME,
112
+ redactSensitiveOutput(envelopeError.message),
113
+ envelopeError.code,
114
+ );
115
+ }
116
+ throw new ExternalToolError(
117
+ TWITTER_CLI_TOOL_NAME,
118
+ redactSensitiveOutput(result.stderr.trim() || `twitter-cli exited with code ${result.code}`),
119
+ );
120
+ }
121
+
122
+ try {
123
+ return JSON.parse(result.stdout) as T;
124
+ } catch {
125
+ throw new ExternalToolError(
126
+ TWITTER_CLI_TOOL_NAME,
127
+ `twitter-cli returned non-JSON output: ${redactSensitiveOutput(result.stdout.slice(0, 200))}`,
128
+ );
129
+ }
130
+ }
131
+
132
+ function parseCliErrorEnvelope(stdout: string): { code?: string; message: string } | null {
133
+ try {
134
+ const parsed = JSON.parse(stdout) as {
135
+ ok?: unknown;
136
+ error?: { code?: unknown; message?: unknown };
137
+ };
138
+ if (parsed.ok !== false || typeof parsed.error?.message !== "string") return null;
139
+ return {
140
+ code: typeof parsed.error.code === "string" ? parsed.error.code : undefined,
141
+ message: parsed.error.message,
142
+ };
143
+ } catch {
144
+ return null;
145
+ }
146
+ }
147
+
148
+ function adaptRawTweet(raw: RawTweet): TwitterTweet {
149
+ return {
150
+ id: stringValue(raw.id),
151
+ text: stringValue(raw.text).slice(0, 280),
152
+ author:
153
+ stringValue(raw.author?.username) ||
154
+ stringValue(raw.author?.screenName) ||
155
+ stringValue(raw.username) ||
156
+ "unknown",
157
+ likes: numberValue(raw.metrics?.likes ?? raw.likeCount ?? raw.likes),
158
+ retweets: numberValue(raw.metrics?.retweets ?? raw.retweetCount ?? raw.retweets),
159
+ replies: numberValue(raw.metrics?.replies ?? raw.replyCount ?? raw.replies),
160
+ views: nullableNumberValue(raw.metrics?.views ?? raw.viewCount ?? raw.views),
161
+ url: stringValue(raw.url ?? raw.permanentUrl),
162
+ created: normalizeCreatedAt(raw.createdAt ?? raw.created_at),
163
+ };
164
+ }
165
+
166
+ function normalizeCreatedAt(value: string | number | undefined): string {
167
+ if (typeof value === "number") {
168
+ const millis = value > 1_000_000_000_000 ? value : value * 1000;
169
+ return new Date(millis).toISOString();
170
+ }
171
+ if (typeof value === "string" && value.length > 0) {
172
+ const millis = Date.parse(value);
173
+ if (!Number.isNaN(millis)) return new Date(millis).toISOString();
174
+ }
175
+ return new Date(0).toISOString();
176
+ }
177
+
178
+ function stringValue(value: unknown): string {
179
+ return typeof value === "string" ? value : "";
180
+ }
181
+
182
+ function numberValue(value: unknown): number {
183
+ return typeof value === "number" && Number.isFinite(value) ? value : 0;
184
+ }
185
+
186
+ function nullableNumberValue(value: unknown): number | null {
187
+ return typeof value === "number" && Number.isFinite(value) ? value : null;
188
+ }
189
+
190
+ export function redactSensitiveOutput(input: string): string {
191
+ const xCookieNames =
192
+ "auth_token|ct0|twid|kdt|guest_id|guest_id_ads|guest_id_marketing|personalization_id";
193
+ return input
194
+ .slice(0, MAX_OUTPUT_CHARS)
195
+ .replace(/\b(cookie|set-cookie)\s*:\s*[^\r\n]+/gi, "$1: [redacted]")
196
+ .replace(new RegExp(`\\b(${xCookieNames})\\b\\s*[:=]\\s*[^;\\s,)]+`, "gi"), "$1=[redacted]")
197
+ .replace(new RegExp(`\\b(${xCookieNames})=([^;\\s,)]+)`, "gi"), "$1=[redacted]");
198
+ }
199
+
200
+ function runCommand(command: string, args: readonly string[]): Promise<CommandResult> {
201
+ return new Promise((resolve, reject) => {
202
+ const child = spawn(command, [...args], { stdio: ["ignore", "pipe", "pipe"] });
203
+ let stdout = "";
204
+ let stderr = "";
205
+ let settled = false;
206
+
207
+ const timeout = setTimeout(() => {
208
+ if (settled) return;
209
+ settled = true;
210
+ child.kill("SIGTERM");
211
+ reject(new Error(`${command} timed out after ${COMMAND_TIMEOUT_MS}ms`));
212
+ }, COMMAND_TIMEOUT_MS);
213
+
214
+ child.stdout.on("data", (chunk: Buffer) => {
215
+ stdout = (stdout + chunk.toString("utf8")).slice(0, MAX_OUTPUT_CHARS);
216
+ });
217
+ child.stderr.on("data", (chunk: Buffer) => {
218
+ stderr = (stderr + chunk.toString("utf8")).slice(0, MAX_OUTPUT_CHARS);
219
+ });
220
+ child.on("error", (err) => {
221
+ if (settled) return;
222
+ settled = true;
223
+ clearTimeout(timeout);
224
+ reject(err);
225
+ });
226
+ child.on("close", (code) => {
227
+ if (settled) return;
228
+ settled = true;
229
+ clearTimeout(timeout);
230
+ resolve({ code, stdout, stderr });
231
+ });
232
+ });
233
+ }