opencandle 0.6.0 → 0.7.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (154) hide show
  1. package/README.md +10 -3
  2. package/dist/cli.js +36 -0
  3. package/dist/cli.js.map +1 -1
  4. package/dist/config.d.ts +10 -0
  5. package/dist/config.js +13 -0
  6. package/dist/config.js.map +1 -1
  7. package/dist/infra/index.d.ts +0 -1
  8. package/dist/infra/index.js +0 -1
  9. package/dist/infra/index.js.map +1 -1
  10. package/dist/onboarding/connect.d.ts +2 -2
  11. package/dist/onboarding/connect.js +10 -3
  12. package/dist/onboarding/connect.js.map +1 -1
  13. package/dist/onboarding/provider-status.d.ts +48 -0
  14. package/dist/onboarding/provider-status.js +285 -0
  15. package/dist/onboarding/provider-status.js.map +1 -0
  16. package/dist/onboarding/providers.d.ts +85 -8
  17. package/dist/onboarding/providers.js +87 -9
  18. package/dist/onboarding/providers.js.map +1 -1
  19. package/dist/onboarding/state.d.ts +1 -0
  20. package/dist/onboarding/state.js +5 -0
  21. package/dist/onboarding/state.js.map +1 -1
  22. package/dist/onboarding/tool-tags.d.ts +12 -1
  23. package/dist/onboarding/tool-tags.js +31 -1
  24. package/dist/onboarding/tool-tags.js.map +1 -1
  25. package/dist/onboarding/validation.d.ts +2 -2
  26. package/dist/onboarding/validation.js.map +1 -1
  27. package/dist/pi/opencandle-extension.js +91 -15
  28. package/dist/pi/opencandle-extension.js.map +1 -1
  29. package/dist/pi/tool-adapter.d.ts +4 -1
  30. package/dist/pi/tool-adapter.js +5 -4
  31. package/dist/pi/tool-adapter.js.map +1 -1
  32. package/dist/prompts/context-builder.js +1 -1
  33. package/dist/prompts/policy-cards.js +1 -1
  34. package/dist/prompts/policy-cards.js.map +1 -1
  35. package/dist/providers/external-tool-error.d.ts +10 -0
  36. package/dist/providers/external-tool-error.js +21 -0
  37. package/dist/providers/external-tool-error.js.map +1 -0
  38. package/dist/providers/reddit-cli.d.ts +36 -0
  39. package/dist/providers/reddit-cli.js +201 -0
  40. package/dist/providers/reddit-cli.js.map +1 -0
  41. package/dist/providers/reddit.d.ts +1 -1
  42. package/dist/providers/reddit.js +7 -35
  43. package/dist/providers/reddit.js.map +1 -1
  44. package/dist/providers/twitter-cli.d.ts +40 -0
  45. package/dist/providers/twitter-cli.js +153 -0
  46. package/dist/providers/twitter-cli.js.map +1 -0
  47. package/dist/providers/twitter.d.ts +0 -8
  48. package/dist/providers/twitter.js +4 -54
  49. package/dist/providers/twitter.js.map +1 -1
  50. package/dist/providers/wrap-provider.js +30 -0
  51. package/dist/providers/wrap-provider.js.map +1 -1
  52. package/dist/providers/yahoo-finance.js +53 -32
  53. package/dist/providers/yahoo-finance.js.map +1 -1
  54. package/dist/routing/planning.d.ts +1 -1
  55. package/dist/routing/planning.js.map +1 -1
  56. package/dist/runtime/answer-contracts.d.ts +1 -1
  57. package/dist/runtime/answer-contracts.js +12 -1
  58. package/dist/runtime/answer-contracts.js.map +1 -1
  59. package/dist/runtime/tool-defaults-wrapper.js +6 -2
  60. package/dist/runtime/tool-defaults-wrapper.js.map +1 -1
  61. package/dist/sentiment/index.d.ts +1 -0
  62. package/dist/sentiment/index.js +1 -0
  63. package/dist/sentiment/index.js.map +1 -1
  64. package/dist/sentiment/insights.d.ts +17 -0
  65. package/dist/sentiment/insights.js +206 -0
  66. package/dist/sentiment/insights.js.map +1 -0
  67. package/dist/sentiment/pipeline.js +13 -1
  68. package/dist/sentiment/pipeline.js.map +1 -1
  69. package/dist/sentiment/scorer.d.ts +2 -0
  70. package/dist/sentiment/scorer.js +10 -1
  71. package/dist/sentiment/scorer.js.map +1 -1
  72. package/dist/sentiment/types.d.ts +2 -0
  73. package/dist/sentiment/types.js.map +1 -1
  74. package/dist/system-prompt.js +3 -7
  75. package/dist/system-prompt.js.map +1 -1
  76. package/dist/tools/index.d.ts +5 -2
  77. package/dist/tools/index.js +8 -8
  78. package/dist/tools/index.js.map +1 -1
  79. package/dist/tools/sentiment/insight-format.d.ts +2 -0
  80. package/dist/tools/sentiment/insight-format.js +36 -0
  81. package/dist/tools/sentiment/insight-format.js.map +1 -0
  82. package/dist/tools/sentiment/query-match.d.ts +3 -0
  83. package/dist/tools/sentiment/query-match.js +113 -0
  84. package/dist/tools/sentiment/query-match.js.map +1 -0
  85. package/dist/tools/sentiment/reddit-sentiment.d.ts +12 -1
  86. package/dist/tools/sentiment/reddit-sentiment.js +263 -117
  87. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  88. package/dist/tools/sentiment/sentiment-summary.d.ts +9 -1
  89. package/dist/tools/sentiment/sentiment-summary.js +217 -201
  90. package/dist/tools/sentiment/sentiment-summary.js.map +1 -1
  91. package/dist/tools/sentiment/twitter-sentiment.d.ts +11 -1
  92. package/dist/tools/sentiment/twitter-sentiment.js +187 -64
  93. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
  94. package/dist/tools/sentiment/web-sentiment.js +4 -0
  95. package/dist/tools/sentiment/web-sentiment.js.map +1 -1
  96. package/dist/types/sentiment.d.ts +52 -0
  97. package/gui/server/invoke-tool.ts +17 -3
  98. package/gui/server/model-setup.ts +10 -3
  99. package/gui/server/projector.ts +6 -2
  100. package/gui/server/server.ts +18 -0
  101. package/gui/server/tool-metadata.ts +80 -16
  102. package/gui/server/ws-hub.ts +19 -0
  103. package/gui/web/dist/assets/CatalogOverlay-CgeY5Pkp.js +1 -0
  104. package/gui/web/dist/assets/index-C6W_2eAn.js +69 -0
  105. package/gui/web/dist/assets/{index-2KZtKBmu.css → index-hwbx24a5.css} +1 -1
  106. package/gui/web/dist/index.html +2 -2
  107. package/package.json +5 -6
  108. package/src/cli.ts +41 -0
  109. package/src/config.ts +27 -0
  110. package/src/infra/index.ts +0 -1
  111. package/src/onboarding/connect.ts +20 -4
  112. package/src/onboarding/provider-status.ts +410 -0
  113. package/src/onboarding/providers.ts +148 -18
  114. package/src/onboarding/state.ts +9 -0
  115. package/src/onboarding/tool-tags.ts +45 -2
  116. package/src/onboarding/validation.ts +2 -2
  117. package/src/pi/opencandle-extension.ts +115 -17
  118. package/src/pi/tool-adapter.ts +14 -4
  119. package/src/prompts/context-builder.ts +1 -1
  120. package/src/prompts/policy-cards.ts +1 -1
  121. package/src/providers/external-tool-error.ts +20 -0
  122. package/src/providers/reddit-cli.ts +317 -0
  123. package/src/providers/reddit.ts +7 -63
  124. package/src/providers/twitter-cli.ts +233 -0
  125. package/src/providers/twitter.ts +4 -73
  126. package/src/providers/wrap-provider.ts +34 -0
  127. package/src/providers/yahoo-finance.ts +65 -32
  128. package/src/routing/planning.ts +1 -0
  129. package/src/runtime/answer-contracts.ts +23 -2
  130. package/src/runtime/tool-defaults-wrapper.ts +12 -2
  131. package/src/sentiment/index.ts +1 -0
  132. package/src/sentiment/insights.ts +269 -0
  133. package/src/sentiment/pipeline.ts +13 -1
  134. package/src/sentiment/scorer.ts +12 -1
  135. package/src/sentiment/types.ts +3 -0
  136. package/src/system-prompt.ts +3 -7
  137. package/src/tools/index.ts +9 -8
  138. package/src/tools/sentiment/insight-format.ts +50 -0
  139. package/src/tools/sentiment/query-match.ts +117 -0
  140. package/src/tools/sentiment/reddit-sentiment.ts +354 -141
  141. package/src/tools/sentiment/sentiment-summary.ts +283 -237
  142. package/src/tools/sentiment/twitter-sentiment.ts +262 -78
  143. package/src/tools/sentiment/web-sentiment.ts +4 -0
  144. package/src/types/sentiment.ts +59 -0
  145. package/dist/infra/browser.d.ts +0 -35
  146. package/dist/infra/browser.js +0 -105
  147. package/dist/infra/browser.js.map +0 -1
  148. package/dist/tools/interaction/twitter-login.d.ts +0 -8
  149. package/dist/tools/interaction/twitter-login.js +0 -87
  150. package/dist/tools/interaction/twitter-login.js.map +0 -1
  151. package/gui/web/dist/assets/CatalogOverlay-eJ2cBk33.js +0 -1
  152. package/gui/web/dist/assets/index-CveNgtDg.js +0 -69
  153. package/src/infra/browser.ts +0 -113
  154. package/src/tools/interaction/twitter-login.ts +0 -105
