opencandle 0.2.0 → 0.4.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 (401) hide show
  1. package/README.md +110 -87
  2. package/assets/logo.svg +187 -0
  3. package/dist/analysts/orchestrator.js +1 -2
  4. package/dist/analysts/orchestrator.js.map +1 -1
  5. package/dist/cli.d.ts +1 -1
  6. package/dist/cli.js +38 -2
  7. package/dist/cli.js.map +1 -1
  8. package/dist/config.d.ts +34 -5
  9. package/dist/config.js +29 -8
  10. package/dist/config.js.map +1 -1
  11. package/dist/infra/browser.d.ts +10 -0
  12. package/dist/infra/browser.js +1 -0
  13. package/dist/infra/browser.js.map +1 -1
  14. package/dist/infra/cache.d.ts +4 -0
  15. package/dist/infra/cache.js +4 -0
  16. package/dist/infra/cache.js.map +1 -1
  17. package/dist/infra/native-dependencies.d.ts +1 -0
  18. package/dist/infra/native-dependencies.js +10 -0
  19. package/dist/infra/native-dependencies.js.map +1 -0
  20. package/dist/infra/node-version.d.ts +2 -0
  21. package/dist/infra/node-version.js +23 -0
  22. package/dist/infra/node-version.js.map +1 -0
  23. package/dist/infra/rate-limiter.js +6 -0
  24. package/dist/infra/rate-limiter.js.map +1 -1
  25. package/dist/memory/index.d.ts +2 -0
  26. package/dist/memory/index.js +1 -0
  27. package/dist/memory/index.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 +6 -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/onboarding/connect.d.ts +35 -0
  37. package/dist/onboarding/connect.js +118 -0
  38. package/dist/onboarding/connect.js.map +1 -0
  39. package/dist/onboarding/credential-interceptor.d.ts +44 -0
  40. package/dist/onboarding/credential-interceptor.js +72 -0
  41. package/dist/onboarding/credential-interceptor.js.map +1 -0
  42. package/dist/onboarding/degradation-accumulator.d.ts +21 -0
  43. package/dist/onboarding/degradation-accumulator.js +55 -0
  44. package/dist/onboarding/degradation-accumulator.js.map +1 -0
  45. package/dist/onboarding/prompt-user.d.ts +23 -0
  46. package/dist/onboarding/prompt-user.js +61 -0
  47. package/dist/onboarding/prompt-user.js.map +1 -0
  48. package/dist/onboarding/providers.d.ts +116 -0
  49. package/dist/onboarding/providers.js +239 -0
  50. package/dist/onboarding/providers.js.map +1 -0
  51. package/dist/onboarding/state.d.ts +31 -2
  52. package/dist/onboarding/state.js +141 -13
  53. package/dist/onboarding/state.js.map +1 -1
  54. package/dist/onboarding/tool-helpers.d.ts +34 -0
  55. package/dist/onboarding/tool-helpers.js +80 -0
  56. package/dist/onboarding/tool-helpers.js.map +1 -0
  57. package/dist/onboarding/tool-tags.d.ts +37 -0
  58. package/dist/onboarding/tool-tags.js +149 -0
  59. package/dist/onboarding/tool-tags.js.map +1 -0
  60. package/dist/onboarding/validation.d.ts +19 -0
  61. package/dist/onboarding/validation.js +117 -0
  62. package/dist/onboarding/validation.js.map +1 -0
  63. package/dist/pi/opencandle-extension.d.ts +7 -1
  64. package/dist/pi/opencandle-extension.js +488 -13
  65. package/dist/pi/opencandle-extension.js.map +1 -1
  66. package/dist/pi/session-storage.d.ts +2 -0
  67. package/dist/pi/session-storage.js +5 -0
  68. package/dist/pi/session-storage.js.map +1 -0
  69. package/dist/pi/session.d.ts +4 -1
  70. package/dist/pi/session.js +25 -3
  71. package/dist/pi/session.js.map +1 -1
  72. package/dist/pi/setup.d.ts +1 -2
  73. package/dist/pi/setup.js +67 -120
  74. package/dist/pi/setup.js.map +1 -1
  75. package/dist/pi/tool-adapter.d.ts +2 -2
  76. package/dist/pi/tool-adapter.js +14 -1
  77. package/dist/pi/tool-adapter.js.map +1 -1
  78. package/dist/prompts/context-builder.d.ts +22 -0
  79. package/dist/prompts/context-builder.js +47 -11
  80. package/dist/prompts/context-builder.js.map +1 -1
  81. package/dist/prompts/disclaimer.d.ts +6 -0
  82. package/dist/prompts/disclaimer.js +9 -0
  83. package/dist/prompts/disclaimer.js.map +1 -0
  84. package/dist/prompts/workflow-prompts.d.ts +8 -0
  85. package/dist/prompts/workflow-prompts.js +39 -5
  86. package/dist/prompts/workflow-prompts.js.map +1 -1
  87. package/dist/providers/alpha-vantage.js +20 -1
  88. package/dist/providers/alpha-vantage.js.map +1 -1
  89. package/dist/providers/exa-search.d.ts +39 -0
  90. package/dist/providers/exa-search.js +276 -0
  91. package/dist/providers/exa-search.js.map +1 -0
  92. package/dist/providers/finnhub.d.ts +17 -0
  93. package/dist/providers/finnhub.js +94 -0
  94. package/dist/providers/finnhub.js.map +1 -0
  95. package/dist/providers/fred.js +13 -1
  96. package/dist/providers/fred.js.map +1 -1
  97. package/dist/providers/index.d.ts +2 -0
  98. package/dist/providers/index.js +1 -0
  99. package/dist/providers/index.js.map +1 -1
  100. package/dist/providers/provider-credential-error.d.ts +8 -0
  101. package/dist/providers/provider-credential-error.js +22 -0
  102. package/dist/providers/provider-credential-error.js.map +1 -0
  103. package/dist/providers/reddit.d.ts +8 -0
  104. package/dist/providers/reddit.js +36 -9
  105. package/dist/providers/reddit.js.map +1 -1
  106. package/dist/providers/twitter.js +2 -8
  107. package/dist/providers/twitter.js.map +1 -1
  108. package/dist/providers/web-search.d.ts +17 -0
  109. package/dist/providers/web-search.js +224 -0
  110. package/dist/providers/web-search.js.map +1 -0
  111. package/dist/providers/wrap-provider.d.ts +7 -0
  112. package/dist/providers/wrap-provider.js +15 -0
  113. package/dist/providers/wrap-provider.js.map +1 -1
  114. package/dist/providers/yahoo-finance.js +70 -33
  115. package/dist/providers/yahoo-finance.js.map +1 -1
  116. package/dist/routing/classify-intent.js +22 -0
  117. package/dist/routing/classify-intent.js.map +1 -1
  118. package/dist/routing/defaults.js +1 -1
  119. package/dist/routing/defaults.js.map +1 -1
  120. package/dist/routing/index.d.ts +4 -0
  121. package/dist/routing/index.js +3 -0
  122. package/dist/routing/index.js.map +1 -1
  123. package/dist/routing/router-llm-client.d.ts +11 -0
  124. package/dist/routing/router-llm-client.js +42 -0
  125. package/dist/routing/router-llm-client.js.map +1 -0
  126. package/dist/routing/router-prompt.d.ts +2 -0
  127. package/dist/routing/router-prompt.js +138 -0
  128. package/dist/routing/router-prompt.js.map +1 -0
  129. package/dist/routing/router-types.d.ts +62 -0
  130. package/dist/routing/router-types.js +2 -0
  131. package/dist/routing/router-types.js.map +1 -0
  132. package/dist/routing/router.d.ts +10 -0
  133. package/dist/routing/router.js +194 -0
  134. package/dist/routing/router.js.map +1 -0
  135. package/dist/runtime/session-coordinator.d.ts +63 -4
  136. package/dist/runtime/session-coordinator.js +155 -4
  137. package/dist/runtime/session-coordinator.js.map +1 -1
  138. package/dist/runtime/tool-defaults-wrapper.d.ts +3 -0
  139. package/dist/runtime/tool-defaults-wrapper.js +25 -0
  140. package/dist/runtime/tool-defaults-wrapper.js.map +1 -0
  141. package/dist/sentiment/adapters/finnhub.d.ts +7 -0
  142. package/dist/sentiment/adapters/finnhub.js +39 -0
  143. package/dist/sentiment/adapters/finnhub.js.map +1 -0
  144. package/dist/sentiment/adapters/reddit.d.ts +11 -0
  145. package/dist/sentiment/adapters/reddit.js +54 -0
  146. package/dist/sentiment/adapters/reddit.js.map +1 -0
  147. package/dist/sentiment/adapters/twitter.d.ts +9 -0
  148. package/dist/sentiment/adapters/twitter.js +32 -0
  149. package/dist/sentiment/adapters/twitter.js.map +1 -0
  150. package/dist/sentiment/adapters/web.d.ts +9 -0
  151. package/dist/sentiment/adapters/web.js +40 -0
  152. package/dist/sentiment/adapters/web.js.map +1 -0
  153. package/dist/sentiment/index.d.ts +16 -0
  154. package/dist/sentiment/index.js +44 -0
  155. package/dist/sentiment/index.js.map +1 -0
  156. package/dist/sentiment/keywords.d.ts +2 -0
  157. package/dist/sentiment/keywords.js +9 -0
  158. package/dist/sentiment/keywords.js.map +1 -0
  159. package/dist/sentiment/pipeline.d.ts +9 -0
  160. package/dist/sentiment/pipeline.js +57 -0
  161. package/dist/sentiment/pipeline.js.map +1 -0
  162. package/dist/sentiment/scorer.d.ts +9 -0
  163. package/dist/sentiment/scorer.js +64 -0
  164. package/dist/sentiment/scorer.js.map +1 -0
  165. package/dist/sentiment/store.d.ts +24 -0
  166. package/dist/sentiment/store.js +182 -0
  167. package/dist/sentiment/store.js.map +1 -0
  168. package/dist/sentiment/trends.d.ts +13 -0
  169. package/dist/sentiment/trends.js +73 -0
  170. package/dist/sentiment/trends.js.map +1 -0
  171. package/dist/sentiment/types.d.ts +66 -0
  172. package/dist/sentiment/types.js +54 -0
  173. package/dist/sentiment/types.js.map +1 -0
  174. package/dist/system-prompt.js +29 -13
  175. package/dist/system-prompt.js.map +1 -1
  176. package/dist/tool-kit.d.ts +4 -4
  177. package/dist/tools/fundamentals/company-overview.d.ts +4 -2
  178. package/dist/tools/fundamentals/company-overview.js +27 -27
  179. package/dist/tools/fundamentals/company-overview.js.map +1 -1
  180. package/dist/tools/fundamentals/comps.d.ts +1 -1
  181. package/dist/tools/fundamentals/comps.js +45 -45
  182. package/dist/tools/fundamentals/comps.js.map +1 -1
  183. package/dist/tools/fundamentals/dcf.d.ts +1 -1
  184. package/dist/tools/fundamentals/dcf.js +82 -82
  185. package/dist/tools/fundamentals/dcf.js.map +1 -1
  186. package/dist/tools/fundamentals/earnings.d.ts +4 -2
  187. package/dist/tools/fundamentals/earnings.js +25 -25
  188. package/dist/tools/fundamentals/earnings.js.map +1 -1
  189. package/dist/tools/fundamentals/financials.d.ts +4 -2
  190. package/dist/tools/fundamentals/financials.js +23 -23
  191. package/dist/tools/fundamentals/financials.js.map +1 -1
  192. package/dist/tools/fundamentals/sec-filings.d.ts +1 -1
  193. package/dist/tools/index.d.ts +28 -1
  194. package/dist/tools/index.js +35 -2
  195. package/dist/tools/index.js.map +1 -1
  196. package/dist/tools/interaction/ask-user.d.ts +1 -1
  197. package/dist/tools/interaction/ask-user.js +28 -64
  198. package/dist/tools/interaction/ask-user.js.map +1 -1
  199. package/dist/tools/interaction/twitter-login.d.ts +1 -1
  200. package/dist/tools/macro/fear-greed.d.ts +1 -1
  201. package/dist/tools/macro/fred-data.d.ts +4 -2
  202. package/dist/tools/macro/fred-data.js +26 -26
  203. package/dist/tools/macro/fred-data.js.map +1 -1
  204. package/dist/tools/market/crypto-history.d.ts +1 -1
  205. package/dist/tools/market/crypto-price.d.ts +1 -1
  206. package/dist/tools/market/search-ticker.d.ts +1 -1
  207. package/dist/tools/market/stock-history.d.ts +1 -1
  208. package/dist/tools/market/stock-quote.d.ts +1 -1
  209. package/dist/tools/options/option-chain.d.ts +1 -1
  210. package/dist/tools/options/option-chain.js +4 -1
  211. package/dist/tools/options/option-chain.js.map +1 -1
  212. package/dist/tools/portfolio/correlation.d.ts +1 -1
  213. package/dist/tools/portfolio/predictions.d.ts +1 -1
  214. package/dist/tools/portfolio/risk-analysis.d.ts +1 -1
  215. package/dist/tools/portfolio/tracker.d.ts +1 -1
  216. package/dist/tools/portfolio/watchlist.d.ts +1 -1
  217. package/dist/tools/sentiment/reddit-sentiment.d.ts +4 -2
  218. package/dist/tools/sentiment/reddit-sentiment.js +107 -22
  219. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  220. package/dist/tools/sentiment/sentiment-summary.d.ts +7 -0
  221. package/dist/tools/sentiment/sentiment-summary.js +230 -0
  222. package/dist/tools/sentiment/sentiment-summary.js.map +1 -0
  223. package/dist/tools/sentiment/sentiment-trend.d.ts +22 -0
  224. package/dist/tools/sentiment/sentiment-trend.js +39 -0
  225. package/dist/tools/sentiment/sentiment-trend.js.map +1 -0
  226. package/dist/tools/sentiment/twitter-sentiment.d.ts +1 -1
  227. package/dist/tools/sentiment/twitter-sentiment.js +17 -0
  228. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
  229. package/dist/tools/sentiment/web-search.d.ts +11 -0
  230. package/dist/tools/sentiment/web-search.js +115 -0
  231. package/dist/tools/sentiment/web-search.js.map +1 -0
  232. package/dist/tools/sentiment/web-sentiment.d.ts +8 -0
  233. package/dist/tools/sentiment/web-sentiment.js +66 -0
  234. package/dist/tools/sentiment/web-sentiment.js.map +1 -0
  235. package/dist/tools/technical/backtest.d.ts +1 -1
  236. package/dist/tools/technical/indicators.d.ts +1 -1
  237. package/dist/tools/technical/indicators.js +7 -1
  238. package/dist/tools/technical/indicators.js.map +1 -1
  239. package/dist/types/index.d.ts +1 -1
  240. package/dist/types/sentiment.d.ts +21 -0
  241. package/dist/workflows/options-screener.js +7 -2
  242. package/dist/workflows/options-screener.js.map +1 -1
  243. package/dist/workflows/portfolio-builder.js +3 -3
  244. package/dist/workflows/portfolio-builder.js.map +1 -1
  245. package/gui/server/background-quotes.ts +31 -0
  246. package/gui/server/chat-event-adapter.ts +142 -0
  247. package/gui/server/invoke-tool.ts +89 -0
  248. package/gui/server/live-chat-event-adapter.ts +181 -0
  249. package/gui/server/model-setup.ts +100 -0
  250. package/gui/server/package.json +5 -0
  251. package/gui/server/projector.ts +212 -0
  252. package/gui/server/server.ts +592 -0
  253. package/gui/server/session-actions.ts +31 -0
  254. package/gui/server/tool-metadata.ts +88 -0
  255. package/gui/server/websocket.ts +128 -0
  256. package/gui/server/writer-lock.ts +118 -0
  257. package/gui/shared/chat-events.ts +118 -0
  258. package/gui/shared/event-reducer.ts +186 -0
  259. package/gui/web/dist/assets/CatalogOverlay-D1ImSJTe.js +1 -0
  260. package/gui/web/dist/assets/index-DBrWq43L.css +1 -0
  261. package/gui/web/dist/assets/index-RflHaj0y.js +67 -0
  262. package/gui/web/dist/assets/logo-CWpt6Y2a.svg +187 -0
  263. package/gui/web/dist/index.html +17 -0
  264. package/package.json +62 -20
  265. package/src/analysts/contracts.ts +189 -0
  266. package/src/analysts/orchestrator.ts +300 -0
  267. package/src/cli.ts +205 -0
  268. package/src/config.ts +161 -0
  269. package/src/index.ts +5 -0
  270. package/src/infra/browser.ts +111 -0
  271. package/src/infra/cache.ts +103 -0
  272. package/src/infra/http-client.ts +68 -0
  273. package/src/infra/index.ts +18 -0
  274. package/src/infra/native-dependencies.ts +12 -0
  275. package/src/infra/node-version.ts +24 -0
  276. package/src/infra/open-url.ts +28 -0
  277. package/src/infra/opencandle-paths.ts +64 -0
  278. package/src/infra/rate-limiter.ts +64 -0
  279. package/src/memory/index.ts +10 -0
  280. package/src/memory/manager.ts +159 -0
  281. package/src/memory/preference-extractor.ts +106 -0
  282. package/src/memory/retrieval.ts +70 -0
  283. package/src/memory/sqlite.ts +172 -0
  284. package/src/memory/storage.ts +204 -0
  285. package/src/memory/tool-defaults.ts +87 -0
  286. package/src/memory/types.ts +67 -0
  287. package/src/onboarding/connect.ts +184 -0
  288. package/src/onboarding/credential-interceptor.ts +134 -0
  289. package/src/onboarding/degradation-accumulator.ts +79 -0
  290. package/src/onboarding/prompt-user.ts +85 -0
  291. package/src/onboarding/providers.ts +315 -0
  292. package/src/onboarding/state.ts +218 -0
  293. package/src/onboarding/tool-helpers.ts +111 -0
  294. package/src/onboarding/tool-tags.ts +201 -0
  295. package/src/onboarding/validation.ts +158 -0
  296. package/src/pi/opencandle-extension.ts +724 -0
  297. package/src/pi/session-storage.ts +5 -0
  298. package/src/pi/session.ts +81 -0
  299. package/src/pi/setup.ts +371 -0
  300. package/src/pi/tool-adapter.ts +36 -0
  301. package/src/prompts/context-builder.ts +204 -0
  302. package/src/prompts/disclaimer.ts +9 -0
  303. package/src/prompts/sections.ts +46 -0
  304. package/src/prompts/workflow-prompts.ts +279 -0
  305. package/src/providers/alpha-vantage.ts +292 -0
  306. package/src/providers/coingecko.ts +96 -0
  307. package/src/providers/exa-search.ts +373 -0
  308. package/src/providers/fear-greed.ts +45 -0
  309. package/src/providers/finnhub.ts +124 -0
  310. package/src/providers/fred.ts +83 -0
  311. package/src/providers/index.ts +9 -0
  312. package/src/providers/provider-credential-error.ts +23 -0
  313. package/src/providers/reddit.ts +151 -0
  314. package/src/providers/sec-edgar.ts +96 -0
  315. package/src/providers/twitter.ts +173 -0
  316. package/src/providers/web-search.ts +293 -0
  317. package/src/providers/with-fallback.ts +41 -0
  318. package/src/providers/wrap-provider.ts +64 -0
  319. package/src/providers/yahoo-finance.ts +367 -0
  320. package/src/routing/classify-intent.ts +194 -0
  321. package/src/routing/defaults.ts +29 -0
  322. package/src/routing/entity-extractor.ts +140 -0
  323. package/src/routing/index.ts +26 -0
  324. package/src/routing/router-llm-client.ts +51 -0
  325. package/src/routing/router-prompt.ts +159 -0
  326. package/src/routing/router-types.ts +66 -0
  327. package/src/routing/router.ts +213 -0
  328. package/src/routing/slot-resolver.ts +152 -0
  329. package/src/routing/types.ts +63 -0
  330. package/src/runtime/evidence.ts +77 -0
  331. package/src/runtime/index.ts +55 -0
  332. package/src/runtime/prompt-step.ts +75 -0
  333. package/src/runtime/provider-ids.ts +15 -0
  334. package/src/runtime/provider-tracker.ts +40 -0
  335. package/src/runtime/run-context.ts +22 -0
  336. package/src/runtime/session-coordinator.ts +406 -0
  337. package/src/runtime/tool-defaults-wrapper.ts +35 -0
  338. package/src/runtime/validation.ts +214 -0
  339. package/src/runtime/workflow-events.ts +75 -0
  340. package/src/runtime/workflow-runner.ts +188 -0
  341. package/src/runtime/workflow-types.ts +102 -0
  342. package/src/sentiment/adapters/finnhub.ts +44 -0
  343. package/src/sentiment/adapters/reddit.ts +65 -0
  344. package/src/sentiment/adapters/twitter.ts +36 -0
  345. package/src/sentiment/adapters/web.ts +44 -0
  346. package/src/sentiment/index.ts +58 -0
  347. package/src/sentiment/keywords.ts +9 -0
  348. package/src/sentiment/pipeline.ts +68 -0
  349. package/src/sentiment/scorer.ts +78 -0
  350. package/src/sentiment/store.ts +260 -0
  351. package/src/sentiment/trends.ts +90 -0
  352. package/src/sentiment/types.ts +108 -0
  353. package/src/system-prompt.ts +115 -0
  354. package/src/tool-kit.ts +68 -0
  355. package/src/tools/AGENTS.md +36 -0
  356. package/src/tools/fundamentals/company-overview.ts +54 -0
  357. package/src/tools/fundamentals/comps.ts +156 -0
  358. package/src/tools/fundamentals/dcf.ts +267 -0
  359. package/src/tools/fundamentals/earnings.ts +47 -0
  360. package/src/tools/fundamentals/financials.ts +54 -0
  361. package/src/tools/fundamentals/sec-filings.ts +61 -0
  362. package/src/tools/index.ts +88 -0
  363. package/src/tools/interaction/ask-user.ts +81 -0
  364. package/src/tools/interaction/twitter-login.ts +93 -0
  365. package/src/tools/macro/fear-greed.ts +41 -0
  366. package/src/tools/macro/fred-data.ts +54 -0
  367. package/src/tools/market/crypto-history.ts +51 -0
  368. package/src/tools/market/crypto-price.ts +53 -0
  369. package/src/tools/market/search-ticker.ts +53 -0
  370. package/src/tools/market/stock-history.ts +79 -0
  371. package/src/tools/market/stock-quote.ts +64 -0
  372. package/src/tools/options/greeks.ts +82 -0
  373. package/src/tools/options/option-chain.ts +91 -0
  374. package/src/tools/portfolio/correlation.ts +162 -0
  375. package/src/tools/portfolio/predictions.ts +253 -0
  376. package/src/tools/portfolio/risk-analysis.ts +134 -0
  377. package/src/tools/portfolio/tracker.ts +147 -0
  378. package/src/tools/portfolio/watchlist.ts +153 -0
  379. package/src/tools/sentiment/reddit-sentiment.ts +164 -0
  380. package/src/tools/sentiment/sentiment-summary.ts +256 -0
  381. package/src/tools/sentiment/sentiment-trend.ts +58 -0
  382. package/src/tools/sentiment/twitter-sentiment.ts +96 -0
  383. package/src/tools/sentiment/web-search.ts +150 -0
  384. package/src/tools/sentiment/web-sentiment.ts +76 -0
  385. package/src/tools/technical/backtest.ts +246 -0
  386. package/src/tools/technical/indicators.ts +258 -0
  387. package/src/types/fundamentals.ts +46 -0
  388. package/src/types/index.ts +20 -0
  389. package/src/types/macro.ts +27 -0
  390. package/src/types/market.ts +43 -0
  391. package/src/types/options.ts +35 -0
  392. package/src/types/portfolio.ts +41 -0
  393. package/src/types/sentiment.ts +70 -0
  394. package/src/workflows/compare-assets.ts +39 -0
  395. package/src/workflows/index.ts +4 -0
  396. package/src/workflows/options-screener.ts +49 -0
  397. package/src/workflows/portfolio-builder.ts +52 -0
  398. package/src/workflows/types.ts +4 -0
  399. package/dist/tools/sentiment/news-sentiment.d.ts +0 -7
  400. package/dist/tools/sentiment/news-sentiment.js +0 -55
  401. package/dist/tools/sentiment/news-sentiment.js.map +0 -1
