opencandle 0.3.0 → 0.5.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 (408) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +106 -14
  3. package/assets/logo.svg +187 -0
  4. package/dist/cli.d.ts +1 -1
  5. package/dist/cli.js +40 -3
  6. package/dist/cli.js.map +1 -1
  7. package/dist/config.d.ts +25 -0
  8. package/dist/config.js +72 -0
  9. package/dist/config.js.map +1 -1
  10. package/dist/infra/browser.d.ts +11 -3
  11. package/dist/infra/browser.js +2 -1
  12. package/dist/infra/browser.js.map +1 -1
  13. package/dist/infra/native-dependencies.d.ts +1 -0
  14. package/dist/infra/native-dependencies.js +10 -0
  15. package/dist/infra/native-dependencies.js.map +1 -0
  16. package/dist/infra/node-version.d.ts +2 -0
  17. package/dist/infra/node-version.js +23 -0
  18. package/dist/infra/node-version.js.map +1 -0
  19. package/dist/infra/rate-limiter.d.ts +4 -0
  20. package/dist/infra/rate-limiter.js +5 -1
  21. package/dist/infra/rate-limiter.js.map +1 -1
  22. package/dist/memory/index.d.ts +2 -0
  23. package/dist/memory/index.js +1 -0
  24. package/dist/memory/index.js.map +1 -1
  25. package/dist/memory/manager.d.ts +9 -0
  26. package/dist/memory/manager.js +28 -11
  27. package/dist/memory/manager.js.map +1 -1
  28. package/dist/memory/sqlite.js +42 -4
  29. package/dist/memory/sqlite.js.map +1 -1
  30. package/dist/memory/storage.d.ts +7 -0
  31. package/dist/memory/storage.js +3 -3
  32. package/dist/memory/storage.js.map +1 -1
  33. package/dist/memory/tool-defaults.d.ts +8 -0
  34. package/dist/memory/tool-defaults.js +59 -0
  35. package/dist/memory/tool-defaults.js.map +1 -0
  36. package/dist/memory/types.js +4 -0
  37. package/dist/memory/types.js.map +1 -1
  38. package/dist/onboarding/connect.d.ts +13 -1
  39. package/dist/onboarding/connect.js +21 -10
  40. package/dist/onboarding/connect.js.map +1 -1
  41. package/dist/onboarding/prompt-user.d.ts +1 -1
  42. package/dist/onboarding/providers.d.ts +7 -0
  43. package/dist/onboarding/providers.js +6 -3
  44. package/dist/onboarding/providers.js.map +1 -1
  45. package/dist/onboarding/tool-helpers.d.ts +1 -1
  46. package/dist/pi/opencandle-extension.d.ts +7 -1
  47. package/dist/pi/opencandle-extension.js +391 -21
  48. package/dist/pi/opencandle-extension.js.map +1 -1
  49. package/dist/pi/session-storage.d.ts +2 -0
  50. package/dist/pi/session-storage.js +5 -0
  51. package/dist/pi/session-storage.js.map +1 -0
  52. package/dist/pi/session.d.ts +4 -1
  53. package/dist/pi/session.js +25 -3
  54. package/dist/pi/session.js.map +1 -1
  55. package/dist/pi/setup.d.ts +1 -1
  56. package/dist/pi/setup.js +11 -1
  57. package/dist/pi/setup.js.map +1 -1
  58. package/dist/pi/tool-adapter.d.ts +2 -2
  59. package/dist/pi/tool-adapter.js +14 -1
  60. package/dist/pi/tool-adapter.js.map +1 -1
  61. package/dist/prompts/context-builder.d.ts +40 -3
  62. package/dist/prompts/context-builder.js +140 -19
  63. package/dist/prompts/context-builder.js.map +1 -1
  64. package/dist/prompts/disclaimer.d.ts +6 -0
  65. package/dist/prompts/disclaimer.js +9 -0
  66. package/dist/prompts/disclaimer.js.map +1 -0
  67. package/dist/prompts/policy-cards.d.ts +13 -0
  68. package/dist/prompts/policy-cards.js +197 -0
  69. package/dist/prompts/policy-cards.js.map +1 -0
  70. package/dist/prompts/sections.js +3 -3
  71. package/dist/prompts/sections.js.map +1 -1
  72. package/dist/prompts/workflow-prompts.d.ts +8 -0
  73. package/dist/prompts/workflow-prompts.js +208 -22
  74. package/dist/prompts/workflow-prompts.js.map +1 -1
  75. package/dist/providers/alpha-vantage.js +23 -1
  76. package/dist/providers/alpha-vantage.js.map +1 -1
  77. package/dist/providers/sec-edgar.d.ts +8 -1
  78. package/dist/providers/sec-edgar.js +172 -5
  79. package/dist/providers/sec-edgar.js.map +1 -1
  80. package/dist/providers/yahoo-finance.d.ts +2 -0
  81. package/dist/providers/yahoo-finance.js +203 -35
  82. package/dist/providers/yahoo-finance.js.map +1 -1
  83. package/dist/routing/classify-intent.d.ts +3 -0
  84. package/dist/routing/classify-intent.js +82 -3
  85. package/dist/routing/classify-intent.js.map +1 -1
  86. package/dist/routing/defaults.js +4 -4
  87. package/dist/routing/defaults.js.map +1 -1
  88. package/dist/routing/entity-extractor.d.ts +1 -0
  89. package/dist/routing/entity-extractor.js +158 -12
  90. package/dist/routing/entity-extractor.js.map +1 -1
  91. package/dist/routing/index.d.ts +10 -0
  92. package/dist/routing/index.js +7 -0
  93. package/dist/routing/index.js.map +1 -1
  94. package/dist/routing/legacy-rule-router.d.ts +9 -0
  95. package/dist/routing/legacy-rule-router.js +12 -0
  96. package/dist/routing/legacy-rule-router.js.map +1 -0
  97. package/dist/routing/planning.d.ts +54 -0
  98. package/dist/routing/planning.js +531 -0
  99. package/dist/routing/planning.js.map +1 -0
  100. package/dist/routing/route-manifest.d.ts +35 -0
  101. package/dist/routing/route-manifest.js +221 -0
  102. package/dist/routing/route-manifest.js.map +1 -0
  103. package/dist/routing/router-llm-client.d.ts +11 -0
  104. package/dist/routing/router-llm-client.js +42 -0
  105. package/dist/routing/router-llm-client.js.map +1 -0
  106. package/dist/routing/router-prompt.d.ts +2 -0
  107. package/dist/routing/router-prompt.js +141 -0
  108. package/dist/routing/router-prompt.js.map +1 -0
  109. package/dist/routing/router-types.d.ts +71 -0
  110. package/dist/routing/router-types.js +2 -0
  111. package/dist/routing/router-types.js.map +1 -0
  112. package/dist/routing/router.d.ts +11 -0
  113. package/dist/routing/router.js +638 -0
  114. package/dist/routing/router.js.map +1 -0
  115. package/dist/routing/slot-resolver.js +46 -6
  116. package/dist/routing/slot-resolver.js.map +1 -1
  117. package/dist/routing/turn-context.d.ts +44 -0
  118. package/dist/routing/turn-context.js +45 -0
  119. package/dist/routing/turn-context.js.map +1 -0
  120. package/dist/routing/types.d.ts +13 -1
  121. package/dist/runtime/answer-contracts.d.ts +82 -0
  122. package/dist/runtime/answer-contracts.js +414 -0
  123. package/dist/runtime/answer-contracts.js.map +1 -0
  124. package/dist/runtime/artifact-contracts.d.ts +14 -0
  125. package/dist/runtime/artifact-contracts.js +57 -0
  126. package/dist/runtime/artifact-contracts.js.map +1 -0
  127. package/dist/runtime/planning-evidence.d.ts +99 -0
  128. package/dist/runtime/planning-evidence.js +445 -0
  129. package/dist/runtime/planning-evidence.js.map +1 -0
  130. package/dist/runtime/session-coordinator.d.ts +81 -3
  131. package/dist/runtime/session-coordinator.js +201 -17
  132. package/dist/runtime/session-coordinator.js.map +1 -1
  133. package/dist/runtime/tool-defaults-wrapper.d.ts +3 -0
  134. package/dist/runtime/tool-defaults-wrapper.js +25 -0
  135. package/dist/runtime/tool-defaults-wrapper.js.map +1 -0
  136. package/dist/sentiment/store.js +5 -0
  137. package/dist/sentiment/store.js.map +1 -1
  138. package/dist/system-prompt.js +23 -12
  139. package/dist/system-prompt.js.map +1 -1
  140. package/dist/tool-kit.d.ts +4 -4
  141. package/dist/tools/fundamentals/company-overview.d.ts +1 -1
  142. package/dist/tools/fundamentals/company-overview.js +1 -1
  143. package/dist/tools/fundamentals/company-overview.js.map +1 -1
  144. package/dist/tools/fundamentals/comps.d.ts +1 -1
  145. package/dist/tools/fundamentals/comps.js +1 -1
  146. package/dist/tools/fundamentals/comps.js.map +1 -1
  147. package/dist/tools/fundamentals/dcf.d.ts +1 -1
  148. package/dist/tools/fundamentals/dcf.js +1 -1
  149. package/dist/tools/fundamentals/dcf.js.map +1 -1
  150. package/dist/tools/fundamentals/earnings.d.ts +1 -1
  151. package/dist/tools/fundamentals/earnings.js +1 -1
  152. package/dist/tools/fundamentals/earnings.js.map +1 -1
  153. package/dist/tools/fundamentals/financials.d.ts +1 -1
  154. package/dist/tools/fundamentals/financials.js +1 -1
  155. package/dist/tools/fundamentals/financials.js.map +1 -1
  156. package/dist/tools/fundamentals/sec-filings.d.ts +2 -1
  157. package/dist/tools/fundamentals/sec-filings.js +19 -2
  158. package/dist/tools/fundamentals/sec-filings.js.map +1 -1
  159. package/dist/tools/index.d.ts +29 -1
  160. package/dist/tools/index.js +30 -0
  161. package/dist/tools/index.js.map +1 -1
  162. package/dist/tools/interaction/ask-user.d.ts +1 -1
  163. package/dist/tools/interaction/twitter-login.d.ts +1 -1
  164. package/dist/tools/macro/fear-greed.d.ts +1 -1
  165. package/dist/tools/macro/fear-greed.js +1 -1
  166. package/dist/tools/macro/fear-greed.js.map +1 -1
  167. package/dist/tools/macro/fred-data.d.ts +1 -1
  168. package/dist/tools/macro/fred-data.js +29 -5
  169. package/dist/tools/macro/fred-data.js.map +1 -1
  170. package/dist/tools/market/crypto-history.d.ts +1 -1
  171. package/dist/tools/market/crypto-history.js +18 -2
  172. package/dist/tools/market/crypto-history.js.map +1 -1
  173. package/dist/tools/market/crypto-price.d.ts +1 -1
  174. package/dist/tools/market/crypto-price.js +1 -1
  175. package/dist/tools/market/crypto-price.js.map +1 -1
  176. package/dist/tools/market/search-ticker.d.ts +1 -1
  177. package/dist/tools/market/search-ticker.js +1 -1
  178. package/dist/tools/market/search-ticker.js.map +1 -1
  179. package/dist/tools/market/stock-history.d.ts +1 -1
  180. package/dist/tools/market/stock-history.js +1 -1
  181. package/dist/tools/market/stock-history.js.map +1 -1
  182. package/dist/tools/market/stock-quote.d.ts +1 -1
  183. package/dist/tools/market/stock-quote.js +1 -1
  184. package/dist/tools/market/stock-quote.js.map +1 -1
  185. package/dist/tools/options/greeks.js +0 -1
  186. package/dist/tools/options/greeks.js.map +1 -1
  187. package/dist/tools/options/option-chain.d.ts +1 -1
  188. package/dist/tools/options/option-chain.js +13 -5
  189. package/dist/tools/options/option-chain.js.map +1 -1
  190. package/dist/tools/portfolio/correlation.d.ts +1 -1
  191. package/dist/tools/portfolio/correlation.js +1 -1
  192. package/dist/tools/portfolio/correlation.js.map +1 -1
  193. package/dist/tools/portfolio/holdings-overlap.d.ts +8 -0
  194. package/dist/tools/portfolio/holdings-overlap.js +105 -0
  195. package/dist/tools/portfolio/holdings-overlap.js.map +1 -0
  196. package/dist/tools/portfolio/predictions.d.ts +1 -1
  197. package/dist/tools/portfolio/predictions.js +1 -1
  198. package/dist/tools/portfolio/predictions.js.map +1 -1
  199. package/dist/tools/portfolio/risk-analysis.d.ts +1 -1
  200. package/dist/tools/portfolio/risk-analysis.js +1 -1
  201. package/dist/tools/portfolio/risk-analysis.js.map +1 -1
  202. package/dist/tools/portfolio/tracker.d.ts +1 -1
  203. package/dist/tools/portfolio/tracker.js +1 -1
  204. package/dist/tools/portfolio/tracker.js.map +1 -1
  205. package/dist/tools/portfolio/watchlist.d.ts +1 -1
  206. package/dist/tools/portfolio/watchlist.js +12 -4
  207. package/dist/tools/portfolio/watchlist.js.map +1 -1
  208. package/dist/tools/sentiment/reddit-sentiment.d.ts +1 -1
  209. package/dist/tools/sentiment/reddit-sentiment.js +1 -1
  210. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  211. package/dist/tools/sentiment/sentiment-summary.d.ts +1 -1
  212. package/dist/tools/sentiment/sentiment-summary.js +57 -2
  213. package/dist/tools/sentiment/sentiment-summary.js.map +1 -1
  214. package/dist/tools/sentiment/sentiment-trend.d.ts +1 -1
  215. package/dist/tools/sentiment/twitter-sentiment.d.ts +1 -1
  216. package/dist/tools/sentiment/twitter-sentiment.js +1 -1
  217. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
  218. package/dist/tools/sentiment/web-search.d.ts +1 -1
  219. package/dist/tools/sentiment/web-search.js +32 -3
  220. package/dist/tools/sentiment/web-search.js.map +1 -1
  221. package/dist/tools/sentiment/web-sentiment.d.ts +1 -1
  222. package/dist/tools/sentiment/web-sentiment.js +1 -1
  223. package/dist/tools/sentiment/web-sentiment.js.map +1 -1
  224. package/dist/tools/technical/backtest.d.ts +3 -3
  225. package/dist/tools/technical/backtest.js +41 -27
  226. package/dist/tools/technical/backtest.js.map +1 -1
  227. package/dist/tools/technical/indicators.d.ts +1 -1
  228. package/dist/tools/technical/indicators.js +8 -4
  229. package/dist/tools/technical/indicators.js.map +1 -1
  230. package/dist/types/options.d.ts +10 -0
  231. package/dist/types/portfolio.d.ts +27 -0
  232. package/dist/workflows/compare-assets.js +38 -2
  233. package/dist/workflows/compare-assets.js.map +1 -1
  234. package/dist/workflows/options-screener.js +94 -8
  235. package/dist/workflows/options-screener.js.map +1 -1
  236. package/dist/workflows/portfolio-builder.js +9 -5
  237. package/dist/workflows/portfolio-builder.js.map +1 -1
  238. package/gui/server/ask-user-bridge.ts +82 -0
  239. package/gui/server/background-quotes.ts +31 -0
  240. package/gui/server/chat-event-adapter.ts +142 -0
  241. package/gui/server/gui-session-manager.ts +5 -0
  242. package/gui/server/invoke-tool.ts +89 -0
  243. package/gui/server/live-chat-event-adapter.ts +181 -0
  244. package/gui/server/model-setup.ts +100 -0
  245. package/gui/server/package.json +5 -0
  246. package/gui/server/projector.ts +254 -0
  247. package/gui/server/prompt-observation.ts +61 -0
  248. package/gui/server/server.ts +703 -0
  249. package/gui/server/session-actions.ts +31 -0
  250. package/gui/server/session-entry-wait.ts +81 -0
  251. package/gui/server/tool-metadata.ts +88 -0
  252. package/gui/server/websocket.ts +128 -0
  253. package/gui/server/writer-lock.ts +118 -0
  254. package/gui/shared/chat-events.ts +118 -0
  255. package/gui/shared/event-reducer.ts +186 -0
  256. package/gui/web/dist/assets/CatalogOverlay-Bmp6Knu7.js +1 -0
  257. package/gui/web/dist/assets/index-Bxt9QpLX.css +1 -0
  258. package/gui/web/dist/assets/index-CZ9DHZYy.js +67 -0
  259. package/gui/web/dist/assets/logo-CWpt6Y2a.svg +187 -0
  260. package/gui/web/dist/index.html +17 -0
  261. package/package.json +50 -18
  262. package/src/analysts/contracts.ts +189 -0
  263. package/src/analysts/orchestrator.ts +300 -0
  264. package/src/cli.ts +206 -0
  265. package/src/config.ts +245 -0
  266. package/src/index.ts +5 -0
  267. package/src/infra/browser.ts +111 -0
  268. package/src/infra/cache.ts +103 -0
  269. package/src/infra/http-client.ts +68 -0
  270. package/src/infra/index.ts +18 -0
  271. package/src/infra/native-dependencies.ts +12 -0
  272. package/src/infra/node-version.ts +24 -0
  273. package/src/infra/open-url.ts +28 -0
  274. package/src/infra/opencandle-paths.ts +64 -0
  275. package/src/infra/rate-limiter.ts +73 -0
  276. package/src/memory/index.ts +10 -0
  277. package/src/memory/manager.ts +192 -0
  278. package/src/memory/preference-extractor.ts +106 -0
  279. package/src/memory/retrieval.ts +70 -0
  280. package/src/memory/sqlite.ts +172 -0
  281. package/src/memory/storage.ts +205 -0
  282. package/src/memory/tool-defaults.ts +87 -0
  283. package/src/memory/types.ts +71 -0
  284. package/src/onboarding/connect.ts +184 -0
  285. package/src/onboarding/credential-interceptor.ts +134 -0
  286. package/src/onboarding/degradation-accumulator.ts +79 -0
  287. package/src/onboarding/prompt-user.ts +85 -0
  288. package/src/onboarding/providers.ts +315 -0
  289. package/src/onboarding/state.ts +218 -0
  290. package/src/onboarding/tool-helpers.ts +111 -0
  291. package/src/onboarding/tool-tags.ts +201 -0
  292. package/src/onboarding/validation.ts +158 -0
  293. package/src/pi/opencandle-extension.ts +955 -0
  294. package/src/pi/session-storage.ts +5 -0
  295. package/src/pi/session.ts +81 -0
  296. package/src/pi/setup.ts +381 -0
  297. package/src/pi/tool-adapter.ts +36 -0
  298. package/src/prompts/context-builder.ts +315 -0
  299. package/src/prompts/disclaimer.ts +9 -0
  300. package/src/prompts/policy-cards.ts +220 -0
  301. package/src/prompts/sections.ts +46 -0
  302. package/src/prompts/workflow-prompts.ts +433 -0
  303. package/src/providers/alpha-vantage.ts +315 -0
  304. package/src/providers/coingecko.ts +96 -0
  305. package/src/providers/exa-search.ts +373 -0
  306. package/src/providers/fear-greed.ts +45 -0
  307. package/src/providers/finnhub.ts +124 -0
  308. package/src/providers/fred.ts +83 -0
  309. package/src/providers/index.ts +9 -0
  310. package/src/providers/provider-credential-error.ts +23 -0
  311. package/src/providers/reddit.ts +151 -0
  312. package/src/providers/sec-edgar.ts +312 -0
  313. package/src/providers/twitter.ts +173 -0
  314. package/src/providers/web-search.ts +293 -0
  315. package/src/providers/with-fallback.ts +41 -0
  316. package/src/providers/wrap-provider.ts +64 -0
  317. package/src/providers/yahoo-finance.ts +534 -0
  318. package/src/routing/classify-intent.ts +285 -0
  319. package/src/routing/defaults.ts +29 -0
  320. package/src/routing/entity-extractor.ts +291 -0
  321. package/src/routing/index.ts +70 -0
  322. package/src/routing/legacy-rule-router.ts +13 -0
  323. package/src/routing/planning.ts +732 -0
  324. package/src/routing/route-manifest.ts +287 -0
  325. package/src/routing/router-llm-client.ts +51 -0
  326. package/src/routing/router-prompt.ts +163 -0
  327. package/src/routing/router-types.ts +87 -0
  328. package/src/routing/router.ts +712 -0
  329. package/src/routing/slot-resolver.ts +190 -0
  330. package/src/routing/turn-context.ts +111 -0
  331. package/src/routing/types.ts +75 -0
  332. package/src/runtime/answer-contracts.ts +633 -0
  333. package/src/runtime/artifact-contracts.ts +76 -0
  334. package/src/runtime/evidence.ts +77 -0
  335. package/src/runtime/planning-evidence.ts +591 -0
  336. package/src/runtime/prompt-step.ts +75 -0
  337. package/src/runtime/provider-tracker.ts +40 -0
  338. package/src/runtime/run-context.ts +22 -0
  339. package/src/runtime/session-coordinator.ts +472 -0
  340. package/src/runtime/tool-defaults-wrapper.ts +35 -0
  341. package/src/runtime/validation.ts +214 -0
  342. package/src/runtime/workflow-events.ts +75 -0
  343. package/src/runtime/workflow-runner.ts +188 -0
  344. package/src/runtime/workflow-types.ts +102 -0
  345. package/src/sentiment/adapters/finnhub.ts +44 -0
  346. package/src/sentiment/adapters/reddit.ts +65 -0
  347. package/src/sentiment/adapters/twitter.ts +36 -0
  348. package/src/sentiment/adapters/web.ts +44 -0
  349. package/src/sentiment/index.ts +58 -0
  350. package/src/sentiment/keywords.ts +9 -0
  351. package/src/sentiment/pipeline.ts +68 -0
  352. package/src/sentiment/scorer.ts +78 -0
  353. package/src/sentiment/store.ts +260 -0
  354. package/src/sentiment/trends.ts +90 -0
  355. package/src/sentiment/types.ts +108 -0
  356. package/src/system-prompt.ts +118 -0
  357. package/src/tool-kit.ts +68 -0
  358. package/src/tools/AGENTS.md +36 -0
  359. package/src/tools/fundamentals/company-overview.ts +54 -0
  360. package/src/tools/fundamentals/comps.ts +156 -0
  361. package/src/tools/fundamentals/dcf.ts +267 -0
  362. package/src/tools/fundamentals/earnings.ts +47 -0
  363. package/src/tools/fundamentals/financials.ts +54 -0
  364. package/src/tools/fundamentals/sec-filings.ts +84 -0
  365. package/src/tools/index.ts +91 -0
  366. package/src/tools/interaction/ask-user.ts +81 -0
  367. package/src/tools/interaction/twitter-login.ts +93 -0
  368. package/src/tools/macro/fear-greed.ts +41 -0
  369. package/src/tools/macro/fred-data.ts +80 -0
  370. package/src/tools/market/crypto-history.ts +67 -0
  371. package/src/tools/market/crypto-price.ts +53 -0
  372. package/src/tools/market/search-ticker.ts +53 -0
  373. package/src/tools/market/stock-history.ts +79 -0
  374. package/src/tools/market/stock-quote.ts +64 -0
  375. package/src/tools/options/greeks.ts +81 -0
  376. package/src/tools/options/option-chain.ts +96 -0
  377. package/src/tools/portfolio/correlation.ts +162 -0
  378. package/src/tools/portfolio/holdings-overlap.ts +123 -0
  379. package/src/tools/portfolio/predictions.ts +253 -0
  380. package/src/tools/portfolio/risk-analysis.ts +134 -0
  381. package/src/tools/portfolio/tracker.ts +147 -0
  382. package/src/tools/portfolio/watchlist.ts +159 -0
  383. package/src/tools/sentiment/reddit-sentiment.ts +164 -0
  384. package/src/tools/sentiment/sentiment-summary.ts +316 -0
  385. package/src/tools/sentiment/sentiment-trend.ts +58 -0
  386. package/src/tools/sentiment/twitter-sentiment.ts +96 -0
  387. package/src/tools/sentiment/web-search.ts +183 -0
  388. package/src/tools/sentiment/web-sentiment.ts +76 -0
  389. package/src/tools/technical/backtest.ts +267 -0
  390. package/src/tools/technical/indicators.ts +256 -0
  391. package/src/types/fundamentals.ts +46 -0
  392. package/src/types/index.ts +20 -0
  393. package/src/types/macro.ts +27 -0
  394. package/src/types/market.ts +43 -0
  395. package/src/types/options.ts +52 -0
  396. package/src/types/portfolio.ts +73 -0
  397. package/src/types/sentiment.ts +70 -0
  398. package/src/workflows/compare-assets.ts +75 -0
  399. package/src/workflows/index.ts +4 -0
  400. package/src/workflows/options-screener.ts +127 -0
  401. package/src/workflows/portfolio-builder.ts +56 -0
  402. package/src/workflows/types.ts +4 -0
  403. package/dist/runtime/index.d.ts +0 -16
  404. package/dist/runtime/index.js +0 -10
  405. package/dist/runtime/index.js.map +0 -1
  406. package/dist/runtime/provider-ids.d.ts +0 -14
  407. package/dist/runtime/provider-ids.js +0 -14
  408. package/dist/runtime/provider-ids.js.map +0 -1
