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,118 @@
1
+ // Shared `runProviderConnect` function — the actual side-effectful flow that
2
+ // opens a browser, collects a pasted API key, validates it against the
3
+ // provider's API, persists it to `~/.opencandle/config.json`, and refreshes
4
+ // the cached Config so the next tool call sees the new credential.
5
+ //
6
+ // Called from two places:
7
+ // 1. The Pi `tool_result` extension handler when the user picks "Connect now"
8
+ // in the just-in-time prompt (Task Group 10).
9
+ // 2. The `/connect` Pi command (Task Group 13).
10
+ //
11
+ // Validation (Task Group 8 final pass): before persisting, the pasted key is
12
+ // validated via `validateCredential`, which makes a single cheap request to
13
+ // the provider's canonical API and classifies the response. A hard
14
+ // auth failure (401/403 or a provider-specific error-in-200-body) is
15
+ // returned as `invalid_key` WITHOUT persisting the bad value; a transient
16
+ // failure (timeout, 5xx, network error) warns the user but still persists
17
+ // the key so they aren't blocked on a provider outage — the next real tool
18
+ // call will surface any lingering issue via the credential-required tag.
19
+ import { openInBrowser } from "../infra/open-url.js";
20
+ import { loadFileConfig, saveFileConfig, loadConfig, } from "../config.js";
21
+ import { getProvider, getCredentialSource, } from "./providers.js";
22
+ import { loadOnboardingState, markProviderCompleted, saveOnboardingState, } from "./state.js";
23
+ import { validateCredential } from "./validation.js";
24
+ function writeNested(obj, path, value) {
25
+ if (path.length === 0)
26
+ return obj;
27
+ const [head, ...rest] = path;
28
+ const next = { ...obj };
29
+ if (rest.length === 0) {
30
+ next[head] = value;
31
+ }
32
+ else {
33
+ const child = next[head] && typeof next[head] === "object"
34
+ ? next[head]
35
+ : {};
36
+ next[head] = writeNested(child, rest, value);
37
+ }
38
+ return next;
39
+ }
40
+ /**
41
+ * Persist an already-validated provider credential.
42
+ *
43
+ * Writes the key into `~/.opencandle/config.json` (preserving sibling fields),
44
+ * refreshes the cached `Config` so the next tool call sees the new value, and
45
+ * marks the provider as completed in the onboarding state.
46
+ *
47
+ * Shared by the TUI `/connect` flow (`runProviderConnect`) and the GUI
48
+ * provider-setup form. Validation is the caller's responsibility — both
49
+ * call sites run `validateCredential` before invoking this helper.
50
+ */
51
+ export function persistProviderCredential(providerId, key) {
52
+ const descriptor = getProvider(providerId);
53
+ const existing = loadFileConfig();
54
+ const updated = writeNested(existing, descriptor.configPath, key);
55
+ saveFileConfig(updated);
56
+ loadConfig();
57
+ const state = loadOnboardingState();
58
+ saveOnboardingState(markProviderCompleted(state, providerId));
59
+ }
60
+ /**
61
+ * Run the connect flow for a single provider.
62
+ *
63
+ * Env-precedence short-circuit: if the provider's configured credential
64
+ * already comes from an environment variable, the file-config write would
65
+ * be invisible to the next tool call (env vars win in `resolveConfig`).
66
+ * Notify the user explicitly and return `blocked_by_env` — do not open
67
+ * the browser or collect a pasted value.
68
+ */
69
+ export async function runProviderConnect(ctx, providerId) {
70
+ const descriptor = getProvider(providerId);
71
+ // Env-precedence check.
72
+ if (getCredentialSource(providerId) === "env") {
73
+ ctx.ui.notify(`${descriptor.displayName} is currently set via the ${descriptor.envVar} environment variable. ` +
74
+ `To change it from here, unset that variable first and reopen /connect ${descriptor.aliases[0] ?? descriptor.id}. ` +
75
+ `Otherwise, update ${descriptor.envVar} directly in your shell profile.`, "warning");
76
+ return { status: "blocked_by_env" };
77
+ }
78
+ // Open the signup URL in the user's browser. We ignore errors here — if
79
+ // the browser can't be opened, the user can still paste a key they
80
+ // already have, so we continue rather than bailing.
81
+ await openInBrowser(descriptor.signupUrl).catch(() => { });
82
+ ctx.ui.notify(`Opening ${descriptor.displayName} signup in your browser...`, "info");
83
+ // Prompt for the key. Uses the provider's instructionsHint as the
84
+ // placeholder text to remind the user what they're pasting.
85
+ const pasted = await ctx.ui.input(`Paste your ${descriptor.displayName} API key`, descriptor.instructionsHint);
86
+ if (!pasted) {
87
+ return { status: "cancelled" };
88
+ }
89
+ const trimmed = pasted.trim();
90
+ if (trimmed.length === 0) {
91
+ return { status: "cancelled" };
92
+ }
93
+ // Validate the key with the provider BEFORE persisting. Hard auth failures
94
+ // short-circuit here without writing anything; transient failures warn
95
+ // but proceed to persist so users don't get stuck on a provider outage.
96
+ ctx.ui.notify(`Verifying your ${descriptor.displayName} key...`, "info");
97
+ const validation = await validateCredential(providerId, trimmed);
98
+ if (validation.status === "invalid") {
99
+ const statusHint = validation.httpStatus !== undefined ? ` (HTTP ${validation.httpStatus})` : "";
100
+ const messageHint = validation.message ? ` — ${validation.message}` : "";
101
+ ctx.ui.notify(`${descriptor.displayName} rejected the key${statusHint}${messageHint}. ` +
102
+ `Your existing configuration was not changed. Re-run /connect ${descriptor.aliases[0] ?? descriptor.id} to try a different key.`, "error");
103
+ return {
104
+ status: "invalid_key",
105
+ httpStatus: validation.httpStatus,
106
+ message: validation.message,
107
+ };
108
+ }
109
+ if (validation.status === "transient") {
110
+ ctx.ui.notify(`Couldn't reach ${descriptor.displayName} to verify the key (${validation.reason}). ` +
111
+ `Saving it anyway — the next request will surface any issue.`, "warning");
112
+ // Fall through to persist.
113
+ }
114
+ persistProviderCredential(providerId, trimmed);
115
+ ctx.ui.notify(`${descriptor.displayName} connected. Your key has been saved.`, "info");
116
+ return { status: "connected" };
117
+ }
118
+ //# sourceMappingURL=connect.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connect.js","sourceRoot":"","sources":["../../src/onboarding/connect.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,44 @@
1
+ import type { ProviderId } from "./providers.js";
2
+ import type { OnboardingState } from "./state.js";
3
+ import type { CredentialRequiredReason } from "./tool-tags.js";
4
+ export interface InterceptInput {
5
+ /** The provider id parsed from the tagged tool-result content. */
6
+ provider: ProviderId;
7
+ /** Why the credential was flagged as required. */
8
+ reason: CredentialRequiredReason;
9
+ /** Current persistent onboarding state (providers map, welcome flag). */
10
+ state: OnboardingState;
11
+ /** Providers already prompted at least once in this session (across workflows). */
12
+ sessionPromptedSet: ReadonlySet<ProviderId>;
13
+ /**
14
+ * Whether a hard-tier provider has already fired a prompt earlier in the
15
+ * current workflow invocation. When true, subsequent hard-provider prompts
16
+ * in the same workflow SHALL be silenced per the "at most one prompt per
17
+ * workflow" requirement in `conversational-provider-setup.spec.md`.
18
+ */
19
+ hardPromptFiredInWorkflow: boolean;
20
+ /** Injected clock for deterministic snooze-expiry tests. */
21
+ now: Date;
22
+ }
23
+ /** Outcome of the interception decision. */
24
+ export type InterceptAction = {
25
+ action: "prompt";
26
+ provider: ProviderId;
27
+ reason: CredentialRequiredReason;
28
+ } | {
29
+ action: "skip";
30
+ provider: ProviderId;
31
+ remediation: string;
32
+ /** True when the skip is due to explicit never_ask — downstream gap-note
33
+ * synthesis should suppress the `/connect` remediation in that case. */
34
+ silenced: boolean;
35
+ };
36
+ /**
37
+ * Pure decision function — given an `InterceptInput`, return either a
38
+ * `prompt` action (the extension handler should pause and call `promptUser`)
39
+ * or a `skip` action (the handler should replace the tool result with a
40
+ * `[OPENCANDLE_SKIPPED ...]` placeholder).
41
+ *
42
+ * No side effects. No Pi imports. Trivially unit-testable.
43
+ */
44
+ export declare function resolveCredentialRequired(input: InterceptInput): InterceptAction;
@@ -0,0 +1,72 @@
1
+ // Pure decision function for handling `[OPENCANDLE_CREDENTIAL_REQUIRED ...]`
2
+ // tool results. This module has NO Pi dependencies so it can be tested in
3
+ // isolation — the Pi extension `tool_result` handler is a thin wrapper that
4
+ // calls this function, inspects the returned `InterceptAction`, and drives
5
+ // the appropriate side effects (promptUser, state mutation, tool rerun).
6
+ //
7
+ // Decision table — see design.md Decision 3 for the canonical version.
8
+ import { getProvider } from "./providers.js";
9
+ function buildRemediation(providerId) {
10
+ const descriptor = getProvider(providerId);
11
+ // Prefer the first alias as the friendly target; fall back to the id.
12
+ const alias = descriptor.aliases[0] ?? providerId;
13
+ return `run /connect ${alias} to unlock`;
14
+ }
15
+ function skip(providerId, silenced) {
16
+ return {
17
+ action: "skip",
18
+ provider: providerId,
19
+ remediation: buildRemediation(providerId),
20
+ silenced,
21
+ };
22
+ }
23
+ /**
24
+ * Pure decision function — given an `InterceptInput`, return either a
25
+ * `prompt` action (the extension handler should pause and call `promptUser`)
26
+ * or a `skip` action (the handler should replace the tool result with a
27
+ * `[OPENCANDLE_SKIPPED ...]` placeholder).
28
+ *
29
+ * No side effects. No Pi imports. Trivially unit-testable.
30
+ */
31
+ export function resolveCredentialRequired(input) {
32
+ const { provider, reason, state, sessionPromptedSet, hardPromptFiredInWorkflow, now, } = input;
33
+ const descriptor = getProvider(provider);
34
+ const entry = state.providers[provider];
35
+ // 1. Never-ask: always skip, marked silenced so the gap-note copy drops the
36
+ // `/connect` remediation.
37
+ if (entry?.status === "never_ask") {
38
+ return skip(provider, true);
39
+ }
40
+ // 2. Completed + missing (defensive — shouldn't normally happen because the
41
+ // tool layer only emits `missing` when `hasCredential` returns false, and
42
+ // a completed state implies the credential was persisted).
43
+ if (entry?.status === "completed" && reason === "missing") {
44
+ return skip(provider, false);
45
+ }
46
+ // 3. Session-level dedup: a previous prompt this session already settled
47
+ // the user's answer for this provider. Don't re-ask until next session.
48
+ if (sessionPromptedSet.has(provider)) {
49
+ return skip(provider, false);
50
+ }
51
+ // 4. Per-workflow cap: at most one hard-provider prompt per workflow run.
52
+ if (hardPromptFiredInWorkflow && descriptor.tier === "hard") {
53
+ return skip(provider, false);
54
+ }
55
+ // 5. Active snooze: skip without decrementing the session set (the user
56
+ // will see the prompt again if they try a workflow after snoozeUntil).
57
+ if (entry?.status === "snoozed") {
58
+ const until = Date.parse(entry.snoozeUntil);
59
+ if (!Number.isNaN(until) && now.getTime() < until) {
60
+ return skip(provider, false);
61
+ }
62
+ // snoozed but expired → fall through to prompt.
63
+ }
64
+ // 6. Completed + stale: re-prompt. This is the ONE stale-credential path
65
+ // this change ships — the full failure-as-invitation story is deferred.
66
+ if (entry?.status === "completed" && reason === "stale") {
67
+ return { action: "prompt", provider, reason };
68
+ }
69
+ // 7. Default: prompt.
70
+ return { action: "prompt", provider, reason };
71
+ }
72
+ //# sourceMappingURL=credential-interceptor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"credential-interceptor.js","sourceRoot":"","sources":["../../src/onboarding/credential-interceptor.ts"],"names":[],"mappings":"AAAA,6EAA6E;AAC7E,0EAA0E;AAC1E,4EAA4E;AAC5E,2EAA2E;AAC3E,yEAAyE;AACzE,EAAE;AACF,uEAAuE;AAGvE,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAwC7C,SAAS,gBAAgB,CAAC,UAAsB;IAC9C,MAAM,UAAU,GAAG,WAAW,CAAC,UAAU,CAAC,CAAC;IAC3C,sEAAsE;IACtE,MAAM,KAAK,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,UAAU,CAAC;IAClD,OAAO,gBAAgB,KAAK,YAAY,CAAC;AAC3C,CAAC;AAED,SAAS,IAAI,CACX,UAAsB,EACtB,QAAiB;IAEjB,OAAO;QACL,MAAM,EAAE,MAAM;QACd,QAAQ,EAAE,UAAU;QACpB,WAAW,EAAE,gBAAgB,CAAC,UAAU,CAAC;QACzC,QAAQ;KACT,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,yBAAyB,CACvC,KAAqB;IAErB,MAAM,EACJ,QAAQ,EACR,MAAM,EACN,KAAK,EACL,kBAAkB,EAClB,yBAAyB,EACzB,GAAG,GACJ,GAAG,KAAK,CAAC;IAEV,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;IACzC,MAAM,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IAExC,4EAA4E;IAC5E,6BAA6B;IAC7B,IAAI,KAAK,EAAE,MAAM,KAAK,WAAW,EAAE,CAAC;QAClC,OAAO,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;IAC9B,CAAC;IAED,4EAA4E;IAC5E,6EAA6E;IAC7E,8DAA8D;IAC9D,IAAI,KAAK,EAAE,MAAM,KAAK,WAAW,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;QAC1D,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,yEAAyE;IACzE,2EAA2E;IAC3E,IAAI,kBAAkB,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;QACrC,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,0EAA0E;IAC1E,IAAI,yBAAyB,IAAI,UAAU,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;QAC5D,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;IAC/B,CAAC;IAED,wEAAwE;IACxE,0EAA0E;IAC1E,IAAI,KAAK,EAAE,MAAM,KAAK,SAAS,EAAE,CAAC;QAChC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;QAC5C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,OAAO,EAAE,GAAG,KAAK,EAAE,CAAC;YAClD,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAC;QAC/B,CAAC;QACD,gDAAgD;IAClD,CAAC;IAED,yEAAyE;IACzE,2EAA2E;IAC3E,IAAI,KAAK,EAAE,MAAM,KAAK,WAAW,IAAI,MAAM,KAAK,OAAO,EAAE,CAAC;QACxD,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC;IAChD,CAAC;IAED,sBAAsB;IACtB,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAC;AAChD,CAAC"}
@@ -0,0 +1,21 @@
1
+ import type { ProviderId } from "./providers.js";
2
+ import type { OnboardingState } from "./state.js";
3
+ export interface DegradationAccumulator {
4
+ /** Record that a soft-tier provider fell back during this turn. Idempotent. */
5
+ record(provider: ProviderId): void;
6
+ /** Number of distinct providers recorded since the last reset. */
7
+ size(): number;
8
+ /** True when nothing has been recorded since the last reset. */
9
+ isEmpty(): boolean;
10
+ /**
11
+ * Build a newline-delimited list of `[OPENCANDLE_SKIPPED ...]` tags — one
12
+ * per distinct recorded provider. Providers whose `state.providers[id]`
13
+ * entry has `status: "never_ask"` get the `(silenced)` suffix in their
14
+ * remediation so the system prompt instruction suppresses the `/connect`
15
+ * link in the final answer. Returns `null` when no providers are recorded.
16
+ */
17
+ buildCombinedAnnotation(state: OnboardingState): string | null;
18
+ /** Clear all recorded providers. Call this at turn_start. */
19
+ reset(): void;
20
+ }
21
+ export declare function createDegradationAccumulator(): DegradationAccumulator;
@@ -0,0 +1,55 @@
1
+ // Per-turn accumulator for soft-tier provider degradations. When a soft
2
+ // provider falls back to a keyless alternative (e.g. Brave → DuckDuckGo, Exa →
3
+ // keyless MCP, Finnhub → cached/free source), the tool result content carries
4
+ // a `[OPENCANDLE_SOFT_DEGRADED ...]` tag. The extension's `tool_result` hook
5
+ // records the provider id in this accumulator WITHOUT mutating the tool
6
+ // result; at `turn_end` the accumulator builds a single combined
7
+ // `[OPENCANDLE_SKIPPED ...]`-style annotation that gets appended to the
8
+ // session as a sidecar entry for observability.
9
+ //
10
+ // Per-provider state (never_ask / snoozed / completed) lives in the persistent
11
+ // OnboardingState on disk — the accumulator stays pure and consults the
12
+ // caller-supplied state snapshot at `buildCombinedAnnotation` time. This keeps
13
+ // the accumulator trivially testable without mocking the filesystem.
14
+ import { getProvider } from "./providers.js";
15
+ import { buildSkippedTag } from "./tool-tags.js";
16
+ export function createDegradationAccumulator() {
17
+ const recorded = new Set();
18
+ return {
19
+ record(provider) {
20
+ recorded.add(provider);
21
+ },
22
+ size() {
23
+ return recorded.size;
24
+ },
25
+ isEmpty() {
26
+ return recorded.size === 0;
27
+ },
28
+ buildCombinedAnnotation(state) {
29
+ if (recorded.size === 0)
30
+ return null;
31
+ const lines = [];
32
+ for (const provider of recorded) {
33
+ const descriptor = getProvider(provider);
34
+ const entry = state.providers[provider];
35
+ const silenced = entry?.status === "never_ask";
36
+ const alias = descriptor.aliases[0] ?? descriptor.id;
37
+ const baseRemediation = `run /connect ${alias} to unlock`;
38
+ const remediation = silenced
39
+ ? `${baseRemediation} (silenced)`
40
+ : baseRemediation;
41
+ lines.push(buildSkippedTag({
42
+ provider,
43
+ reason: "credential_not_provided",
44
+ remediation,
45
+ silenced: silenced || undefined,
46
+ }));
47
+ }
48
+ return lines.join("\n");
49
+ },
50
+ reset() {
51
+ recorded.clear();
52
+ },
53
+ };
54
+ }
55
+ //# sourceMappingURL=degradation-accumulator.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"degradation-accumulator.js","sourceRoot":"","sources":["../../src/onboarding/degradation-accumulator.ts"],"names":[],"mappings":"AAAA,wEAAwE;AACxE,+EAA+E;AAC/E,8EAA8E;AAC9E,6EAA6E;AAC7E,wEAAwE;AACxE,iEAAiE;AACjE,wEAAwE;AACxE,gDAAgD;AAChD,EAAE;AACF,+EAA+E;AAC/E,wEAAwE;AACxE,+EAA+E;AAC/E,qEAAqE;AAGrE,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,OAAO,EAAE,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAqBjD,MAAM,UAAU,4BAA4B;IAC1C,MAAM,QAAQ,GAAG,IAAI,GAAG,EAAc,CAAC;IAEvC,OAAO;QACL,MAAM,CAAC,QAAoB;YACzB,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACzB,CAAC;QACD,IAAI;YACF,OAAO,QAAQ,CAAC,IAAI,CAAC;QACvB,CAAC;QACD,OAAO;YACL,OAAO,QAAQ,CAAC,IAAI,KAAK,CAAC,CAAC;QAC7B,CAAC;QACD,uBAAuB,CAAC,KAAsB;YAC5C,IAAI,QAAQ,CAAC,IAAI,KAAK,CAAC;gBAAE,OAAO,IAAI,CAAC;YACrC,MAAM,KAAK,GAAa,EAAE,CAAC;YAC3B,KAAK,MAAM,QAAQ,IAAI,QAAQ,EAAE,CAAC;gBAChC,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,CAAC;gBACzC,MAAM,KAAK,GAAG,KAAK,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;gBACxC,MAAM,QAAQ,GAAG,KAAK,EAAE,MAAM,KAAK,WAAW,CAAC;gBAC/C,MAAM,KAAK,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,UAAU,CAAC,EAAE,CAAC;gBACrD,MAAM,eAAe,GAAG,gBAAgB,KAAK,YAAY,CAAC;gBAC1D,MAAM,WAAW,GAAG,QAAQ;oBAC1B,CAAC,CAAC,GAAG,eAAe,aAAa;oBACjC,CAAC,CAAC,eAAe,CAAC;gBACpB,KAAK,CAAC,IAAI,CACR,eAAe,CAAC;oBACd,QAAQ;oBACR,MAAM,EAAE,yBAAyB;oBACjC,WAAW;oBACX,QAAQ,EAAE,QAAQ,IAAI,SAAS;iBAChC,CAAC,CACH,CAAC;YACJ,CAAC;YACD,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1B,CAAC;QACD,KAAK;YACH,QAAQ,CAAC,KAAK,EAAE,CAAC;QACnB,CAAC;KACF,CAAC;AACJ,CAAC"}
@@ -0,0 +1,23 @@
1
+ import type { ExtensionContext } from "@earendil-works/pi-coding-agent";
2
+ import type { AskUserHandler } from "../types/index.js";
3
+ export interface PromptOptions {
4
+ question: string;
5
+ questionType: "select" | "text" | "confirm";
6
+ options?: string[];
7
+ placeholder?: string;
8
+ reason?: string;
9
+ }
10
+ export interface PromptResult {
11
+ answer: string | null;
12
+ cancelled: boolean;
13
+ }
14
+ /**
15
+ * Ask the user a structured question, routing to the appropriate UI primitive
16
+ * (`ctx.ui.select` / `ctx.ui.input` / `ctx.ui.confirm`).
17
+ *
18
+ * When an `askUserHandler` is injected (test harness, programmatic runs), the
19
+ * handler takes precedence over `ctx.ui` and provides the answer directly.
20
+ *
21
+ * When neither a handler nor a UI is available, returns `{answer: null, cancelled: true}`.
22
+ */
23
+ export declare function promptUser(ctx: ExtensionContext, opts: PromptOptions, handler?: AskUserHandler): Promise<PromptResult>;
@@ -0,0 +1,61 @@
1
+ // Shared prompt primitive used by both the `ask_user` tool and the
2
+ // credential-interception handler in the Pi `tool_result` hook.
3
+ //
4
+ // The original `ask_user` tool embedded its UI routing in a closure inside
5
+ // `execute()`. Extracting it here means the `tool_result` handler can call
6
+ // the same logic without synthesizing a fake tool call (Pi has no "execute
7
+ // a tool now" API), and the headless harness's `askUserHandler` injection
8
+ // point is preserved for automated flows.
9
+ /**
10
+ * Ask the user a structured question, routing to the appropriate UI primitive
11
+ * (`ctx.ui.select` / `ctx.ui.input` / `ctx.ui.confirm`).
12
+ *
13
+ * When an `askUserHandler` is injected (test harness, programmatic runs), the
14
+ * handler takes precedence over `ctx.ui` and provides the answer directly.
15
+ *
16
+ * When neither a handler nor a UI is available, returns `{answer: null, cancelled: true}`.
17
+ */
18
+ export async function promptUser(ctx, opts, handler) {
19
+ // Priority: injected handler > UI > no-UI fallback.
20
+ if (handler) {
21
+ const result = await handler({
22
+ question: opts.question,
23
+ questionType: opts.questionType,
24
+ options: opts.options,
25
+ placeholder: opts.placeholder,
26
+ reason: opts.reason,
27
+ });
28
+ if (result.cancelled) {
29
+ return { answer: null, cancelled: true };
30
+ }
31
+ return { answer: result.answer ?? null, cancelled: false };
32
+ }
33
+ if (!ctx?.hasUI) {
34
+ return { answer: null, cancelled: true };
35
+ }
36
+ switch (opts.questionType) {
37
+ case "select": {
38
+ const options = opts.options ?? [];
39
+ if (options.length === 0) {
40
+ return { answer: null, cancelled: true };
41
+ }
42
+ const choice = await ctx.ui.select(opts.question, options);
43
+ if (choice === undefined) {
44
+ return { answer: null, cancelled: true };
45
+ }
46
+ return { answer: choice, cancelled: false };
47
+ }
48
+ case "text": {
49
+ const input = await ctx.ui.input(opts.question, opts.placeholder ?? "");
50
+ if (input === undefined || input.trim() === "") {
51
+ return { answer: null, cancelled: true };
52
+ }
53
+ return { answer: input.trim(), cancelled: false };
54
+ }
55
+ case "confirm": {
56
+ const confirmed = await ctx.ui.confirm(opts.question, opts.reason ?? "");
57
+ return { answer: confirmed ? "Yes" : "No", cancelled: false };
58
+ }
59
+ }
60
+ }
61
+ //# sourceMappingURL=prompt-user.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"prompt-user.js","sourceRoot":"","sources":["../../src/onboarding/prompt-user.ts"],"names":[],"mappings":"AAAA,mEAAmE;AACnE,gEAAgE;AAChE,EAAE;AACF,2EAA2E;AAC3E,2EAA2E;AAC3E,2EAA2E;AAC3E,0EAA0E;AAC1E,0CAA0C;AAkB1C;;;;;;;;GAQG;AACH,MAAM,CAAC,KAAK,UAAU,UAAU,CAC9B,GAAqB,EACrB,IAAmB,EACnB,OAAwB;IAExB,oDAAoD;IACpD,IAAI,OAAO,EAAE,CAAC;QACZ,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC;YAC3B,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,MAAM,EAAE,IAAI,CAAC,MAAM;SACpB,CAAC,CAAC;QACH,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;YACrB,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;QAC3C,CAAC;QACD,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;IAC7D,CAAC;IAED,IAAI,CAAC,GAAG,EAAE,KAAK,EAAE,CAAC;QAChB,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;IAC3C,CAAC;IAED,QAAQ,IAAI,CAAC,YAAY,EAAE,CAAC;QAC1B,KAAK,QAAQ,CAAC,CAAC,CAAC;YACd,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,EAAE,CAAC;YACnC,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACzB,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;YAC3C,CAAC;YACD,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;YAC3D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;gBACzB,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;YAC3C,CAAC;YACD,OAAO,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;QAC9C,CAAC;QAED,KAAK,MAAM,CAAC,CAAC,CAAC;YACZ,MAAM,KAAK,GAAG,MAAM,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,WAAW,IAAI,EAAE,CAAC,CAAC;YACxE,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;gBAC/C,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC;YAC3C,CAAC;YACD,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,IAAI,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;QACpD,CAAC;QAED,KAAK,SAAS,CAAC,CAAC,CAAC;YACf,MAAM,SAAS,GAAG,MAAM,GAAG,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC;YACzE,OAAO,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,EAAE,SAAS,EAAE,KAAK,EAAE,CAAC;QAChE,CAAC;IACH,CAAC;AACH,CAAC"}
@@ -0,0 +1,116 @@
1
+ export type ProviderId = "alpha_vantage" | "fred" | "finnhub" | "brave" | "exa";
2
+ export type ProviderCategory = "fundamentals" | "macro" | "news" | "web_search";
3
+ export type ProviderTier = "hard" | "soft";
4
+ export interface ProviderDescriptor {
5
+ readonly id: ProviderId;
6
+ readonly displayName: string;
7
+ readonly category: ProviderCategory;
8
+ /**
9
+ * `hard` providers pause the workflow with a just-in-time prompt when their
10
+ * credential is missing; they have no meaningful fallback.
11
+ *
12
+ * `soft` providers silently use a fallback path and surface a post-answer
13
+ * gap note in the final output.
14
+ */
15
+ readonly tier: ProviderTier;
16
+ /** Lowercase friendly aliases accepted by `/connect` in addition to the id. */
17
+ readonly aliases: readonly string[];
18
+ readonly signupUrl: string;
19
+ readonly freeTier: boolean;
20
+ readonly envVar: string;
21
+ /** Nested key path into `OpenCandleFileConfig` where the key is persisted. */
22
+ readonly configPath: readonly string[];
23
+ readonly unlocks: readonly string[];
24
+ /**
25
+ * Human copy describing the degraded experience when missing, or `null`
26
+ * when there is no fallback (hard tier).
27
+ */
28
+ readonly fallbackDescription: string | null;
29
+ readonly snoozeDurationDays: number;
30
+ readonly instructionsHint: string;
31
+ }
32
+ export declare const PROVIDERS: readonly [{
33
+ readonly id: "alpha_vantage";
34
+ readonly displayName: "Alpha Vantage";
35
+ readonly category: "fundamentals";
36
+ readonly tier: "hard";
37
+ readonly aliases: readonly ["financials", "fundamentals", "company-financials", "alphavantage"];
38
+ readonly signupUrl: "https://www.alphavantage.co/support/#api-key";
39
+ readonly freeTier: true;
40
+ readonly envVar: "ALPHA_VANTAGE_API_KEY";
41
+ readonly configPath: readonly ["providers", "alphaVantage", "apiKey"];
42
+ readonly unlocks: readonly ["company fundamentals", "income/balance/cashflow statements", "DCF valuation", "earnings history"];
43
+ readonly fallbackDescription: null;
44
+ readonly snoozeDurationDays: 7;
45
+ readonly instructionsHint: "Free, about 30 seconds, signup opens in your browser";
46
+ }, {
47
+ readonly id: "fred";
48
+ readonly displayName: "FRED";
49
+ readonly category: "macro";
50
+ readonly tier: "hard";
51
+ readonly aliases: readonly ["economy", "macro", "economic-data", "st-louis-fed"];
52
+ readonly signupUrl: "https://fredaccount.stlouisfed.org/apikeys";
53
+ readonly freeTier: true;
54
+ readonly envVar: "FRED_API_KEY";
55
+ readonly configPath: readonly ["providers", "fred", "apiKey"];
56
+ readonly unlocks: readonly ["interest rates", "inflation data", "yield curve", "economic indicators"];
57
+ readonly fallbackDescription: null;
58
+ readonly snoozeDurationDays: 7;
59
+ readonly instructionsHint: "Free, about 30 seconds, requires a St. Louis Fed account";
60
+ }, {
61
+ readonly id: "finnhub";
62
+ readonly displayName: "Finnhub";
63
+ readonly category: "news";
64
+ readonly tier: "soft";
65
+ readonly aliases: readonly ["news", "company-news", "finnhub-news"];
66
+ readonly signupUrl: "https://finnhub.io/register";
67
+ readonly freeTier: true;
68
+ readonly envVar: "FINNHUB_API_KEY";
69
+ readonly configPath: readonly ["providers", "finnhub", "apiKey"];
70
+ readonly unlocks: readonly ["ticker-tagged company news", "sentiment enrichment with a dedicated news source"];
71
+ readonly fallbackDescription: "Other sentiment sources (Reddit, Twitter, web search) continue to work without Finnhub";
72
+ readonly snoozeDurationDays: 7;
73
+ readonly instructionsHint: "Free, about 30 seconds, signup opens in your browser";
74
+ }, {
75
+ readonly id: "brave";
76
+ readonly displayName: "Brave Search";
77
+ readonly category: "web_search";
78
+ readonly tier: "soft";
79
+ readonly aliases: readonly ["brave", "brave-search"];
80
+ readonly signupUrl: "https://brave.com/search/api/";
81
+ readonly freeTier: true;
82
+ readonly envVar: "BRAVE_API_KEY";
83
+ readonly configPath: readonly ["providers", "brave", "apiKey"];
84
+ readonly unlocks: readonly ["tier-2 web search with freshness control", "independent search index outside of DuckDuckGo"];
85
+ readonly fallbackDescription: "Web search continues to work via DuckDuckGo (free, no key needed, lower-quality freshness)";
86
+ readonly snoozeDurationDays: 7;
87
+ readonly instructionsHint: "Free tier available, signup opens in your browser";
88
+ }, {
89
+ readonly id: "exa";
90
+ readonly displayName: "Exa";
91
+ readonly category: "web_search";
92
+ readonly tier: "soft";
93
+ readonly aliases: readonly ["exa", "exa-search"];
94
+ readonly signupUrl: "https://dashboard.exa.ai/";
95
+ readonly freeTier: false;
96
+ readonly envVar: "EXA_API_KEY";
97
+ readonly configPath: readonly ["providers", "exa", "apiKey"];
98
+ readonly unlocks: readonly ["tier-1 semantic web search", "full article text and highlights", "higher freshness accuracy than DuckDuckGo"];
99
+ readonly fallbackDescription: "Exa search continues to work via the keyless Exa MCP endpoint, which has lower rate limits but similar quality";
100
+ readonly snoozeDurationDays: 7;
101
+ readonly instructionsHint: "Paid with free tier, signup opens in your browser";
102
+ }];
103
+ export declare function listAllProviders(): readonly ProviderDescriptor[];
104
+ export declare function getProvider(id: ProviderId): ProviderDescriptor;
105
+ export declare function getProvidersByCategory(category: ProviderCategory): readonly ProviderDescriptor[];
106
+ export declare function getProvidersByTier(tier: ProviderTier): readonly ProviderDescriptor[];
107
+ export declare function hasCredential(id: ProviderId): boolean;
108
+ export declare function getCredentialSource(id: ProviderId): "env" | "file" | "absent";
109
+ export declare function getCredential(id: ProviderId): {
110
+ source: "env" | "file";
111
+ value: string;
112
+ } | {
113
+ source: "absent";
114
+ value?: undefined;
115
+ };
116
+ export declare function resolveProviderFromArgument(arg: string): ProviderDescriptor | readonly ProviderDescriptor[] | undefined;