@@ -0,0 +1,239 @@
1
+ // Provider registry — single source of truth for OpenCandle's credentialed
2
+ // third-party data providers. Every setup pathway iterates this registry:
3
+ // first-run startup, the `/connect` command, the `tool_result` credential
4
+ // interception handler, and the gap-note generator all read from here.
5
+ //
6
+ // Adding a new credentialed provider is a two-step change: add its `ProviderId`
7
+ // to the literal union below, and add its descriptor to the `PROVIDERS` array.
8
+ // The `satisfies` check ensures TypeScript fails the build if the union and
9
+ // the array ever disagree.
10
+ import { getConfig, loadFileConfig } from "../config.js";
11
+ // Declaration order matters: picker display order, per-workflow prompt priority,
12
+ // getProvidersByCategory/getProvidersByTier iteration order.
13
+ export const PROVIDERS = [
14
+ {
15
+ id: "alpha_vantage",
16
+ displayName: "Alpha Vantage",
17
+ category: "fundamentals",
18
+ tier: "hard",
19
+ aliases: ["financials", "fundamentals", "company-financials", "alphavantage"],
20
+ signupUrl: "https://www.alphavantage.co/support/#api-key",
21
+ freeTier: true,
22
+ envVar: "ALPHA_VANTAGE_API_KEY",
23
+ configPath: ["providers", "alphaVantage", "apiKey"],
24
+ unlocks: [
25
+ "company fundamentals",
26
+ "income/balance/cashflow statements",
27
+ "DCF valuation",
28
+ "earnings history",
29
+ ],
30
+ fallbackDescription: null,
31
+ snoozeDurationDays: 7,
32
+ instructionsHint: "Free, about 30 seconds, signup opens in your browser",
33
+ },
34
+ {
35
+ id: "fred",
36
+ displayName: "FRED",
37
+ category: "macro",
38
+ tier: "hard",
39
+ aliases: ["economy", "macro", "economic-data", "st-louis-fed"],
40
+ signupUrl: "https://fredaccount.stlouisfed.org/apikeys",
41
+ freeTier: true,
42
+ envVar: "FRED_API_KEY",
43
+ configPath: ["providers", "fred", "apiKey"],
44
+ unlocks: [
45
+ "interest rates",
46
+ "inflation data",
47
+ "yield curve",
48
+ "economic indicators",
49
+ ],
50
+ fallbackDescription: null,
51
+ snoozeDurationDays: 7,
52
+ instructionsHint: "Free, about 30 seconds, requires a St. Louis Fed account",
53
+ },
54
+ {
55
+ id: "finnhub",
56
+ displayName: "Finnhub",
57
+ category: "news",
58
+ tier: "soft",
59
+ aliases: ["news", "company-news", "finnhub-news"],
60
+ signupUrl: "https://finnhub.io/register",
61
+ freeTier: true,
62
+ envVar: "FINNHUB_API_KEY",
63
+ configPath: ["providers", "finnhub", "apiKey"],
64
+ unlocks: [
65
+ "ticker-tagged company news",
66
+ "sentiment enrichment with a dedicated news source",
67
+ ],
68
+ // Finnhub is a soft enrichment source — sentiment-summary continues to work
69
+ // with Twitter/Reddit/web search when Finnhub is missing. The fallback is
70
+ // "the other sentiment sources still run".
71
+ fallbackDescription: "Other sentiment sources (Reddit, Twitter, web search) continue to work without Finnhub",
72
+ snoozeDurationDays: 7,
73
+ instructionsHint: "Free, about 30 seconds, signup opens in your browser",
74
+ },
75
+ {
76
+ id: "brave",
77
+ displayName: "Brave Search",
78
+ category: "web_search",
79
+ tier: "soft",
80
+ aliases: ["brave", "brave-search"],
81
+ signupUrl: "https://brave.com/search/api/",
82
+ freeTier: true,
83
+ envVar: "BRAVE_API_KEY",
84
+ configPath: ["providers", "brave", "apiKey"],
85
+ unlocks: [
86
+ "tier-2 web search with freshness control",
87
+ "independent search index outside of DuckDuckGo",
88
+ ],
89
+ fallbackDescription: "Web search continues to work via DuckDuckGo (free, no key needed, lower-quality freshness)",
90
+ snoozeDurationDays: 7,
91
+ instructionsHint: "Free tier available, signup opens in your browser",
92
+ },
93
+ {
94
+ id: "exa",
95
+ displayName: "Exa",
96
+ category: "web_search",
97
+ tier: "soft",
98
+ // Note: "search" is a multi-provider alias shared with Brave. The registry
99
+ // exposes it via resolveProviderFromArgument as a sub-picker case, not as a
100
+ // single-provider alias — so it intentionally does NOT appear in either
101
+ // provider's `aliases` array here. Keeping aliases unique-per-provider lets
102
+ // resolveProviderFromArgument cleanly distinguish the alias case from the
103
+ // sub-picker case.
104
+ aliases: ["exa", "exa-search"],
105
+ signupUrl: "https://dashboard.exa.ai/",
106
+ freeTier: false,
107
+ envVar: "EXA_API_KEY",
108
+ configPath: ["providers", "exa", "apiKey"],
109
+ unlocks: [
110
+ "tier-1 semantic web search",
111
+ "full article text and highlights",
112
+ "higher freshness accuracy than DuckDuckGo",
113
+ ],
114
+ fallbackDescription: "Exa search continues to work via the keyless Exa MCP endpoint, which has lower rate limits but similar quality",
115
+ snoozeDurationDays: 7,
116
+ instructionsHint: "Paid with free tier, signup opens in your browser",
117
+ },
118
+ ];
119
+ // -----------------------------------------------------------------------------
120
+ // Lookup helpers
121
+ // -----------------------------------------------------------------------------
122
+ // Lazy-built index — populated on first helper call so module import has no
123
+ // side effects. The test `importing the registry does not read the filesystem`
124
+ // relies on this.
125
+ let providersById;
126
+ function byId() {
127
+ if (!providersById) {
128
+ providersById = new Map(PROVIDERS.map((p) => [p.id, p]));
129
+ }
130
+ return providersById;
131
+ }
132
+ export function listAllProviders() {
133
+ return PROVIDERS;
134
+ }
135
+ export function getProvider(id) {
136
+ const found = byId().get(id);
137
+ if (!found) {
138
+ throw new Error(`Unknown provider id: "${id}"`);
139
+ }
140
+ return found;
141
+ }
142
+ export function getProvidersByCategory(category) {
143
+ return PROVIDERS.filter((p) => p.category === category);
144
+ }
145
+ export function getProvidersByTier(tier) {
146
+ return PROVIDERS.filter((p) => p.tier === tier);
147
+ }
148
+ // -----------------------------------------------------------------------------
149
+ // Credential source helpers
150
+ // -----------------------------------------------------------------------------
151
+ function readConfigValueByPath(obj, path) {
152
+ let cursor = obj;
153
+ for (const segment of path) {
154
+ if (cursor && typeof cursor === "object" && segment in cursor) {
155
+ cursor = cursor[segment];
156
+ }
157
+ else {
158
+ return undefined;
159
+ }
160
+ }
161
+ return typeof cursor === "string" && cursor.length > 0 ? cursor : undefined;
162
+ }
163
+ // Provider-id → `Config` field mapping. `Config` (in src/config.ts) is the
164
+ // canonical env-or-file resolved shape that tool implementations already use.
165
+ // `hasCredential` reads from `getConfig()` so that tests mocking `getConfig`
166
+ // see a consistent view; `getCredentialSource` reads `process.env` +
167
+ // `loadFileConfig` directly because it needs to distinguish env from file.
168
+ const CONFIG_FIELD_BY_ID = {
169
+ alpha_vantage: "alphaVantageApiKey",
170
+ fred: "fredApiKey",
171
+ finnhub: "finnhubApiKey",
172
+ brave: "braveApiKey",
173
+ exa: "exaApiKey",
174
+ };
175
+ export function hasCredential(id) {
176
+ const field = CONFIG_FIELD_BY_ID[id];
177
+ const value = getConfig()[field];
178
+ return typeof value === "string" && value.length > 0;
179
+ }
180
+ export function getCredentialSource(id) {
181
+ return getCredential(id).source;
182
+ }
183
+ export function getCredential(id) {
184
+ const descriptor = getProvider(id);
185
+ const envValue = process.env[descriptor.envVar];
186
+ if (envValue && envValue.length > 0)
187
+ return { source: "env", value: envValue };
188
+ // Lazy file-config read — only invoked when env is absent.
189
+ const fileConfig = loadFileConfig();
190
+ const fileValue = readConfigValueByPath(fileConfig, descriptor.configPath);
191
+ if (fileValue)
192
+ return { source: "file", value: fileValue };
193
+ return { source: "absent" };
194
+ }
195
+ // -----------------------------------------------------------------------------
196
+ // /connect argument resolution
197
+ // -----------------------------------------------------------------------------
198
+ export function resolveProviderFromArgument(arg) {
199
+ const needle = arg.trim().toLowerCase();
200
+ if (!needle)
201
+ return undefined;
202
+ // 1. Exact provider id match (case-insensitive).
203
+ for (const p of PROVIDERS) {
204
+ if (p.id === needle)
205
+ return p;
206
+ }
207
+ // 2. Exact alias match.
208
+ for (const p of PROVIDERS) {
209
+ if (p.aliases.includes(needle))
210
+ return p;
211
+ }
212
+ // 3. Category match: if the needle matches a category name, return the
213
+ // providers in that category. One match → single descriptor. Multiple
214
+ // matches → array (triggers the sub-picker in the /connect handler).
215
+ const categories = [
216
+ "fundamentals",
217
+ "macro",
218
+ "news",
219
+ "web_search",
220
+ ];
221
+ const normalizedCategory = needle.replace("-", "_");
222
+ if (categories.includes(normalizedCategory)) {
223
+ const group = getProvidersByCategory(normalizedCategory);
224
+ if (group.length === 1)
225
+ return group[0];
226
+ if (group.length > 1)
227
+ return group;
228
+ }
229
+ // 4. Special shared alias: "search" → both web_search providers.
230
+ if (needle === "search" || needle === "web" || needle === "web-search") {
231
+ const searchProviders = getProvidersByCategory("web_search");
232
+ if (searchProviders.length === 1)
233
+ return searchProviders[0];
234
+ if (searchProviders.length > 1)
235
+ return searchProviders;
236
+ }
237
+ return undefined;
238
+ }
239
+ //# sourceMappingURL=providers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"providers.js","sourceRoot":"","sources":["../../src/onboarding/providers.ts"],"names":[],"mappings":"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"}
@@ -1,8 +1,37 @@
1
- export declare const ONBOARDING_VERSION = 1;
1
+ import type { ProviderId } from "./providers.js";
2
+ export declare const ONBOARDING_VERSION = 2;
3
+ export type ProviderOnboardingEntry = {
4
+ status: "completed";
5
+ lastPromptAt: string;
6
+ } | {
7
+ status: "snoozed";
8
+ lastPromptAt: string;
9
+ snoozeUntil: string;
10
+ } | {
11
+ status: "never_ask";
12
+ lastPromptAt: string;
13
+ };
2
14
  export interface OnboardingState {
3
15
  version: number;
4
- financeSetupStatus?: "dismissed" | "completed";
16
+ /** ISO 8601 timestamp the welcome message was first seeded, or undefined if never. */
17
+ welcomeShownAt?: string;
18
+ providers: Partial<Record<ProviderId, ProviderOnboardingEntry>>;
5
19
  }