@@ -1,8 +1,13 @@
1
1
  import { Type } from "@sinclair/typebox";
2
+ import { promptUser } from "../../onboarding/prompt-user.js";
3
+ import { getProvider } from "../../onboarding/providers.js";
4
+ import { loadOnboardingState, markProviderNeverAsk, saveOnboardingState, } from "../../onboarding/state.js";
5
+ import { buildExternalToolRequiredTag, buildSkippedTag } from "../../onboarding/tool-tags.js";
2
6
  import { getTwitterSentiment } from "../../providers/twitter.js";
3
7
  import { wrapProvider } from "../../providers/wrap-provider.js";
4
8
  import { TwitterAdapter } from "../../sentiment/adapters/twitter.js";
5
9
  import { getSentimentPipeline } from "../../sentiment/index.js";
10
+ import { formatInsightSection } from "./insight-format.js";
6
11
  import { renderUntrustedText, untrustedContentHeader } from "./untrusted-text.js";
7
12
  const params = Type.Object({
8
13
  query: Type.String({
@@ -11,72 +16,190 @@ const params = Type.Object({
11
16
  limit: Type.Optional(Type.Number({ description: "Max tweets to fetch. Default: 50, max: 200" })),
12
17
  hours: Type.Optional(Type.Number({ description: "Lookback window in hours. Default: 24" })),
13
18
  });
14
- export const twitterSentimentTool = {
15
- name: "get_twitter_sentiment",
16
- label: "Twitter Sentiment",
17
- description: "Fetch recent tweets for a stock ticker or search query and compute engagement-weighted sentiment. Returns tweet data, sentiment score, and co-mentioned tickers. Requires a Twitter session via trigger_twitter_login.",
18
- parameters: params,
19
- async execute(_toolCallId, args) {
20
- const limit = Math.min(args.limit ?? 50, 200);
21
- const hours = args.hours ?? 24;
22
- const providerResult = await wrapProvider("twitter", () => getTwitterSentiment(args.query, limit, hours));
23
- if (providerResult.status === "unavailable") {
24
- const isLoginIssue = providerResult.reason.includes("No Twitter session") ||
25
- providerResult.reason.includes("session expired");
26
- const text = isLoginIssue
27
- ? `⚠ Twitter sentiment unavailable: ${providerResult.reason}\n[LOGIN_NEEDED] Use ask_user to confirm, then call trigger_twitter_login. After success, retry this tool.`
28
- : `⚠ Twitter sentiment unavailable (${providerResult.reason}).`;
29
- return {
30
- content: [{ type: "text", text }],
31
- details: null,
32
- };
33
- }
34
- const result = providerResult.data;
35
- const sentimentLabel = result.sentimentScore > 0.3
36
- ? "Bullish"
37
- : result.sentimentScore < -0.3
38
- ? "Bearish"
39
- : result.sentimentScore > 0
40
- ? "Leaning Bullish"
41
- : result.sentimentScore < 0
42
- ? "Leaning Bearish"
43
- : "Neutral";
44
- const lines = [
45
- `**Twitter: ${result.query}** ${result.tweetCount} tweets (last ${hours}h, ${result.fetchedAt})`,
46
- `Sentiment: ${result.sentimentScore.toFixed(2)} (${sentimentLabel}) | Bullish: ${result.bullishCount} | Bearish: ${result.bearishCount}`,
47
- ];
48
- if (result.topMentions.length > 0) {
49
- lines.push(`Co-mentions: ${result.topMentions.map((t) => `$${t}`).join(", ")}`);
50
- }
19
+ const INSTALL_CONTINUE_LABEL = "Continue after installing twitter-cli";
20
+ const SESSION_CONTINUE_LABEL = "Continue after refreshing X/Twitter session";
21
+ const SKIP_ONCE_LABEL = "Skip X/Twitter once";
22
+ const ALWAYS_SKIP_LABEL = "Always skip X/Twitter";
23
+ export function createTwitterSentimentTool(options = {}) {
24
+ return {
25
+ name: "get_twitter_sentiment",
26
+ label: "Twitter Sentiment",
27
+ description: "Fetch recent tweets for a stock ticker or search query and compute engagement-weighted sentiment. Returns tweet data, sentiment score, and co-mentioned tickers. Requires twitter-cli and a browser X/Twitter session.",
28
+ parameters: params,
29
+ async execute(_toolCallId, args, _signal, _onUpdate, ctx) {
30
+ const limit = Math.min(args.limit ?? 50, 200);
31
+ const hours = args.hours ?? 24;
32
+ const descriptor = getProvider("twitter");
33
+ const state = loadOnboardingState();
34
+ if (state.providers.twitter?.status === "never_ask") {
35
+ return skippedResult("You previously asked not to be reminded about X/Twitter.", true, "run opencandle doctor --enable twitter to re-enable X/Twitter sentiment (silenced)");
36
+ }
37
+ const fetchTwitterSentiment = () => wrapProvider("twitter", () => getTwitterSentiment(args.query, limit, hours));
38
+ const providerResult = await fetchTwitterSentiment();
39
+ if (providerResult.status === "unavailable") {
40
+ const reason = providerResult.reason;
41
+ const unavailableKind = classifyUnavailableReason(reason);
42
+ const askUserHandler = options.askUserHandler ??
43
+ ctx?.askUserHandler;
44
+ const canAsk = askUserHandler != null || ctx?.hasUI === true;
45
+ if (unavailableKind !== "other" && canAsk) {
46
+ const promptResult = await promptUser((ctx ?? { hasUI: false }), {
47
+ question: setupQuestion(unavailableKind, reason),
48
+ questionType: "select",
49
+ options: [
50
+ unavailableKind === "install" ? INSTALL_CONTINUE_LABEL : SESSION_CONTINUE_LABEL,
51
+ SKIP_ONCE_LABEL,
52
+ ALWAYS_SKIP_LABEL,
53
+ ],
54
+ reason: "X/Twitter sentiment needs the twitter-cli command and a browser session before it can fetch tweets.",
55
+ }, askUserHandler);
56
+ if (!promptResult.cancelled && promptResult.answer?.startsWith("Continue")) {
57
+ const retried = await fetchTwitterSentiment();
58
+ if (retried.status === "ok") {
59
+ return formatTwitterSentimentResult(retried.data, args.query, hours, {
60
+ stale: retried.stale,
61
+ timestamp: retried.timestamp,
62
+ });
63
+ }
64
+ return passiveUnavailableResult(retried.reason, classifyUnavailableReason(retried.reason), { interceptable: false });
65
+ }
66
+ if (!promptResult.cancelled && promptResult.answer === ALWAYS_SKIP_LABEL) {
67
+ saveOnboardingState(markProviderNeverAsk(state, "twitter"));
68
+ return skippedResult(`User chose to always skip ${descriptor.displayName} data. Do not ask about X/Twitter setup again unless the user explicitly reconnects it.`, true, "run opencandle doctor --enable twitter to re-enable X/Twitter sentiment (silenced)");
69
+ }
70
+ return skippedResult(`User chose to skip ${descriptor.displayName} data for this request. Do not ask about X/Twitter setup again in this turn.`, false, "user chose to skip X/Twitter for this request");
71
+ }
72
+ return passiveUnavailableResult(reason, unavailableKind);
73
+ }
74
+ return formatTwitterSentimentResult(providerResult.data, args.query, hours, {
75
+ stale: providerResult.stale,
76
+ timestamp: providerResult.timestamp,
77
+ });
78
+ },
79
+ };
80
+ }
81
+ export const twitterSentimentTool = createTwitterSentimentTool();
82
+ function classifyUnavailableReason(reason) {
83
+ if (reason.includes("not installed") || reason.includes("uv tool install"))
84
+ return "install";
85
+ if (/no twitter cookies|no cookies|401|unauthorized|expired|session/i.test(reason)) {
86
+ return "session";
87
+ }
88
+ return "other";
89
+ }
90
+ function setupQuestion(kind, reason) {
91
+ if (kind === "install") {
92
+ return `X/Twitter sentiment requires twitter-cli. Install it with \`uv tool install twitter-cli\`, then choose whether to continue or skip X/Twitter. Current status: ${reason}`;
93
+ }
94
+ return `X/Twitter sentiment needs an active browser session. Log into or refresh x.com in a supported browser, then choose whether to continue or skip X/Twitter. Current status: ${reason}`;
95
+ }
96
+ function passiveUnavailableResult(reason, kind, options = {}) {
97
+ if ((options.interceptable ?? true) && (kind === "install" || kind === "session")) {
98
+ const tag = buildExternalToolRequiredTag({
99
+ provider: "twitter",
100
+ reason: kind === "install"
101
+ ? "not_installed"
102
+ : /401|unauthorized|expired/i.test(reason)
103
+ ? "session_stale"
104
+ : "session_missing",
105
+ installCmd: "uv tool install twitter-cli",
106
+ loginCmd: "log into x.com in a supported browser",
107
+ fallback: "reddit-web-news",
108
+ });
109
+ const guidance = kind === "install"
110
+ ? "Twitter sentiment requires twitter-cli. Install it with `uv tool install twitter-cli`, then choose whether to continue or skip X/Twitter."
111
+ : "Twitter sentiment requires an active X/Twitter browser session. Log into or refresh x.com in a supported browser, then retry after confirmation.";
112
+ return {
113
+ content: [
114
+ {
115
+ type: "text",
116
+ text: `⚠ Twitter sentiment unavailable: ${reason}\n${tag}\n${guidance}`,
117
+ },
118
+ ],
119
+ details: null,
120
+ };
121
+ }
122
+ const text = `⚠ Twitter sentiment unavailable (${reason}).`;
123
+ return {
124
+ content: [{ type: "text", text }],
125
+ details: null,
126
+ };
127
+ }
128
+ function skippedResult(message, silenced, remediation) {
129
+ return {
130
+ content: [
131
+ {
132
+ type: "text",
133
+ text: `${buildSkippedTag({
134
+ provider: "twitter",
135
+ reason: "credential_not_provided",
136
+ remediation,
137
+ silenced,
138
+ })}\n\n${message}`,
139
+ },
140
+ ],
141
+ details: null,
142
+ };
143
+ }
144
+ async function formatTwitterSentimentResult(result, query, hours, metadata) {
145
+ const sentimentLabel = result.sentimentScore > 0.3
146
+ ? "Bullish"
147
+ : result.sentimentScore < -0.3
148
+ ? "Bearish"
149
+ : result.sentimentScore > 0
150
+ ? "Leaning Bullish"
151
+ : result.sentimentScore < 0
152
+ ? "Leaning Bearish"
153
+ : "Neutral";
154
+ const lines = [
155
+ `**Twitter: ${result.query}** — ${result.tweetCount} tweets (last ${hours}h, ${result.fetchedAt})`,
156
+ `Sentiment: ${result.sentimentScore.toFixed(2)} (${sentimentLabel}) | Bullish: ${result.bullishCount} | Bearish: ${result.bearishCount}`,
157
+ ];
158
+ if (result.topMentions.length > 0) {
159
+ lines.push(`Co-mentions: ${result.topMentions.map((t) => `$${t}`).join(", ")}`);
160
+ }
161
+ lines.push("");
162
+ lines.push(untrustedContentHeader("tweets"));
163
+ lines.push("| Author | Tweet | ❤️ | 🔁 | 💬 |");
164
+ lines.push("|--------|-------|----|----|----|");
165
+ const top = result.tweets.slice(0, 15);
166
+ for (const tweet of top) {
167
+ const text = renderUntrustedText(tweet.text, 100);
168
+ lines.push(`| @${tweet.author} | ${text} | ${tweet.likes} | ${tweet.retweets} | ${tweet.replies} |`);
169
+ }
170
+ if (metadata.stale) {
51
171
  lines.push("");
52
- lines.push(untrustedContentHeader("tweets"));
53
- lines.push("| Author | Tweet | ❤️ | 🔁 | 💬 |");
54
- lines.push("|--------|-------|----|----|----|");
55
- const top = result.tweets.slice(0, 15);
56
- for (const tweet of top) {
57
- const text = renderUntrustedText(tweet.text, 100);
58
- lines.push(`| @${tweet.author} | ${text} | ${tweet.likes} | ${tweet.retweets} | ${tweet.replies} |`);
172
+ lines.push(`⚠ Stale data (cached at ${metadata.timestamp})`);
173
+ }
174
+ let details = result;
175
+ // Index in sentiment store and append trend context
176
+ try {
177
+ const adapter = new TwitterAdapter();
178
+ const records = adapter.mapToRecords(result, query);
179
+ const pipeline = getSentimentPipeline();
180
+ const pipelineResult = await pipeline.processRecords(records, query);
181
+ if (pipelineResult.insight) {
182
+ const insight = metadata.stale
183
+ ? {
184
+ ...pipelineResult.insight,
185
+ caveats: [
186
+ ...pipelineResult.insight.caveats,
187
+ `Stale data cached at ${metadata.timestamp}.`,
188
+ ],
189
+ }
190
+ : pipelineResult.insight;
191
+ details = { ...result, insight };
192
+ lines.push(...formatInsightSection(insight));
59
193
  }
60
- if (providerResult.stale) {
194
+ if (pipelineResult.trend && pipelineResult.trend.length > 0) {
195
+ const t = pipelineResult.trend[0];
61
196
  lines.push("");
62
- lines.push(`⚠ Stale data (cached at ${providerResult.timestamp})`);
63
- }
64
- // Index in sentiment store and append trend context
65
- try {
66
- const adapter = new TwitterAdapter();
67
- const records = adapter.mapToRecords(result, args.query);
68
- const pipeline = getSentimentPipeline();
69
- const pipelineResult = await pipeline.processRecords(records, args.query);
70
- if (pipelineResult.trend && pipelineResult.trend.length > 0) {
71
- const t = pipelineResult.trend[0];
72
- lines.push("");
73
- lines.push(`Trend: ${t.sparkline} ${t.direction} (${t.delta >= 0 ? "+" : ""}${t.delta.toFixed(2)}, ${t.count} records)`);
74
- }
75
- }
76
- catch {
77
- // Sentiment indexing is best-effort — don't fail the tool
197
+ lines.push(`Trend: ${t.sparkline} ${t.direction} (${t.delta >= 0 ? "+" : ""}${t.delta.toFixed(2)}, ${t.count} records)`);
78
198
  }
79
- return { content: [{ type: "text", text: lines.join("\n") }], details: result };
80
- },
81
- };
199
+ }
200
+ catch {
201
+ // Sentiment indexing is best-effort — don't fail the tool
202
+ }
203
+ return { content: [{ type: "text", text: lines.join("\n") }], details };
204
+ }
82
205
  //# sourceMappingURL=twitter-sentiment.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"twitter-sentiment.js","sourceRoot":"","sources":["../../../src/tools/sentiment/twitter-sentiment.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"twitter-sentiment.js","sourceRoot":"","sources":["../../../src/tools/sentiment/twitter-sentiment.ts"],"names":[],"mappings":"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"}
@@ -2,6 +2,7 @@ import { Type } from "@sinclair/typebox";
2
2
  import { searchWeb } from "../../providers/web-search.js";
3
3
  import { WebAdapter } from "../../sentiment/adapters/web.js";
4
4
  import { getSentimentPipeline } from "../../sentiment/index.js";
5
+ import { formatInsightSection } from "./insight-format.js";
5
6
  import { renderUntrustedText, untrustedContentHeader } from "./untrusted-text.js";
6
7
  const params = Type.Object({
7
8
  query: Type.String({ description: "Ticker or topic to search for web/news sentiment" }),
@@ -42,6 +43,9 @@ export const webSentimentTool = {
42
43
  const avgScore = result.fresh.reduce((s, r) => s + r.sentiment.score, 0) / result.fresh.length;
43
44
  const label = sentimentLabel(avgScore);
44
45
  lines.push(`**Web sentiment for "${args.query}"** — ${result.fresh.length} results (${label}, ${avgScore.toFixed(2)})`);
46
+ if (result.insight) {
47
+ lines.push(...formatInsightSection(result.insight));
48
+ }
45
49
  lines.push("");
46
50
  lines.push(untrustedContentHeader("web sentiment results"));
47
51
  for (const rec of result.fresh.slice(0, limit)) {
@@ -1 +1 @@
1
- {"version":3,"file":"web-sentiment.js","sourceRoot":"","sources":["../../../src/tools/sentiment/web-sentiment.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"web-sentiment.js","sourceRoot":"","sources":["../../../src/tools/sentiment/web-sentiment.ts"],"names":[],"mappings":"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"}
@@ -1,3 +1,4 @@
1
+ import type { SentinelRecord } from "../sentiment/types.js";
1
2
  export interface FearGreedData {
2
3
  value: number;
3
4
  label: string;
@@ -17,6 +18,54 @@ export interface TwitterTweet {
17
18
  url: string;
18
19
  created: string;
19
20
  }
21
+ export type SentimentInsightMethod = "deterministic-keyword-v1" | "llm";
22
+ export type SentimentConfidenceLevel = "low" | "medium" | "high";
23
+ export interface SentimentInsightConfidence {
24
+ level: SentimentConfidenceLevel;
25
+ score: number;
26
+ reasons: string[];
27
+ }
28
+ export interface SentimentInsightDriver {
29
+ label: string;
30
+ count: number;
31
+ polarity: "positive" | "negative" | "mixed";
32
+ terms: string[];
33
+ sourceIds: string[];
34
+ }
35
+ export interface SentimentRepresentativeItem {
36
+ source: "twitter" | "reddit" | "web" | "finnhub" | "aggregate";
37
+ sourceId: string;
38
+ title: string | null;
39
+ excerpt: string;
40
+ url: string | null;
41
+ author: string | null;
42
+ publishedAt: string | null;
43
+ engagement: number | null;
44
+ score: number;
45
+ matchedTerms: string[];
46
+ metadata?: Record<string, unknown>;
47
+ }
48
+ export interface SentimentSourceCoverage {
49
+ sources: string[];
50
+ counts: Record<string, number>;
51
+ missingSources?: string[];
52
+ notes?: string[];
53
+ }
54
+ export interface SentimentInsight {
55
+ label: string;
56
+ score: number;
57
+ sampleSize: number;
58
+ scoredSampleSize: number;
59
+ confidence: SentimentInsightConfidence;
60
+ positiveDrivers: SentimentInsightDriver[];
61
+ negativeDrivers: SentimentInsightDriver[];
62
+ mixedDrivers: SentimentInsightDriver[];
63
+ notableClaims: string[];
64
+ representativeItems: SentimentRepresentativeItem[];
65
+ sourceCoverage: SentimentSourceCoverage;
66
+ caveats: string[];
67
+ method: SentimentInsightMethod;
68
+ }
20
69
  export interface TwitterSentimentResult {
21
70
  query: string;
22
71
  tweetCount: number;
@@ -26,6 +75,7 @@ export interface TwitterSentimentResult {
26
75
  bearishCount: number;
27
76
  topMentions: string[];
28
77
  fetchedAt: string;
78
+ insight?: SentimentInsight;
29
79
  }
30
80
  export interface WebSearchResult {
31
81
  title: string;
@@ -62,4 +112,6 @@ export interface RedditSentimentResult {
62
112
  bullishCount: number;
63
113
  bearishCount: number;
64
114
  fetchedAt: string;
115
+ insight?: SentimentInsight;
116
+ records?: SentinelRecord[];
65
117
  }
@@ -6,6 +6,7 @@ import { Value } from "@sinclair/typebox/value";
6
6
  import { getDefaults } from "../../src/memory/tool-defaults.js";
7
7
  import { wrapWithDefaults } from "../../src/runtime/tool-defaults-wrapper.js";
8
8
  import { getAllTools } from "../../src/tools/index.js";
9
+ import type { AskUserHandler } from "../../src/types/index.js";
9
10
  import { buildToolInvokeAckMessage } from "./tool-invoke-ack.js";
10
11
 
11
12
  export interface InvokeToolResult {
@@ -30,6 +31,7 @@ export interface ToolInvokeControllerOptions {
30
31
  onMarketStateChanged?: () => void;
31
32
  getTools?: typeof getAllTools;
32
33
  invokeTool?: typeof invokeToolFromUi;
34
+ askUserHandler?: AskUserHandler;
33
35
  }
34
36
 
35
37
  export function createToolInvokeController({
@@ -39,6 +41,7 @@ export function createToolInvokeController({
39
41
  onMarketStateChanged,
40
42
  getTools = getAllTools,
41
43
  invokeTool = invokeToolFromUi,
44
+ askUserHandler,
42
45
  }: ToolInvokeControllerOptions): ToolInvokeController {
43
46
  async function handleToolInvoke(
44
47
  toolName: string,
@@ -47,7 +50,7 @@ export function createToolInvokeController({
47
50
  if (role !== "writer") throw new Error("Read-only follower mode");
48
51
  const tool = getTools().find((candidate) => candidate.name === toolName);
49
52
  if (!tool) throw new Error(`Unknown tool: ${toolName}`);
50
- const result = await invokeTool(getSessionManager(), tool, args, "ui");
53
+ const result = await invokeTool(getSessionManager(), tool, args, "ui", { askUserHandler });
51
54
  if (!result.isError && marketStateToolMapping(toolName) != null) {
52
55
  onMarketStateChanged?.();
53
56
  }
@@ -94,7 +97,7 @@ export async function invokeToolFromUi(
94
97
  tool: AgentTool<TSchema, unknown>,
95
98
  args: Record<string, unknown>,
96
99
  source: "ui" | "background" = "ui",
97
- options: { recordTranscript?: boolean } = {},
100
+ options: { askUserHandler?: AskUserHandler; recordTranscript?: boolean } = {},
98
101
  ): Promise<InvokeToolResult> {
99
102
  if (!Value.Check(tool.parameters, args)) {
100
103
  const errors = [...Value.Errors(tool.parameters, args)]
@@ -130,7 +133,18 @@ export async function invokeToolFromUi(
130
133
  let result: AgentToolResult<unknown>;
131
134
  let isError = false;
132
135
  try {
133
- result = await wrapped.execute(toolCallId, args as never);
136
+ result = await (
137
+ wrapped.execute as (
138
+ id: string,
139
+ params: never,
140
+ signal: AbortSignal | undefined,
141
+ onUpdate: undefined,
142
+ ctx: { askUserHandler?: AskUserHandler; hasUI: false },
143
+ ) => ReturnType<typeof wrapped.execute>
144
+ )(toolCallId, args as never, undefined, undefined, {
145
+ askUserHandler: options.askUserHandler,
146
+ hasUI: false,
147
+ });
134
148
  } catch (error) {
135
149
  isError = true;
136
150
  result = {
@@ -1,6 +1,10 @@
1
1
  import type { Api, Model } from "@earendil-works/pi-ai";
2
2
  import { persistProviderCredential } from "../../src/onboarding/connect.js";
3
- import { getCredentialSource, PROVIDERS, type ProviderId } from "../../src/onboarding/providers.js";
3
+ import {
4
+ getCredentialSource,
5
+ isApiKeyProvider,
6
+ PROVIDERS,
7
+ } from "../../src/onboarding/providers.js";
4
8
  import { validateCredential } from "../../src/onboarding/validation.js";
5
9
 
6
10
  export type ModelSetupRequirement = "ready" | "select_model" | "connect_auth";
@@ -193,6 +197,9 @@ export function createModelSetupController({
193
197
 
194
198
  const descriptor = PROVIDERS.find((candidate) => candidate.id === providerId);
195
199
  if (!descriptor) throw new Error(`Unknown provider: ${providerId}`);
200
+ if (!isApiKeyProvider(descriptor)) {
201
+ throw new Error(`${descriptor.displayName} is not configured with an API key.`);
202
+ }
196
203
 
197
204
  if (getCredentialSource(descriptor.id) === "env") {
198
205
  throw new Error(
@@ -203,7 +210,7 @@ export function createModelSetupController({
203
210
  const trimmed = apiKey.trim();
204
211
  if (!trimmed) throw new Error(`Paste a ${descriptor.displayName} API key first.`);
205
212
 
206
- const validation = await validateCredential(descriptor.id as ProviderId, trimmed);
213
+ const validation = await validateCredential(descriptor.id, trimmed);
207
214
  if (validation.status === "invalid") {
208
215
  const statusHint =
209
216
  validation.httpStatus !== undefined ? ` (HTTP ${validation.httpStatus})` : "";
@@ -213,7 +220,7 @@ export function createModelSetupController({
213
220
  );
214
221
  }
215
222
 
216
- persistProviderCredential(descriptor.id as ProviderId, trimmed);
223
+ persistProviderCredential(descriptor.id, trimmed);
217
224
 
218
225
  const verifiedNote =
219
226
  validation.status === "transient"
@@ -36,6 +36,7 @@ const DIRECT_TOOL_GAP_PROVIDERS: Record<string, string> = {
36
36
  compare_companies: "alpha_vantage",
37
37
  get_economic_data: "fred",
38
38
  get_twitter_sentiment: "twitter",
39
+ get_reddit_sentiment: "reddit",
39
40
  };
40
41
 
41
42
  export function createEmptyDashboardState(): DashboardState {
@@ -213,7 +214,8 @@ function parseSkippedProviders(text: string): string[] {
213
214
 
214
215
  function parseSoftGapProviders(text: string): string[] {
215
216
  const providers: string[] = [];
216
- const re = /\[OPENCANDLE_(?:SKIPPED|SOFT_DEGRADED)[^\]]*provider=([a-z0-9_-]+)/gi;
217
+ const re =
218
+ /\[OPENCANDLE_(?:SKIPPED|SOFT_DEGRADED|EXTERNAL_TOOL_REQUIRED)[^\]]*provider=([a-z0-9_-]+)/gi;
217
219
  while (true) {
218
220
  const match = re.exec(text);
219
221
  if (!match) break;
@@ -226,7 +228,9 @@ function inferDirectToolGapProvider(
226
228
  toolName: string | undefined,
227
229
  text: string,
228
230
  ): string | undefined {
229
- if (!toolName || !/(?:⚠|unavailable|No .*data found|LOGIN_NEEDED)/i.test(text)) return undefined;
231
+ if (!toolName || !/(?:⚠|unavailable|No .*data found|EXTERNAL_TOOL_SETUP)/i.test(text)) {
232
+ return undefined;
233
+ }
230
234
  return DIRECT_TOOL_GAP_PROVIDERS[toolName];
231
235
  }
232
236
 
@@ -13,6 +13,7 @@ import {
13
13
  SettingsManager,
14
14
  } from "@earendil-works/pi-coding-agent";
15
15
  import { createOpenCandleSession } from "../../src/index.js";
16
+ import { probeProviderStatus } from "../../src/onboarding/provider-status.js";
16
17
  import type { ChatEvent } from "../shared/chat-events.js";
17
18
  import { createAskUserBridge } from "./ask-user-bridge.js";
18
19
  import {
@@ -117,6 +118,7 @@ const toolInvokeController = createToolInvokeController({
117
118
  getSessionManager: () => sessionManager,
118
119
  broadcastState: () => wsHub.broadcastState(),
119
120
  onMarketStateChanged: () => quoteSnapshotStore.invalidate(),
121
+ askUserHandler: askUserBridge.ask,
120
122
  });
121
123
  const sessionActionsController = createSessionActionsController({
122
124
  role: lockResult.role,
@@ -226,6 +228,22 @@ async function handleHttpRequest(req: IncomingMessage, res: ServerResponse): Pro
226
228
  return;
227
229
  }
228
230
 
231
+ if (url.pathname === "/api/diagnostics/twitter-cli" && req.method === "GET") {
232
+ if (!allowTrustedGuiRequest(req, res, "Diagnostics API")) return;
233
+ const mode = url.searchParams.get("mode") === "session" ? "session" : "install";
234
+ const force = url.searchParams.get("force") === "1";
235
+ writeJson(res, await probeProviderStatus("twitter", { mode, force }));
236
+ return;
237
+ }
238
+
239
+ if (url.pathname === "/api/diagnostics/reddit-cli" && req.method === "GET") {
240
+ if (!allowTrustedGuiRequest(req, res, "Diagnostics API")) return;
241
+ const mode = url.searchParams.get("mode") === "session" ? "session" : "install";
242
+ const force = url.searchParams.get("force") === "1";
243
+ writeJson(res, await probeProviderStatus("reddit", { mode, force }));
244
+ return;
245
+ }
246
+
229
247
  if (url.pathname === "/api/chat/run" && req.method === "POST") {
230
248
  if (!allowTrustedGuiRequest(req, res, "Chat run API")) return;
231
249
  await handleSseChatRun(req, res);