@@ -0,0 +1,315 @@
1
+ import { httpGet, HttpError } from "../infra/http-client.js";
2
+ import { cache, TTL, STALE_LIMIT } from "../infra/cache.js";
3
+ import { rateLimiter } from "../infra/rate-limiter.js";
4
+ import { ProviderCredentialError } from "./provider-credential-error.js";
5
+ import type { CompanyOverview, EarningsData, FinancialStatement } from "../types/fundamentals.js";
6
+ import type { StockQuote, OHLCV } from "../types/market.js";
7
+
8
+ const BASE_URL = "https://www.alphavantage.co/query";
9
+ const MISSING_OVERVIEW_TTL = 15 * 60_000;
10
+
11
+ /**
12
+ * Detects authentication failures in HTTP errors and re-throws them as a
13
+ * typed `ProviderCredentialError` so the tool layer can convert them into
14
+ * a `[OPENCANDLE_CREDENTIAL_REQUIRED ...]` tagged content block.
15
+ *
16
+ * Call from every catch block in this module BEFORE any stale-cache fallback
17
+ * so that a real 401/403 does not silently get masked by cached data.
18
+ */
19
+ function throwIfAuthError(error: unknown): void {
20
+ if (error instanceof HttpError && (error.status === 401 || error.status === 403)) {
21
+ throw new ProviderCredentialError("alpha_vantage", "stale", error.status);
22
+ }
23
+ }
24
+
25
+ function buildUrl(fn: string, params: Record<string, string>, apiKey: string): string {
26
+ const qs = new URLSearchParams({ function: fn, ...params, apikey: apiKey });
27
+ return `${BASE_URL}?${qs}`;
28
+ }
29
+
30
+ function throwIfApiMessage(data: unknown): void {
31
+ if (!data || typeof data !== "object") return;
32
+
33
+ const payload = data as Record<string, unknown>;
34
+ const message = payload.Note ?? payload.Information ?? payload["Error Message"];
35
+ if (typeof message !== "string" || message.length === 0) return;
36
+
37
+ const normalized = message.toLowerCase();
38
+ if (normalized.includes("api call frequency") || normalized.includes("rate limit")) {
39
+ throw new Error(`Alpha Vantage rate limited: ${message}`);
40
+ }
41
+ if (normalized.includes("invalid api")) {
42
+ throw new ProviderCredentialError("alpha_vantage", "stale");
43
+ }
44
+ throw new Error(`Alpha Vantage error: ${message}`);
45
+ }
46
+
47
+ export async function getOverview(
48
+ symbol: string,
49
+ apiKey: string,
50
+ ): Promise<CompanyOverview> {
51
+ const cacheKey = `av:overview:${symbol}`;
52
+ const missingCacheKey = `${cacheKey}:missing`;
53
+ const cached = cache.get<CompanyOverview>(cacheKey);
54
+ if (cached) return cached;
55
+ if (cache.get<string>(missingCacheKey)) {
56
+ throw new Error(`Alpha Vantage: No data found for ${symbol}`);
57
+ }
58
+
59
+ try {
60
+ await rateLimiter.acquire("alphavantage");
61
+
62
+ const url = buildUrl("OVERVIEW", { symbol }, apiKey);
63
+ const data = await httpGet<Record<string, string>>(url);
64
+ throwIfApiMessage(data);
65
+
66
+ if (!data.Symbol) {
67
+ cache.set(missingCacheKey, "missing", MISSING_OVERVIEW_TTL);
68
+ throw new Error(`Alpha Vantage: No data found for ${symbol}`);
69
+ }
70
+
71
+ const result: CompanyOverview = {
72
+ symbol: data.Symbol,
73
+ name: data.Name,
74
+ description: data.Description,
75
+ exchange: data.Exchange,
76
+ sector: data.Sector,
77
+ industry: data.Industry,
78
+ marketCap: parseNum(data.MarketCapitalization),
79
+ pe: parseNullableNum(data.PERatio),
80
+ forwardPe: parseNullableNum(data.ForwardPE),
81
+ eps: parseNullableNum(data.EPS),
82
+ dividendYield: parseNullableNum(data.DividendYield),
83
+ beta: parseNullableNum(data.Beta),
84
+ week52High: parseNum(data["52WeekHigh"]),
85
+ week52Low: parseNum(data["52WeekLow"]),
86
+ avgVolume: 0, // Alpha Vantage OVERVIEW does not expose average volume
87
+ profitMargin: parseNullableNum(data.ProfitMargin),
88
+ revenueGrowth: parseNullableNum(data.QuarterlyRevenueGrowthYOY),
89
+ };
90
+
91
+ cache.set(cacheKey, result, TTL.FUNDAMENTALS);
92
+ return result;
93
+ } catch (error) {
94
+ throwIfAuthError(error);
95
+ const stale = cache.getStale<CompanyOverview>(cacheKey, STALE_LIMIT.FUNDAMENTALS);
96
+ if (stale) return stale.value;
97
+ throw error;
98
+ }
99
+ }
100
+
101
+ export async function getEarnings(
102
+ symbol: string,
103
+ apiKey: string,
104
+ ): Promise<EarningsData> {
105
+ const cacheKey = `av:earnings:${symbol}`;
106
+ const cached = cache.get<EarningsData>(cacheKey);
107
+ if (cached) return cached;
108
+
109
+ try {
110
+ await rateLimiter.acquire("alphavantage");
111
+
112
+ const url = buildUrl("EARNINGS", { symbol }, apiKey);
113
+ const data = await httpGet<{ quarterlyEarnings: any[] }>(url);
114
+ throwIfApiMessage(data);
115
+
116
+ const quarterly = (data.quarterlyEarnings ?? []).slice(0, 8).map((e: any) => ({
117
+ date: e.fiscalDateEnding,
118
+ reportedEPS: parseFloat(e.reportedEPS) || 0,
119
+ estimatedEPS: parseFloat(e.estimatedEPS) || 0,
120
+ surprise: parseFloat(e.surprise) || 0,
121
+ surprisePercent: parseFloat(e.surprisePercentage) || 0,
122
+ }));
123
+
124
+ const result: EarningsData = { symbol, quarterly };
125
+ cache.set(cacheKey, result, TTL.FUNDAMENTALS);
126
+ return result;
127
+ } catch (error) {
128
+ throwIfAuthError(error);
129
+ const stale = cache.getStale<EarningsData>(cacheKey, STALE_LIMIT.FUNDAMENTALS);
130
+ if (stale) return stale.value;
131
+ throw error;
132
+ }
133
+ }
134
+
135
+ export async function getFinancials(
136
+ symbol: string,
137
+ apiKey: string,
138
+ ): Promise<FinancialStatement[]> {
139
+ const cacheKey = `av:financials:${symbol}`;
140
+ const cached = cache.get<FinancialStatement[]>(cacheKey);
141
+ if (cached) return cached;
142
+
143
+ try {
144
+ // Fetch sequentially to respect Alpha Vantage rate limits (5 req/min free tier)
145
+ const incomeData = await fetchStatement<{ annualReports: any[] }>("INCOME_STATEMENT", symbol, apiKey);
146
+ const balanceData = await fetchStatement<{ annualReports: any[] }>("BALANCE_SHEET", symbol, apiKey);
147
+ const cashFlowData = await fetchStatement<{ annualReports: any[] }>("CASH_FLOW", symbol, apiKey);
148
+
149
+ const incomeReports = incomeData.annualReports ?? [];
150
+ const balanceReports = balanceData.annualReports ?? [];
151
+ const cashFlowReports = cashFlowData.annualReports ?? [];
152
+
153
+ // Index balance sheet and cash flow by fiscal date for merging
154
+ const balanceByDate = new Map(
155
+ balanceReports.map((r: any) => [r.fiscalDateEnding, r]),
156
+ );
157
+ const cashFlowByDate = new Map(
158
+ cashFlowReports.map((r: any) => [r.fiscalDateEnding, r]),
159
+ );
160
+
161
+ const statements = incomeReports.slice(0, 4).map((r: any) => {
162
+ const balance = balanceByDate.get(r.fiscalDateEnding) ?? {};
163
+ const cf = cashFlowByDate.get(r.fiscalDateEnding) ?? {};
164
+ const opCashFlow = parseNum(cf.operatingCashflow);
165
+ const capex = parseNum(cf.capitalExpenditures);
166
+
167
+ const totalDebt = parseNum(balance.shortLongTermDebtTotal);
168
+ const cash = parseNum(balance.cashAndCashEquivalentsAtCarryingValue);
169
+
170
+ return {
171
+ fiscalDate: r.fiscalDateEnding,
172
+ revenue: parseNum(r.totalRevenue),
173
+ grossProfit: parseNum(r.grossProfit),
174
+ operatingIncome: parseNum(r.operatingIncome),
175
+ netIncome: parseNum(r.netIncome),
176
+ eps: parseFloat(r.reportedEPS) || 0,
177
+ totalAssets: parseNum(balance.totalAssets),
178
+ totalLiabilities: parseNum(balance.totalLiabilities),
179
+ totalEquity: parseNum(balance.totalShareholderEquity),
180
+ operatingCashFlow: opCashFlow,
181
+ freeCashFlow: opCashFlow - capex,
182
+ totalDebt: totalDebt || undefined,
183
+ cashAndEquivalents: cash || undefined,
184
+ };
185
+ });
186
+
187
+ cache.set(cacheKey, statements, TTL.FUNDAMENTALS);
188
+ return statements;
189
+ } catch (error) {
190
+ throwIfAuthError(error);
191
+ const stale = cache.getStale<FinancialStatement[]>(cacheKey, STALE_LIMIT.FUNDAMENTALS);
192
+ if (stale) return stale.value;
193
+ throw error;
194
+ }
195
+ }
196
+
197
+ async function fetchStatement<T>(fn: string, symbol: string, apiKey: string): Promise<T> {
198
+ await rateLimiter.acquire("alphavantage");
199
+ const url = buildUrl(fn, { symbol }, apiKey);
200
+ const data = await httpGet<T>(url);
201
+ throwIfApiMessage(data);
202
+ return data;
203
+ }
204
+
205
+ export async function getGlobalQuote(
206
+ symbol: string,
207
+ apiKey: string,
208
+ ): Promise<StockQuote> {
209
+ const cacheKey = `av:globalquote:${symbol}`;
210
+ const cached = cache.get<StockQuote>(cacheKey);
211
+ if (cached) return cached;
212
+
213
+ try {
214
+ await rateLimiter.acquire("alphavantage");
215
+
216
+ const url = buildUrl("GLOBAL_QUOTE", { symbol }, apiKey);
217
+ const data = await httpGet<{ "Global Quote": Record<string, string> }>(url);
218
+ throwIfApiMessage(data);
219
+ const gq = data["Global Quote"];
220
+
221
+ if (!gq || !gq["05. price"]) {
222
+ throw new Error(`Alpha Vantage: No quote data for ${symbol}`);
223
+ }
224
+
225
+ const price = parseFloat(gq["05. price"]) || 0;
226
+ const result: StockQuote = {
227
+ symbol: gq["01. symbol"] ?? symbol,
228
+ price,
229
+ change: parseFloat(gq["09. change"]) || 0,
230
+ changePercent: parseFloat(gq["10. change percent"]?.replace("%", "")) || 0,
231
+ open: parseFloat(gq["02. open"]) || 0,
232
+ high: parseFloat(gq["03. high"]) || 0,
233
+ low: parseFloat(gq["04. low"]) || 0,
234
+ previousClose: parseFloat(gq["08. previous close"]) || 0,
235
+ volume: parseInt(gq["06. volume"], 10) || 0,
236
+ marketCap: 0, // Not available from GLOBAL_QUOTE
237
+ pe: null, // Not available from GLOBAL_QUOTE
238
+ week52High: 0, // Not available from GLOBAL_QUOTE
239
+ week52Low: 0, // Not available from GLOBAL_QUOTE
240
+ timestamp: Date.now(),
241
+ };
242
+
243
+ cache.set(cacheKey, result, TTL.QUOTE);
244
+ return result;
245
+ } catch (error) {
246
+ throwIfAuthError(error);
247
+ const stale = cache.getStale<StockQuote>(cacheKey, STALE_LIMIT.QUOTE);
248
+ if (stale) return stale.value;
249
+ throw error;
250
+ }
251
+ }
252
+
253
+ export async function getDailyHistory(
254
+ symbol: string,
255
+ apiKey: string,
256
+ range: string = "6mo",
257
+ ): Promise<OHLCV[]> {
258
+ const cacheKey = `av:daily:${symbol}:${range}`;
259
+ const cached = cache.get<OHLCV[]>(cacheKey);
260
+ if (cached) return cached;
261
+
262
+ try {
263
+ await rateLimiter.acquire("alphavantage");
264
+
265
+ // compact = last 100 data points, full = full 20+ year history
266
+ const daysNeeded = rangeToDays(range);
267
+ const outputsize = daysNeeded > 100 ? "full" : "compact";
268
+ const url = buildUrl("TIME_SERIES_DAILY", { symbol, outputsize }, apiKey);
269
+ const data = await httpGet<{ "Time Series (Daily)": Record<string, Record<string, string>> }>(url);
270
+ throwIfApiMessage(data);
271
+
272
+ const timeSeries = data["Time Series (Daily)"];
273
+ if (!timeSeries) {
274
+ throw new Error(`Alpha Vantage: No daily history for ${symbol}`);
275
+ }
276
+
277
+ const ohlcv: OHLCV[] = Object.entries(timeSeries)
278
+ .map(([date, bar]) => ({
279
+ date,
280
+ open: parseFloat(bar["1. open"]) || 0,
281
+ high: parseFloat(bar["2. high"]) || 0,
282
+ low: parseFloat(bar["3. low"]) || 0,
283
+ close: parseFloat(bar["4. close"]) || 0,
284
+ volume: parseInt(bar["5. volume"], 10) || 0,
285
+ }))
286
+ .sort((a, b) => a.date.localeCompare(b.date))
287
+ .slice(-daysNeeded);
288
+
289
+ cache.set(cacheKey, ohlcv, TTL.HISTORY);
290
+ return ohlcv;
291
+ } catch (error) {
292
+ throwIfAuthError(error);
293
+ const stale = cache.getStale<OHLCV[]>(cacheKey, STALE_LIMIT.HISTORY);
294
+ if (stale) return stale.value;
295
+ throw error;
296
+ }
297
+ }
298
+
299
+ function rangeToDays(range: string): number {
300
+ const map: Record<string, number> = {
301
+ "1d": 1, "5d": 5, "1mo": 22, "3mo": 66, "6mo": 130,
302
+ "1y": 252, "2y": 504, "5y": 1260, "max": 5000,
303
+ };
304
+ return map[range] ?? 130;
305
+ }
306
+
307
+ function parseNum(s: string | undefined): number {
308
+ return parseFloat(s ?? "0") || 0;
309
+ }
310
+
311
+ function parseNullableNum(s: string | undefined): number | null {
312
+ if (!s || s === "None" || s === "-") return null;
313
+ const n = parseFloat(s);
314
+ return isNaN(n) ? null : n;
315
+ }
@@ -0,0 +1,96 @@
1
+ import { httpGet } from "../infra/http-client.js";
2
+ import { cache, TTL, STALE_LIMIT } from "../infra/cache.js";
3
+ import { rateLimiter } from "../infra/rate-limiter.js";
4
+ import type { CryptoPrice, OHLCV } from "../types/market.js";
5
+
6
+ const BASE_URL = "https://api.coingecko.com/api/v3";
7
+
8
+ interface CoinGeckoDetailResponse {
9
+ id: string;
10
+ symbol: string;
11
+ name: string;
12
+ market_data: {
13
+ current_price: { usd: number };
14
+ price_change_24h: number;
15
+ price_change_percentage_24h: number;
16
+ market_cap: { usd: number };
17
+ total_volume: { usd: number };
18
+ high_24h: { usd: number };
19
+ low_24h: { usd: number };
20
+ ath: { usd: number };
21
+ ath_date: { usd: string };
22
+ circulating_supply: number;
23
+ total_supply: number | null;
24
+ };
25
+ }
26
+
27
+ export async function getCryptoPrice(id: string): Promise<CryptoPrice> {
28
+ const cacheKey = `coingecko:price:${id}`;
29
+ const cached = cache.get<CryptoPrice>(cacheKey);
30
+ if (cached) return cached;
31
+
32
+ try {
33
+ await rateLimiter.acquire("coingecko");
34
+
35
+ const url = `${BASE_URL}/coins/${encodeURIComponent(id)}?localization=false&tickers=false&community_data=false&developer_data=false`;
36
+ const data = await httpGet<CoinGeckoDetailResponse>(url);
37
+
38
+ const md = data.market_data;
39
+ const result: CryptoPrice = {
40
+ id: data.id,
41
+ symbol: data.symbol,
42
+ name: data.name,
43
+ price: md.current_price.usd,
44
+ change24h: md.price_change_24h,
45
+ changePercent24h: md.price_change_percentage_24h,
46
+ marketCap: md.market_cap.usd,
47
+ volume24h: md.total_volume.usd,
48
+ high24h: md.high_24h.usd,
49
+ low24h: md.low_24h.usd,
50
+ ath: md.ath.usd,
51
+ athDate: md.ath_date.usd,
52
+ circulatingSupply: md.circulating_supply,
53
+ totalSupply: md.total_supply,
54
+ timestamp: Date.now(),
55
+ };
56
+
57
+ cache.set(cacheKey, result, TTL.QUOTE);
58
+ return result;
59
+ } catch (error) {
60
+ const stale = cache.getStale<CryptoPrice>(cacheKey, STALE_LIMIT.QUOTE);
61
+ if (stale) return stale.value;
62
+ throw error;
63
+ }
64
+ }
65
+
66
+ export async function getCryptoHistory(
67
+ id: string,
68
+ days: number = 180,
69
+ ): Promise<OHLCV[]> {
70
+ const cacheKey = `coingecko:history:${id}:${days}`;
71
+ const cached = cache.get<OHLCV[]>(cacheKey);
72
+ if (cached) return cached;
73
+
74
+ try {
75
+ await rateLimiter.acquire("coingecko");
76
+
77
+ const url = `${BASE_URL}/coins/${encodeURIComponent(id)}/ohlc?vs_currency=usd&days=${days}`;
78
+ const data = await httpGet<number[][]>(url);
79
+
80
+ const ohlcv: OHLCV[] = data.map(([ts, open, high, low, close]) => ({
81
+ date: new Date(ts).toISOString().split("T")[0],
82
+ open,
83
+ high,
84
+ low,
85
+ close,
86
+ volume: 0, // OHLC endpoint doesn't include volume
87
+ }));
88
+
89
+ cache.set(cacheKey, ohlcv, TTL.HISTORY);
90
+ return ohlcv;
91
+ } catch (error) {
92
+ const stale = cache.getStale<OHLCV[]>(cacheKey, STALE_LIMIT.HISTORY);
93
+ if (stale) return stale.value;
94
+ throw error;
95
+ }
96
+ }