6
20
  export declare function getDefaultOnboardingState(): OnboardingState;
7
21
  export declare function loadOnboardingState(path?: string): OnboardingState;
8
22
  export declare function saveOnboardingState(state: OnboardingState, path?: string): void;
23
+ export declare function markProviderCompleted(state: OnboardingState, id: ProviderId): OnboardingState;
24
+ export declare function markProviderSnoozed(state: OnboardingState, id: ProviderId, days: number): OnboardingState;
25
+ export declare function markProviderNeverAsk(state: OnboardingState, id: ProviderId): OnboardingState;
26
+ export declare function markWelcomeShown(state: OnboardingState): OnboardingState;
27
+ export declare function getProviderEntry(state: OnboardingState, id: ProviderId): ProviderOnboardingEntry | undefined;
28
+ export interface ShouldPromptOptions {
29
+ /**
30
+ * When true, treats a `completed` entry as eligible for re-prompt. This is
31
+ * how stale credentials (401 after a previously-connected key) get a fresh
32
+ * offer.
33
+ */
34
+ stale?: boolean;
35
+ }
36
+ export declare function shouldPrompt(state: OnboardingState, id: ProviderId, now: Date, options?: ShouldPromptOptions): boolean;
37
+ export declare function shouldShowWelcome(state: OnboardingState, hasUI: boolean): boolean;
@@ -1,31 +1,159 @@
1
+ // Onboarding state schema and pure helpers.
2
+ //
3
+ // Shape: one flag for the welcome message (`welcomeShownAt`) plus a partial
4
+ // per-provider map with a proper discriminated union for status + snooze.
5
+ //
6
+ // All transition helpers are PURE — they return a new state object rather
7
+ // than mutating. The caller is responsible for persisting via saveOnboardingState.
1
8
  import { existsSync, readFileSync, writeFileSync } from "node:fs";
2
9
  import { ensureParentDir, getOnboardingPath } from "../infra/opencandle-paths.js";
3
- export const ONBOARDING_VERSION = 1;
10
+ export const ONBOARDING_VERSION = 2;
4
11
  export function getDefaultOnboardingState() {
5
- return { version: ONBOARDING_VERSION };
12
+ return { version: ONBOARDING_VERSION, providers: {} };
13
+ }
14
+ // -----------------------------------------------------------------------------
15
+ // Load / save
16
+ // -----------------------------------------------------------------------------
17
+ const STATUS_VALUES = new Set(["completed", "snoozed", "never_ask"]);
18
+ function parseEntry(raw) {
19
+ if (!raw || typeof raw !== "object")
20
+ return undefined;
21
+ const obj = raw;
22
+ const status = obj.status;
23
+ if (typeof status !== "string" || !STATUS_VALUES.has(status))
24
+ return undefined;
25
+ const lastPromptAt = obj.lastPromptAt;
26
+ if (typeof lastPromptAt !== "string")
27
+ return undefined;
28
+ if (status === "snoozed") {
29
+ const snoozeUntil = obj.snoozeUntil;
30
+ if (typeof snoozeUntil !== "string")
31
+ return undefined;
32
+ return { status: "snoozed", lastPromptAt, snoozeUntil };
33
+ }
34
+ if (status === "completed") {
35
+ return { status: "completed", lastPromptAt };
36
+ }
37
+ return { status: "never_ask", lastPromptAt };
38
+ }
39
+ function parseProvidersMap(raw) {
40
+ if (!raw || typeof raw !== "object")
41
+ return {};
42
+ const result = {};
43
+ for (const [key, value] of Object.entries(raw)) {
44
+ const entry = parseEntry(value);
45
+ if (entry)
46
+ result[key] = entry;
47
+ }
48
+ return result;
6
49
  }
7
50
  export function loadOnboardingState(path = getOnboardingPath()) {
8
51
  if (!existsSync(path)) {
9
52
  return getDefaultOnboardingState();
10
53
  }
54
+ let raw;
11
55
  try {
12
- const parsed = JSON.parse(readFileSync(path, "utf-8"));
13
- if (!parsed || typeof parsed !== "object") {
14
- return getDefaultOnboardingState();
15
- }
16
- return {
17
- version: typeof parsed.version === "number" ? parsed.version : ONBOARDING_VERSION,
18
- financeSetupStatus: parsed.financeSetupStatus === "completed" || parsed.financeSetupStatus === "dismissed"
19
- ? parsed.financeSetupStatus
20
- : undefined,
21
- };
56
+ raw = readFileSync(path, "utf-8");
22
57
  }
23
58
  catch {
24
59
  return getDefaultOnboardingState();
25
60
  }
61
+ let parsed;
62
+ try {
63
+ parsed = JSON.parse(raw);
64
+ }
65
+ catch {
66
+ return getDefaultOnboardingState();
67
+ }
68
+ if (!parsed || typeof parsed !== "object") {
69
+ return getDefaultOnboardingState();
70
+ }
71
+ const obj = parsed;
72
+ const version = typeof obj.version === "number" ? obj.version : ONBOARDING_VERSION;
73
+ const welcomeShownAt = typeof obj.welcomeShownAt === "string" ? obj.welcomeShownAt : undefined;
74
+ const providers = parseProvidersMap(obj.providers);
75
+ return { version, welcomeShownAt, providers };
26
76
  }
27
77
  export function saveOnboardingState(state, path = getOnboardingPath()) {
28
78
  ensureParentDir(path);
29
- writeFileSync(path, `${JSON.stringify(state, null, 2)}\n`, "utf-8");
79
+ // Strip undefined fields for cleaner on-disk output.
80
+ const serializable = {
81
+ version: state.version,
82
+ providers: state.providers,
83
+ };
84
+ if (state.welcomeShownAt !== undefined) {
85
+ serializable.welcomeShownAt = state.welcomeShownAt;
86
+ }
87
+ writeFileSync(path, `${JSON.stringify(serializable, null, 2)}\n`, "utf-8");
88
+ }
89
+ // -----------------------------------------------------------------------------
90
+ // Pure transitions
91
+ // -----------------------------------------------------------------------------
92
+ function nowIso() {
93
+ return new Date().toISOString();
94
+ }
95
+ export function markProviderCompleted(state, id) {
96
+ return {
97
+ ...state,
98
+ providers: {
99
+ ...state.providers,
100
+ [id]: { status: "completed", lastPromptAt: nowIso() },
101
+ },
102
+ };
103
+ }
104
+ export function markProviderSnoozed(state, id, days) {
105
+ const now = new Date();
106
+ const snoozeUntil = new Date(now.getTime() + days * 24 * 3600 * 1000);
107
+ return {
108
+ ...state,
109
+ providers: {
110
+ ...state.providers,
111
+ [id]: {
112
+ status: "snoozed",
113
+ lastPromptAt: now.toISOString(),
114
+ snoozeUntil: snoozeUntil.toISOString(),
115
+ },
116
+ },
117
+ };
118
+ }
119
+ export function markProviderNeverAsk(state, id) {
120
+ return {
121
+ ...state,
122
+ providers: {
123
+ ...state.providers,
124
+ [id]: { status: "never_ask", lastPromptAt: nowIso() },
125
+ },
126
+ };
127
+ }
128
+ export function markWelcomeShown(state) {
129
+ return { ...state, welcomeShownAt: nowIso() };
130
+ }
131
+ // -----------------------------------------------------------------------------
132
+ // Pure queries
133
+ // -----------------------------------------------------------------------------
134
+ export function getProviderEntry(state, id) {
135
+ return state.providers[id];
136
+ }
137
+ export function shouldPrompt(state, id, now, options = {}) {
138
+ const entry = state.providers[id];
139
+ if (!entry)
140
+ return true;
141
+ switch (entry.status) {
142
+ case "never_ask":
143
+ return false;
144
+ case "completed":
145
+ return options.stale === true;
146
+ case "snoozed": {
147
+ const until = Date.parse(entry.snoozeUntil);
148
+ if (Number.isNaN(until))
149
+ return true;
150
+ return now.getTime() >= until;
151
+ }
152
+ }
153
+ }
154
+ export function shouldShowWelcome(state, hasUI) {
155
+ if (!hasUI)
156
+ return false;
157
+ return state.welcomeShownAt === undefined;
30
158
  }
31
159
  //# sourceMappingURL=state.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"state.js","sourceRoot":"","sources":["../../src/onboarding/state.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAClE,OAAO,EAAE,eAAe,EAAE,iBAAiB,EAAE,MAAM,8BAA8B,CAAC;AAElF,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,CAAC;AAOpC,MAAM,UAAU,yBAAyB;IACvC,OAAO,EAAE,OAAO,EAAE,kBAAkB,EAAE,CAAC;AACzC,CAAC;AAED,MAAM,UAAU,mBAAmB,CAAC,IAAI,GAAG,iBAAiB,EAAE;IAC5D,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,CAAC;QACtB,OAAO,yBAAyB,EAAE,CAAC;IACrC,CAAC;IAED,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAoB,CAAC;QAC1E,IAAI,CAAC,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC1C,OAAO,yBAAyB,EAAE,CAAC;QACrC,CAAC;QACD,OAAO;YACL,OAAO,EAAE,OAAO,MAAM,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,kBAAkB;YACjF,kBAAkB,EAChB,MAAM,CAAC,kBAAkB,KAAK,WAAW,IAAI,MAAM,CAAC,kBAAkB,KAAK,WAAW;gBACpF,CAAC,CAAC,MAAM,CAAC,kBAAkB;gBAC3B,CAAC,CAAC,SAAS;SAChB,CAAC;IACJ,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,yBAAyB,EAAE,CAAC;IACrC,CAAC;AACH,CAAC;AAED,MAAM,UAAU,mBAAmB,CAAC,KAAsB,EAAE,IAAI,GAAG,iBAAiB,EAAE;IACpF,eAAe,CAAC,IAAI,CAAC,CAAC;IACtB,aAAa,CAAC,IAAI,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;AACtE,CAAC"}
1
+ {"version":3,"file":"state.js","sourceRoot":"","sources":["../../src/onboarding/state.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,34 @@
1
+ import type { AgentToolResult } from "@earendil-works/pi-agent-core";
2
+ import { type ProviderId } from "./providers.js";
3
+ import { type CredentialRequiredReason } from "./tool-tags.js";
4
+ export interface CredentialRequiredDetails {
5
+ credentialRequired: {
6
+ provider: ProviderId;
7
+ reason: CredentialRequiredReason;
8
+ httpStatus?: number;
9
+ };
10
+ }
11
+ /**
12
+ * Wrap a tool's provider-calling logic with a credential-check layer.
13
+ *
14
+ * Usage:
15
+ * ```
16
+ * async execute(_, args) {
17
+ * return withCredentialCheck("alpha_vantage", async () => {
18
+ * const apiKey = getConfig().alphaVantageApiKey!;
19
+ * const result = await wrapProvider("alphavantage", () => getFinancials(args.symbol, apiKey));
20
+ * // ... format success result ...
21
+ * return { content: [...], details: [...] };
22
+ * });
23
+ * }
24
+ * ```
25
+ *
26
+ * On missing credential (upfront): `fn` is NOT called. A tagged tool result
27
+ * is returned directly.
28
+ *
29
+ * On `ProviderCredentialError` thrown during `fn`: the error is caught and a
30
+ * tagged tool result is returned.
31
+ *
32
+ * Any other thrown value (plain Error, string, etc.) is re-thrown unchanged.
33
+ */
34
+ export declare function withCredentialCheck<T>(providerId: ProviderId, fn: () => Promise<AgentToolResult<T>>): Promise<AgentToolResult<T> | AgentToolResult<CredentialRequiredDetails>>;
@@ -0,0 +1,80 @@
1
+ // Tool-boundary helpers for credentialed providers.
2
+ //
3
+ // `withCredentialCheck` is the single-entry helper that OpenCandle tools use
4
+ // to wrap their provider calls. It centralizes:
5
+ // 1. The upfront "credential missing" check via the registry.
6
+ // 2. The catch-and-convert logic for `ProviderCredentialError` thrown from
7
+ // providers on 401/403 (stale credential after HTTP call).
8
+ // 3. The tagged `[OPENCANDLE_CREDENTIAL_REQUIRED ...]` content emission that
9
+ // the Pi `tool_result` extension hook intercepts.
10
+ //
11
+ // Non-credential errors (network timeouts, parse errors, etc.) are re-thrown
12
+ // unchanged so existing error handling in `wrapProvider` continues to work.
13
+ import { ProviderCredentialError } from "../providers/provider-credential-error.js";
14
+ import { getProvider, hasCredential } from "./providers.js";
15
+ import { buildCredentialRequiredTag, } from "./tool-tags.js";
16
+ function buildCredentialRequiredResult(provider, reason, httpStatus) {
17
+ const descriptor = getProvider(provider);
18
+ const tag = buildCredentialRequiredTag({
19
+ provider,
20
+ reason,
21
+ unlocks: descriptor.unlocks,
22
+ fallback: descriptor.fallbackDescription,
23
+ httpStatus,
24
+ });
25
+ // The second line is natural language describing what's missing. Users won't
26
+ // see it directly — the model sees it and uses the information to synthesize
27
+ // a gap note in its final answer. The tagged first line is the machine-
28
+ // readable signal the `tool_result` interception handler keys off of.
29
+ const narrative = reason === "missing"
30
+ ? `${descriptor.displayName} was not fetched for this request because no API key is configured. It unlocks ${descriptor.unlocks.join(", ")}.`
31
+ : `${descriptor.displayName} rejected the configured API key (HTTP ${httpStatus ?? "auth error"}). It may be expired or revoked.`;
32
+ return {
33
+ content: [{ type: "text", text: `${tag}\n\n${narrative}` }],
34
+ details: {
35
+ credentialRequired: {
36
+ provider,
37
+ reason,
38
+ ...(httpStatus !== undefined ? { httpStatus } : {}),
39
+ },
40
+ },
41
+ };
42
+ }
43
+ /**
44
+ * Wrap a tool's provider-calling logic with a credential-check layer.
45
+ *
46
+ * Usage:
47
+ * ```
48
+ * async execute(_, args) {
49
+ * return withCredentialCheck("alpha_vantage", async () => {
50
+ * const apiKey = getConfig().alphaVantageApiKey!;
51
+ * const result = await wrapProvider("alphavantage", () => getFinancials(args.symbol, apiKey));
52
+ * // ... format success result ...
53
+ * return { content: [...], details: [...] };
54
+ * });
55
+ * }
56
+ * ```
57
+ *
58
+ * On missing credential (upfront): `fn` is NOT called. A tagged tool result
59
+ * is returned directly.
60
+ *
61
+ * On `ProviderCredentialError` thrown during `fn`: the error is caught and a
62
+ * tagged tool result is returned.
63
+ *
64
+ * Any other thrown value (plain Error, string, etc.) is re-thrown unchanged.
65
+ */
66
+ export async function withCredentialCheck(providerId, fn) {
67
+ if (!hasCredential(providerId)) {
68
+ return buildCredentialRequiredResult(providerId, "missing");
69
+ }
70
+ try {
71
+ return await fn();
72
+ }
73
+ catch (error) {
74
+ if (error instanceof ProviderCredentialError) {
75
+ return buildCredentialRequiredResult(error.provider, error.reason, error.httpStatus);
76
+ }
77
+ throw error;
78
+ }
79
+ }
80
+ //# sourceMappingURL=tool-helpers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tool-helpers.js","sourceRoot":"","sources":["../../src/onboarding/tool-helpers.ts"],"names":[],"mappings":"AAAA,oDAAoD;AACpD,EAAE;AACF,6EAA6E;AAC7E,gDAAgD;AAChD,gEAAgE;AAChE,6EAA6E;AAC7E,gEAAgE;AAChE,+EAA+E;AAC/E,uDAAuD;AACvD,EAAE;AACF,6EAA6E;AAC7E,4EAA4E;AAG5E,OAAO,EAAE,uBAAuB,EAAE,MAAM,2CAA2C,CAAC;AACpF,OAAO,EAAE,WAAW,EAAE,aAAa,EAAmB,MAAM,gBAAgB,CAAC;AAC7E,OAAO,EACL,0BAA0B,GAE3B,MAAM,gBAAgB,CAAC;AAcxB,SAAS,6BAA6B,CACpC,QAAoB,EACpB,MAAgC,EAChC,UAAmB;IAEnB,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;IACzC,MAAM,GAAG,GAAG,0BAA0B,CAAC;QACrC,QAAQ;QACR,MAAM;QACN,OAAO,EAAE,UAAU,CAAC,OAAO;QAC3B,QAAQ,EAAE,UAAU,CAAC,mBAAmB;QACxC,UAAU;KACX,CAAC,CAAC;IACH,6EAA6E;IAC7E,6EAA6E;IAC7E,wEAAwE;IACxE,sEAAsE;IACtE,MAAM,SAAS,GACb,MAAM,KAAK,SAAS;QAClB,CAAC,CAAC,GAAG,UAAU,CAAC,WAAW,kFAAkF,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG;QAC7I,CAAC,CAAC,GAAG,UAAU,CAAC,WAAW,0CAA0C,UAAU,IAAI,YAAY,kCAAkC,CAAC;IAEtI,OAAO;QACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,GAAG,OAAO,SAAS,EAAE,EAAE,CAAC;QAC3D,OAAO,EAAE;YACP,kBAAkB,EAAE;gBAClB,QAAQ;gBACR,MAAM;gBACN,GAAG,CAAC,UAAU,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;aACpD;SACF;KACF,CAAC;AACJ,CAAC;AAED;;;;;;;;;;;;;;;;;;;;;;GAsBG;AACH,MAAM,CAAC,KAAK,UAAU,mBAAmB,CACvC,UAAsB,EACtB,EAAqC;IAErC,IAAI,CAAC,aAAa,CAAC,UAAU,CAAC,EAAE,CAAC;QAC/B,OAAO,6BAA6B,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;IAC9D,CAAC;IAED,IAAI,CAAC;QACH,OAAO,MAAM,EAAE,EAAE,CAAC;IACpB,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,KAAK,YAAY,uBAAuB,EAAE,CAAC;YAC7C,OAAO,6BAA6B,CAClC,KAAK,CAAC,QAAQ,EACd,KAAK,CAAC,MAAM,EACZ,KAAK,CAAC,UAAU,CACjB,CAAC;QACJ,CAAC;QACD,MAAM,KAAK,CAAC;IACd,CAAC;AACH,CAAC"}
@@ -0,0 +1,37 @@
1
+ import type { ProviderId } from "./providers.js";
2
+ export type CredentialRequiredReason = "missing" | "stale";
3
+ export interface CredentialRequiredTagFields {
4
+ provider: ProviderId;
5
+ reason: CredentialRequiredReason;
6
+ unlocks: readonly string[];
7
+ fallback: string | null;
8
+ httpStatus?: number;
9
+ }
10
+ export interface SoftDegradedTagFields {
11
+ provider: ProviderId;
12
+ fallback: string;
13
+ remediation: string;
14
+ }
15
+ export interface SkippedTagFields {
16
+ provider: ProviderId;
17
+ reason: "credential_not_provided";
18
+ remediation: string;
19
+ silenced?: boolean;
20
+ }
21
+ export interface ConnectedTagFields {
22
+ provider: ProviderId;
23
+ }
24
+ export type ParsedTag = ({
25
+ kind: "credential_required";
26
+ } & CredentialRequiredTagFields) | ({
27
+ kind: "soft_degraded";
28
+ } & SoftDegradedTagFields) | ({
29
+ kind: "skipped";
30
+ } & SkippedTagFields) | ({
31
+ kind: "connected";
32
+ } & ConnectedTagFields);
33
+ export declare function buildCredentialRequiredTag(fields: CredentialRequiredTagFields): string;
34
+ export declare function buildSoftDegradedTag(fields: SoftDegradedTagFields): string;
35
+ export declare function buildSkippedTag(fields: SkippedTagFields): string;
36
+ export declare function buildConnectedTag(fields: ConnectedTagFields): string;
37
+ export declare function parseToolTag(text: string): ParsedTag | undefined;