opencandle 0.1.1 → 0.3.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 (278) hide show
  1. package/README.md +111 -79
  2. package/dist/analysts/contracts.d.ts +31 -0
  3. package/dist/analysts/contracts.js +158 -0
  4. package/dist/analysts/contracts.js.map +1 -0
  5. package/dist/analysts/orchestrator.d.ts +11 -2
  6. package/dist/analysts/orchestrator.js +156 -9
  7. package/dist/analysts/orchestrator.js.map +1 -1
  8. package/dist/cli.js +26 -10
  9. package/dist/cli.js.map +1 -1
  10. package/dist/config.d.ts +29 -5
  11. package/dist/config.js +18 -8
  12. package/dist/config.js.map +1 -1
  13. package/dist/index.d.ts +1 -1
  14. package/dist/index.js +1 -1
  15. package/dist/index.js.map +1 -1
  16. package/dist/infra/cache.d.ts +34 -0
  17. package/dist/infra/cache.js +44 -3
  18. package/dist/infra/cache.js.map +1 -1
  19. package/dist/infra/index.d.ts +1 -1
  20. package/dist/infra/index.js +1 -1
  21. package/dist/infra/index.js.map +1 -1
  22. package/dist/infra/opencandle-paths.d.ts +1 -0
  23. package/dist/infra/opencandle-paths.js +3 -0
  24. package/dist/infra/opencandle-paths.js.map +1 -1
  25. package/dist/infra/rate-limiter.js +7 -0
  26. package/dist/infra/rate-limiter.js.map +1 -1
  27. package/dist/memory/index.d.ts +3 -0
  28. package/dist/memory/index.js +2 -0
  29. package/dist/memory/index.js.map +1 -1
  30. package/dist/memory/manager.d.ts +19 -0
  31. package/dist/memory/manager.js +132 -0
  32. package/dist/memory/manager.js.map +1 -0
  33. package/dist/memory/sqlite.js +12 -1
  34. package/dist/memory/sqlite.js.map +1 -1
  35. package/dist/memory/types.d.ts +21 -0
  36. package/dist/memory/types.js +47 -0
  37. package/dist/memory/types.js.map +1 -0
  38. package/dist/onboarding/connect.d.ts +23 -0
  39. package/dist/onboarding/connect.js +107 -0
  40. package/dist/onboarding/connect.js.map +1 -0
  41. package/dist/onboarding/credential-interceptor.d.ts +44 -0
  42. package/dist/onboarding/credential-interceptor.js +72 -0
  43. package/dist/onboarding/credential-interceptor.js.map +1 -0
  44. package/dist/onboarding/degradation-accumulator.d.ts +21 -0
  45. package/dist/onboarding/degradation-accumulator.js +55 -0
  46. package/dist/onboarding/degradation-accumulator.js.map +1 -0
  47. package/dist/onboarding/prompt-user.d.ts +23 -0
  48. package/dist/onboarding/prompt-user.js +61 -0
  49. package/dist/onboarding/prompt-user.js.map +1 -0
  50. package/dist/onboarding/providers.d.ts +109 -0
  51. package/dist/onboarding/providers.js +236 -0
  52. package/dist/onboarding/providers.js.map +1 -0
  53. package/dist/onboarding/state.d.ts +31 -2
  54. package/dist/onboarding/state.js +141 -13
  55. package/dist/onboarding/state.js.map +1 -1
  56. package/dist/onboarding/tool-helpers.d.ts +34 -0
  57. package/dist/onboarding/tool-helpers.js +80 -0
  58. package/dist/onboarding/tool-helpers.js.map +1 -0
  59. package/dist/onboarding/tool-tags.d.ts +37 -0
  60. package/dist/onboarding/tool-tags.js +149 -0
  61. package/dist/onboarding/tool-tags.js.map +1 -0
  62. package/dist/onboarding/validation.d.ts +19 -0
  63. package/dist/onboarding/validation.js +117 -0
  64. package/dist/onboarding/validation.js.map +1 -0
  65. package/dist/pi/opencandle-extension.d.ts +5 -1
  66. package/dist/pi/opencandle-extension.js +345 -143
  67. package/dist/pi/opencandle-extension.js.map +1 -1
  68. package/dist/pi/session.d.ts +2 -0
  69. package/dist/pi/session.js +1 -1
  70. package/dist/pi/session.js.map +1 -1
  71. package/dist/pi/setup.d.ts +0 -1
  72. package/dist/pi/setup.js +66 -119
  73. package/dist/pi/setup.js.map +1 -1
  74. package/dist/prompts/context-builder.d.ts +26 -0
  75. package/dist/prompts/context-builder.js +127 -0
  76. package/dist/prompts/context-builder.js.map +1 -0
  77. package/dist/prompts/sections.d.ts +13 -0
  78. package/dist/prompts/sections.js +35 -0
  79. package/dist/prompts/sections.js.map +1 -0
  80. package/dist/providers/alpha-vantage.d.ts +3 -0
  81. package/dist/providers/alpha-vantage.js +204 -77
  82. package/dist/providers/alpha-vantage.js.map +1 -1
  83. package/dist/providers/coingecko.js +53 -37
  84. package/dist/providers/coingecko.js.map +1 -1
  85. package/dist/providers/exa-search.d.ts +39 -0
  86. package/dist/providers/exa-search.js +276 -0
  87. package/dist/providers/exa-search.js.map +1 -0
  88. package/dist/providers/fear-greed.js +23 -15
  89. package/dist/providers/fear-greed.js.map +1 -1
  90. package/dist/providers/finnhub.d.ts +17 -0
  91. package/dist/providers/finnhub.js +94 -0
  92. package/dist/providers/finnhub.js.map +1 -0
  93. package/dist/providers/fred.js +48 -28
  94. package/dist/providers/fred.js.map +1 -1
  95. package/dist/providers/index.d.ts +2 -0
  96. package/dist/providers/index.js +1 -0
  97. package/dist/providers/index.js.map +1 -1
  98. package/dist/providers/provider-credential-error.d.ts +8 -0
  99. package/dist/providers/provider-credential-error.js +22 -0
  100. package/dist/providers/provider-credential-error.js.map +1 -0
  101. package/dist/providers/reddit.d.ts +8 -0
  102. package/dist/providers/reddit.js +78 -43
  103. package/dist/providers/reddit.js.map +1 -1
  104. package/dist/providers/twitter.d.ts +20 -0
  105. package/dist/providers/twitter.js +137 -0
  106. package/dist/providers/twitter.js.map +1 -0
  107. package/dist/providers/web-search.d.ts +17 -0
  108. package/dist/providers/web-search.js +224 -0
  109. package/dist/providers/web-search.js.map +1 -0
  110. package/dist/providers/with-fallback.d.ts +15 -0
  111. package/dist/providers/with-fallback.js +32 -0
  112. package/dist/providers/with-fallback.js.map +1 -0
  113. package/dist/providers/wrap-provider.d.ts +20 -0
  114. package/dist/providers/wrap-provider.js +58 -0
  115. package/dist/providers/wrap-provider.js.map +1 -0
  116. package/dist/providers/yahoo-finance.js +77 -57
  117. package/dist/providers/yahoo-finance.js.map +1 -1
  118. package/dist/routing/classify-intent.js +22 -0
  119. package/dist/routing/classify-intent.js.map +1 -1
  120. package/dist/runtime/evidence.d.ts +35 -0
  121. package/dist/runtime/evidence.js +29 -0
  122. package/dist/runtime/evidence.js.map +1 -0
  123. package/dist/runtime/index.d.ts +16 -0
  124. package/dist/runtime/index.js +10 -0
  125. package/dist/runtime/index.js.map +1 -0
  126. package/dist/runtime/prompt-step.d.ts +41 -0
  127. package/dist/runtime/prompt-step.js +42 -0
  128. package/dist/runtime/prompt-step.js.map +1 -0
  129. package/dist/runtime/provider-ids.d.ts +14 -0
  130. package/dist/runtime/provider-ids.js +14 -0
  131. package/dist/runtime/provider-ids.js.map +1 -0
  132. package/dist/runtime/provider-tracker.d.ts +20 -0
  133. package/dist/runtime/provider-tracker.js +36 -0
  134. package/dist/runtime/provider-tracker.js.map +1 -0
  135. package/dist/runtime/run-context.d.ts +11 -0
  136. package/dist/runtime/run-context.js +14 -0
  137. package/dist/runtime/run-context.js.map +1 -0
  138. package/dist/runtime/session-coordinator.d.ts +47 -0
  139. package/dist/runtime/session-coordinator.js +171 -0
  140. package/dist/runtime/session-coordinator.js.map +1 -0
  141. package/dist/runtime/validation.d.ts +44 -0
  142. package/dist/runtime/validation.js +157 -0
  143. package/dist/runtime/validation.js.map +1 -0
  144. package/dist/runtime/workflow-events.d.ts +21 -0
  145. package/dist/runtime/workflow-events.js +31 -0
  146. package/dist/runtime/workflow-events.js.map +1 -0
  147. package/dist/runtime/workflow-runner.d.ts +36 -0
  148. package/dist/runtime/workflow-runner.js +129 -0
  149. package/dist/runtime/workflow-runner.js.map +1 -0
  150. package/dist/runtime/workflow-types.d.ts +60 -0
  151. package/dist/runtime/workflow-types.js +32 -0
  152. package/dist/runtime/workflow-types.js.map +1 -0
  153. package/dist/sentiment/adapters/finnhub.d.ts +7 -0
  154. package/dist/sentiment/adapters/finnhub.js +39 -0
  155. package/dist/sentiment/adapters/finnhub.js.map +1 -0
  156. package/dist/sentiment/adapters/reddit.d.ts +11 -0
  157. package/dist/sentiment/adapters/reddit.js +54 -0
  158. package/dist/sentiment/adapters/reddit.js.map +1 -0
  159. package/dist/sentiment/adapters/twitter.d.ts +9 -0
  160. package/dist/sentiment/adapters/twitter.js +32 -0
  161. package/dist/sentiment/adapters/twitter.js.map +1 -0
  162. package/dist/sentiment/adapters/web.d.ts +9 -0
  163. package/dist/sentiment/adapters/web.js +40 -0
  164. package/dist/sentiment/adapters/web.js.map +1 -0
  165. package/dist/sentiment/index.d.ts +16 -0
  166. package/dist/sentiment/index.js +44 -0
  167. package/dist/sentiment/index.js.map +1 -0
  168. package/dist/sentiment/keywords.d.ts +2 -0
  169. package/dist/sentiment/keywords.js +9 -0
  170. package/dist/sentiment/keywords.js.map +1 -0
  171. package/dist/sentiment/pipeline.d.ts +9 -0
  172. package/dist/sentiment/pipeline.js +57 -0
  173. package/dist/sentiment/pipeline.js.map +1 -0
  174. package/dist/sentiment/scorer.d.ts +9 -0
  175. package/dist/sentiment/scorer.js +64 -0
  176. package/dist/sentiment/scorer.js.map +1 -0
  177. package/dist/sentiment/store.d.ts +24 -0
  178. package/dist/sentiment/store.js +177 -0
  179. package/dist/sentiment/store.js.map +1 -0
  180. package/dist/sentiment/trends.d.ts +13 -0
  181. package/dist/sentiment/trends.js +73 -0
  182. package/dist/sentiment/trends.js.map +1 -0
  183. package/dist/sentiment/types.d.ts +66 -0
  184. package/dist/sentiment/types.js +54 -0
  185. package/dist/sentiment/types.js.map +1 -0
  186. package/dist/system-prompt.js +60 -2
  187. package/dist/system-prompt.js.map +1 -1
  188. package/dist/tool-kit.d.ts +9 -5
  189. package/dist/tool-kit.js +29 -6
  190. package/dist/tool-kit.js.map +1 -1
  191. package/dist/tools/fundamentals/company-overview.d.ts +3 -1
  192. package/dist/tools/fundamentals/company-overview.js +28 -17
  193. package/dist/tools/fundamentals/company-overview.js.map +1 -1
  194. package/dist/tools/fundamentals/comps.js +47 -39
  195. package/dist/tools/fundamentals/comps.js.map +1 -1
  196. package/dist/tools/fundamentals/dcf.js +83 -65
  197. package/dist/tools/fundamentals/dcf.js.map +1 -1
  198. package/dist/tools/fundamentals/earnings.d.ts +3 -1
  199. package/dist/tools/fundamentals/earnings.js +26 -18
  200. package/dist/tools/fundamentals/earnings.js.map +1 -1
  201. package/dist/tools/fundamentals/financials.d.ts +3 -1
  202. package/dist/tools/fundamentals/financials.js +24 -16
  203. package/dist/tools/fundamentals/financials.js.map +1 -1
  204. package/dist/tools/fundamentals/sec-filings.js +9 -1
  205. package/dist/tools/fundamentals/sec-filings.js.map +1 -1
  206. package/dist/tools/index.js +10 -2
  207. package/dist/tools/index.js.map +1 -1
  208. package/dist/tools/interaction/ask-user.d.ts +3 -0
  209. package/dist/tools/interaction/ask-user.js +51 -0
  210. package/dist/tools/interaction/ask-user.js.map +1 -0
  211. package/dist/tools/interaction/twitter-login.d.ts +8 -0
  212. package/dist/tools/interaction/twitter-login.js +77 -0
  213. package/dist/tools/interaction/twitter-login.js.map +1 -0
  214. package/dist/tools/macro/fear-greed.js +9 -1
  215. package/dist/tools/macro/fear-greed.js.map +1 -1
  216. package/dist/tools/macro/fred-data.d.ts +3 -1
  217. package/dist/tools/macro/fred-data.js +27 -16
  218. package/dist/tools/macro/fred-data.js.map +1 -1
  219. package/dist/tools/market/crypto-history.js +9 -1
  220. package/dist/tools/market/crypto-history.js.map +1 -1
  221. package/dist/tools/market/crypto-price.js +9 -1
  222. package/dist/tools/market/crypto-price.js.map +1 -1
  223. package/dist/tools/market/stock-history.js +28 -1
  224. package/dist/tools/market/stock-history.js.map +1 -1
  225. package/dist/tools/market/stock-quote.js +29 -4
  226. package/dist/tools/market/stock-quote.js.map +1 -1
  227. package/dist/tools/options/option-chain.js +9 -1
  228. package/dist/tools/options/option-chain.js.map +1 -1
  229. package/dist/tools/portfolio/correlation.js +15 -3
  230. package/dist/tools/portfolio/correlation.js.map +1 -1
  231. package/dist/tools/portfolio/predictions.js +6 -5
  232. package/dist/tools/portfolio/predictions.js.map +1 -1
  233. package/dist/tools/portfolio/risk-analysis.js +9 -1
  234. package/dist/tools/portfolio/risk-analysis.js.map +1 -1
  235. package/dist/tools/portfolio/tracker.js +6 -3
  236. package/dist/tools/portfolio/tracker.js.map +1 -1
  237. package/dist/tools/portfolio/watchlist.js +6 -1
  238. package/dist/tools/portfolio/watchlist.js.map +1 -1
  239. package/dist/tools/sentiment/reddit-sentiment.d.ts +3 -1
  240. package/dist/tools/sentiment/reddit-sentiment.js +112 -19
  241. package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
  242. package/dist/tools/sentiment/sentiment-summary.d.ts +7 -0
  243. package/dist/tools/sentiment/sentiment-summary.js +230 -0
  244. package/dist/tools/sentiment/sentiment-summary.js.map +1 -0
  245. package/dist/tools/sentiment/sentiment-trend.d.ts +22 -0
  246. package/dist/tools/sentiment/sentiment-trend.js +39 -0
  247. package/dist/tools/sentiment/sentiment-trend.js.map +1 -0
  248. package/dist/tools/sentiment/twitter-sentiment.d.ts +9 -0
  249. package/dist/tools/sentiment/twitter-sentiment.js +75 -0
  250. package/dist/tools/sentiment/twitter-sentiment.js.map +1 -0
  251. package/dist/tools/sentiment/web-search.d.ts +11 -0
  252. package/dist/tools/sentiment/web-search.js +115 -0
  253. package/dist/tools/sentiment/web-search.js.map +1 -0
  254. package/dist/tools/sentiment/web-sentiment.d.ts +8 -0
  255. package/dist/tools/sentiment/web-sentiment.js +66 -0
  256. package/dist/tools/sentiment/web-sentiment.js.map +1 -0
  257. package/dist/tools/technical/backtest.js +9 -1
  258. package/dist/tools/technical/backtest.js.map +1 -1
  259. package/dist/tools/technical/indicators.js +9 -1
  260. package/dist/tools/technical/indicators.js.map +1 -1
  261. package/dist/types/index.d.ts +16 -1
  262. package/dist/types/sentiment.d.ts +41 -0
  263. package/dist/workflows/compare-assets.d.ts +3 -0
  264. package/dist/workflows/compare-assets.js +21 -5
  265. package/dist/workflows/compare-assets.js.map +1 -1
  266. package/dist/workflows/index.d.ts +3 -3
  267. package/dist/workflows/index.js +3 -3
  268. package/dist/workflows/index.js.map +1 -1
  269. package/dist/workflows/options-screener.d.ts +3 -0
  270. package/dist/workflows/options-screener.js +24 -7
  271. package/dist/workflows/options-screener.js.map +1 -1
  272. package/dist/workflows/portfolio-builder.d.ts +3 -0
  273. package/dist/workflows/portfolio-builder.js +30 -9
  274. package/dist/workflows/portfolio-builder.js.map +1 -1
  275. package/package.json +27 -7
  276. package/dist/tools/sentiment/news-sentiment.d.ts +0 -7
  277. package/dist/tools/sentiment/news-sentiment.js +0 -57
  278. package/dist/tools/sentiment/news-sentiment.js.map +0 -1
@@ -0,0 +1,129 @@
1
+ import { createWorkflowRun, transitionStepStatus, } from "./workflow-types.js";
2
+ let runCounter = 0;
3
+ function generateRunId() {
4
+ runCounter += 1;
5
+ return `run_${Date.now()}_${runCounter}`;
6
+ }
7
+ /**
8
+ * Typed workflow execution engine with run IDs, step definitions,
9
+ * state transitions, cancellation, and event logging.
10
+ */
11
+ export class WorkflowRunner {
12
+ eventLogger;
13
+ providerTracker;
14
+ activeRun = null;
15
+ constructor(options = {}) {
16
+ this.eventLogger = options.eventLogger;
17
+ this.providerTracker = options.providerTracker;
18
+ }
19
+ /** Get the currently active run, if any. */
20
+ getActiveRun() {
21
+ return this.activeRun;
22
+ }
23
+ /**
24
+ * Start a new workflow run. If a run is already active, it is cancelled first.
25
+ */
26
+ async start(workflowType, stepDefinitions, executor) {
27
+ // Cancel any active run
28
+ if (this.activeRun && this.activeRun.status === "running") {
29
+ this.cancel();
30
+ }
31
+ const runId = generateRunId();
32
+ const run = createWorkflowRun(runId, workflowType, stepDefinitions);
33
+ this.activeRun = run;
34
+ run.status = "running";
35
+ this.providerTracker?.resetAll();
36
+ this.logEvent(runId, 0, "workflow_started", {
37
+ workflowType,
38
+ stepCount: stepDefinitions.length,
39
+ });
40
+ // Execute steps
41
+ await this.executeSteps(run, executor);
42
+ return run;
43
+ }
44
+ /** Cancel the active run. */
45
+ cancel() {
46
+ const run = this.activeRun;
47
+ if (!run || run.status !== "running")
48
+ return;
49
+ for (let i = run.currentStepIndex; i < run.steps.length; i++) {
50
+ const step = run.steps[i];
51
+ if (step.status === "pending" || step.status === "running") {
52
+ step.status = transitionStepStatus(step.status, "skipped");
53
+ this.logEvent(run.runId, i, "step_skipped", {
54
+ stepType: step.stepType,
55
+ reason: "cancelled",
56
+ });
57
+ }
58
+ }
59
+ run.status = "cancelled";
60
+ this.logEvent(run.runId, run.currentStepIndex, "workflow_cancelled", {
61
+ cancelledAtStep: run.currentStepIndex,
62
+ });
63
+ }
64
+ async executeSteps(run, executor) {
65
+ for (let i = 0; i < run.steps.length; i++) {
66
+ // Check if run was cancelled externally
67
+ if (run.status !== "running")
68
+ return;
69
+ const step = run.steps[i];
70
+ run.currentStepIndex = i;
71
+ // Collect all prior evidence
72
+ const priorEvidence = [];
73
+ for (const [, output] of run.stepOutputs) {
74
+ priorEvidence.push(...output.evidence);
75
+ }
76
+ // Transition to running
77
+ step.status = transitionStepStatus(step.status, "running");
78
+ this.logEvent(run.runId, i, "step_started", { stepType: step.stepType });
79
+ try {
80
+ const context = {
81
+ runId: run.runId,
82
+ providerTracker: this.providerTracker,
83
+ };
84
+ const output = await executor(step, i, priorEvidence, context);
85
+ // If run was cancelled during execution, stop without further transitions
86
+ if (run.status !== "running")
87
+ return;
88
+ step.status = transitionStepStatus(step.status, "completed");
89
+ run.stepOutputs.set(i, output);
90
+ this.logEvent(run.runId, i, "step_completed", {
91
+ stepType: step.stepType,
92
+ evidenceCount: output.evidence.length,
93
+ });
94
+ }
95
+ catch (error) {
96
+ // If run was cancelled during execution, stop without further transitions
97
+ if (run.status !== "running")
98
+ return;
99
+ const message = error instanceof Error ? error.message : "unknown_error";
100
+ if (step.skippable) {
101
+ step.status = transitionStepStatus(step.status, "skipped");
102
+ this.logEvent(run.runId, i, "step_skipped", {
103
+ stepType: step.stepType,
104
+ reason: message,
105
+ });
106
+ }
107
+ else {
108
+ step.status = transitionStepStatus(step.status, "failed");
109
+ this.logEvent(run.runId, i, "step_failed", {
110
+ stepType: step.stepType,
111
+ error: message,
112
+ });
113
+ run.status = "failed";
114
+ return;
115
+ }
116
+ }
117
+ }
118
+ if (run.status === "running") {
119
+ run.status = "completed";
120
+ this.logEvent(run.runId, run.steps.length - 1, "workflow_completed", {
121
+ workflowType: run.workflowType,
122
+ });
123
+ }
124
+ }
125
+ logEvent(runId, stepIndex, eventType, payload) {
126
+ this.eventLogger?.log(runId, stepIndex, eventType, payload);
127
+ }
128
+ }
129
+ //# sourceMappingURL=workflow-runner.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workflow-runner.js","sourceRoot":"","sources":["../../src/runtime/workflow-runner.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,60 @@
1
+ import type { EvidenceRecord } from "./evidence.js";
2
+ /** Status of a single workflow step. */
3
+ export type StepStatus = "pending" | "running" | "completed" | "failed" | "skipped";
4
+ /** Overall status of a workflow run. */
5
+ export type RunStatus = "pending" | "running" | "completed" | "failed" | "cancelled";
6
+ /** Check whether a step status transition is valid. */
7
+ export declare function isValidStepTransition(from: StepStatus, to: StepStatus): boolean;
8
+ /** Transition a step status, throwing on invalid transitions. */
9
+ export declare function transitionStepStatus(from: StepStatus, to: StepStatus): StepStatus;
10
+ /** Definition of a single workflow step. */
11
+ export interface WorkflowStep {
12
+ stepType: string;
13
+ description: string;
14
+ requiredInputs: string[];
15
+ expectedOutputs: string[];
16
+ skippable: boolean;
17
+ status: StepStatus;
18
+ }
19
+ /** Output produced by a completed workflow step. */
20
+ export interface StepOutput {
21
+ stepIndex: number;
22
+ stepType: string;
23
+ evidence: EvidenceRecord[];
24
+ rawText?: string;
25
+ }
26
+ /** Analyst signal direction. */
27
+ export type AnalystSignal = "BUY" | "HOLD" | "SELL";
28
+ /** Structured output from a single analyst role. */
29
+ export interface AnalystOutput {
30
+ role: string;
31
+ signal: AnalystSignal;
32
+ conviction: number;
33
+ thesis: string;
34
+ evidence: EvidenceRecord[];
35
+ rawText?: string;
36
+ }
37
+ /** Debate side in bull/bear adversarial debate. */
38
+ export type DebateSide = "bull" | "bear";
39
+ /** Structured output from a debate step (eval/test only — not used in live path). */
40
+ export interface DebateOutput {
41
+ side: DebateSide;
42
+ thesis: string;
43
+ keyRisk: string;
44
+ concessions: string[];
45
+ remainingConviction: number;
46
+ evidence: EvidenceRecord[];
47
+ rawText: string;
48
+ }
49
+ /** A complete workflow run definition and state. */
50
+ export interface WorkflowRun {
51
+ runId: string;
52
+ workflowType: string;
53
+ steps: WorkflowStep[];
54
+ currentStepIndex: number;
55
+ status: RunStatus;
56
+ stepOutputs: Map<number, StepOutput>;
57
+ createdAt: string;
58
+ }
59
+ /** Create a new workflow run with all steps in pending state. */
60
+ export declare function createWorkflowRun(runId: string, workflowType: string, stepDefinitions: Omit<WorkflowStep, "status">[]): WorkflowRun;
@@ -0,0 +1,32 @@
1
+ /** Valid step status transitions. */
2
+ const VALID_STEP_TRANSITIONS = {
3
+ pending: ["running", "skipped"],
4
+ running: ["completed", "failed", "skipped"],
5
+ completed: [],
6
+ failed: [],
7
+ skipped: [],
8
+ };
9
+ /** Check whether a step status transition is valid. */
10
+ export function isValidStepTransition(from, to) {
11
+ return VALID_STEP_TRANSITIONS[from].includes(to);
12
+ }
13
+ /** Transition a step status, throwing on invalid transitions. */
14
+ export function transitionStepStatus(from, to) {
15
+ if (!isValidStepTransition(from, to)) {
16
+ throw new Error(`Invalid step transition: ${from} → ${to}`);
17
+ }
18
+ return to;
19
+ }
20
+ /** Create a new workflow run with all steps in pending state. */
21
+ export function createWorkflowRun(runId, workflowType, stepDefinitions) {
22
+ return {
23
+ runId,
24
+ workflowType,
25
+ steps: stepDefinitions.map((def) => ({ ...def, status: "pending" })),
26
+ currentStepIndex: 0,
27
+ status: "pending",
28
+ stepOutputs: new Map(),
29
+ createdAt: new Date().toISOString(),
30
+ };
31
+ }
32
+ //# sourceMappingURL=workflow-types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workflow-types.js","sourceRoot":"","sources":["../../src/runtime/workflow-types.ts"],"names":[],"mappings":"AAQA,qCAAqC;AACrC,MAAM,sBAAsB,GAAqC;IAC/D,OAAO,EAAE,CAAC,SAAS,EAAE,SAAS,CAAC;IAC/B,OAAO,EAAE,CAAC,WAAW,EAAE,QAAQ,EAAE,SAAS,CAAC;IAC3C,SAAS,EAAE,EAAE;IACb,MAAM,EAAE,EAAE;IACV,OAAO,EAAE,EAAE;CACZ,CAAC;AAEF,uDAAuD;AACvD,MAAM,UAAU,qBAAqB,CAAC,IAAgB,EAAE,EAAc;IACpE,OAAO,sBAAsB,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;AACnD,CAAC;AAED,iEAAiE;AACjE,MAAM,UAAU,oBAAoB,CAAC,IAAgB,EAAE,EAAc;IACnE,IAAI,CAAC,qBAAqB,CAAC,IAAI,EAAE,EAAE,CAAC,EAAE,CAAC;QACrC,MAAM,IAAI,KAAK,CAAC,4BAA4B,IAAI,MAAM,EAAE,EAAE,CAAC,CAAC;IAC9D,CAAC;IACD,OAAO,EAAE,CAAC;AACZ,CAAC;AA0DD,iEAAiE;AACjE,MAAM,UAAU,iBAAiB,CAC/B,KAAa,EACb,YAAoB,EACpB,eAA+C;IAE/C,OAAO;QACL,KAAK;QACL,YAAY;QACZ,KAAK,EAAE,eAAe,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,GAAG,EAAE,MAAM,EAAE,SAAkB,EAAE,CAAC,CAAC;QAC7E,gBAAgB,EAAE,CAAC;QACnB,MAAM,EAAE,SAAS;QACjB,WAAW,EAAE,IAAI,GAAG,EAAE;QACtB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;KACpC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,7 @@
1
+ import type { SentinelRecord } from "../types.js";
2
+ import type { FinnhubArticle } from "../../providers/finnhub.js";
3
+ export declare class FinnhubAdapter {
4
+ readonly source: "finnhub";
5
+ mapToRecords(articles: FinnhubArticle[], query: string): SentinelRecord[];
6
+ }
7
+ export declare function extractTickersFromQuery(query: string): string[];
@@ -0,0 +1,39 @@
1
+ import { randomUUID } from "node:crypto";
2
+ import { extractEntities } from "../../routing/entity-extractor.js";
3
+ const MAX_TICKERS = 3;
4
+ export class FinnhubAdapter {
5
+ source = "finnhub";
6
+ mapToRecords(articles, query) {
7
+ const fetchedAt = new Date().toISOString();
8
+ return articles.map((article) => ({
9
+ id: randomUUID(),
10
+ source: this.source,
11
+ sourceId: String(article.id),
12
+ query,
13
+ title: article.headline,
14
+ text: article.summary,
15
+ author: article.source,
16
+ url: article.url,
17
+ publishedAt: new Date(article.datetime * 1000).toISOString(),
18
+ fetchedAt,
19
+ engagement: {
20
+ score: 0,
21
+ replies: null,
22
+ shares: null,
23
+ views: null,
24
+ },
25
+ sentiment: {
26
+ score: 0,
27
+ confidence: 0,
28
+ method: "keyword",
29
+ tickers: article.related ? article.related.split(",").map((t) => t.trim()).filter(Boolean) : [],
30
+ },
31
+ metadata: { category: article.category },
32
+ }));
33
+ }
34
+ }
35
+ export function extractTickersFromQuery(query) {
36
+ const entities = extractEntities(query);
37
+ return entities.symbols.slice(0, MAX_TICKERS);
38
+ }
39
+ //# sourceMappingURL=finnhub.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"finnhub.js","sourceRoot":"","sources":["../../../src/sentiment/adapters/finnhub.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAGzC,OAAO,EAAE,eAAe,EAAE,MAAM,mCAAmC,CAAC;AAEpE,MAAM,WAAW,GAAG,CAAC,CAAC;AAEtB,MAAM,OAAO,cAAc;IAChB,MAAM,GAAG,SAAkB,CAAC;IAErC,YAAY,CAAC,QAA0B,EAAE,KAAa;QACpD,MAAM,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAC3C,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;YAChC,EAAE,EAAE,UAAU,EAAE;YAChB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC;YAC5B,KAAK;YACL,KAAK,EAAE,OAAO,CAAC,QAAQ;YACvB,IAAI,EAAE,OAAO,CAAC,OAAO;YACrB,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,GAAG,EAAE,OAAO,CAAC,GAAG;YAChB,WAAW,EAAE,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,CAAC,WAAW,EAAE;YAC5D,SAAS;YACT,UAAU,EAAE;gBACV,KAAK,EAAE,CAAC;gBACR,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,IAAI;aACZ;YACD,SAAS,EAAE;gBACT,KAAK,EAAE,CAAC;gBACR,UAAU,EAAE,CAAC;gBACb,MAAM,EAAE,SAAkB;gBAC1B,OAAO,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE;aAChG;YACD,QAAQ,EAAE,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE;SACzC,CAAC,CAAC,CAAC;IACN,CAAC;CACF;AAED,MAAM,UAAU,uBAAuB,CAAC,KAAa;IACnD,MAAM,QAAQ,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;IACxC,OAAO,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,WAAW,CAAC,CAAC;AAChD,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { SentinelRecord, SentimentAdapter } from "../types.js";
2
+ import type { RedditSentimentResult } from "../../types/sentiment.js";
3
+ import type { RedditComment } from "../../providers/reddit.js";
4
+ export declare class RedditAdapter implements SentimentAdapter {
5
+ readonly source: "reddit";
6
+ mapPostsToRecords(result: RedditSentimentResult, query: string): SentinelRecord[];
7
+ mapCommentsToRecords(comments: RedditComment[], parentId: string, subreddit: string, query: string): SentinelRecord[];
8
+ fetch(_query: string, _options?: {
9
+ hours?: number;
10
+ }): Promise<SentinelRecord[]>;
11
+ }
@@ -0,0 +1,54 @@
1
+ import { randomUUID } from "node:crypto";
2
+ export class RedditAdapter {
3
+ source = "reddit";
4
+ mapPostsToRecords(result, query) {
5
+ const fetchedAt = result.fetchedAt;
6
+ return result.posts.map((post) => ({
7
+ id: randomUUID(),
8
+ source: this.source,
9
+ sourceId: post.id,
10
+ query,
11
+ title: post.title,
12
+ text: post.selftext ? `${post.title}\n${post.selftext}` : post.title,
13
+ author: post.author,
14
+ url: post.url,
15
+ publishedAt: post.created,
16
+ fetchedAt,
17
+ engagement: {
18
+ score: post.score,
19
+ replies: post.comments,
20
+ shares: null,
21
+ views: null,
22
+ },
23
+ sentiment: { score: 0, confidence: 0, method: "keyword", tickers: [] },
24
+ metadata: { subreddit: result.subreddit },
25
+ }));
26
+ }
27
+ mapCommentsToRecords(comments, parentId, subreddit, query) {
28
+ const fetchedAt = new Date().toISOString();
29
+ return comments.map((comment) => ({
30
+ id: randomUUID(),
31
+ source: this.source,
32
+ sourceId: comment.id,
33
+ query,
34
+ title: null,
35
+ text: comment.body,
36
+ author: comment.author,
37
+ url: comment.permalink,
38
+ publishedAt: null,
39
+ fetchedAt,
40
+ engagement: {
41
+ score: comment.score,
42
+ replies: null,
43
+ shares: null,
44
+ views: null,
45
+ },
46
+ sentiment: { score: 0, confidence: 0, method: "keyword", tickers: [] },
47
+ metadata: { isComment: true, parentId, subreddit },
48
+ }));
49
+ }
50
+ async fetch(_query, _options) {
51
+ throw new Error("Use pipeline.run() instead of adapter.fetch() directly");
52
+ }
53
+ }
54
+ //# sourceMappingURL=reddit.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"reddit.js","sourceRoot":"","sources":["../../../src/sentiment/adapters/reddit.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAKzC,MAAM,OAAO,aAAa;IACf,MAAM,GAAG,QAAiB,CAAC;IAEpC,iBAAiB,CAAC,MAA6B,EAAE,KAAa;QAC5D,MAAM,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;QACnC,OAAO,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;YACjC,EAAE,EAAE,UAAU,EAAE;YAChB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,IAAI,CAAC,EAAE;YACjB,KAAK;YACL,KAAK,EAAE,IAAI,CAAC,KAAK;YACjB,IAAI,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,KAAK,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK;YACpE,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,GAAG,EAAE,IAAI,CAAC,GAAG;YACb,WAAW,EAAE,IAAI,CAAC,OAAO;YACzB,SAAS;YACT,UAAU,EAAE;gBACV,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,OAAO,EAAE,IAAI,CAAC,QAAQ;gBACtB,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,IAAI;aACZ;YACD,SAAS,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,EAAE,SAAkB,EAAE,OAAO,EAAE,EAAE,EAAE;YAC/E,QAAQ,EAAE,EAAE,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE;SAC1C,CAAC,CAAC,CAAC;IACN,CAAC;IAED,oBAAoB,CAClB,QAAyB,EACzB,QAAgB,EAChB,SAAiB,EACjB,KAAa;QAEb,MAAM,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QAC3C,OAAO,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;YAChC,EAAE,EAAE,UAAU,EAAE;YAChB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,OAAO,CAAC,EAAE;YACpB,KAAK;YACL,KAAK,EAAE,IAAI;YACX,IAAI,EAAE,OAAO,CAAC,IAAI;YAClB,MAAM,EAAE,OAAO,CAAC,MAAM;YACtB,GAAG,EAAE,OAAO,CAAC,SAAS;YACtB,WAAW,EAAE,IAAI;YACjB,SAAS;YACT,UAAU,EAAE;gBACV,KAAK,EAAE,OAAO,CAAC,KAAK;gBACpB,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,IAAI;aACZ;YACD,SAAS,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,EAAE,SAAkB,EAAE,OAAO,EAAE,EAAE,EAAE;YAC/E,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,QAAQ,EAAE,SAAS,EAAE;SACnD,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,MAAc,EAAE,QAA6B;QACvD,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;IAC5E,CAAC;CACF"}
@@ -0,0 +1,9 @@
1
+ import type { SentinelRecord, SentimentAdapter } from "../types.js";
2
+ import type { TwitterSentimentResult } from "../../types/sentiment.js";
3
+ export declare class TwitterAdapter implements SentimentAdapter {
4
+ readonly source: "twitter";
5
+ mapToRecords(result: TwitterSentimentResult, query: string): SentinelRecord[];
6
+ fetch(_query: string, _options?: {
7
+ hours?: number;
8
+ }): Promise<SentinelRecord[]>;
9
+ }
@@ -0,0 +1,32 @@
1
+ import { randomUUID } from "node:crypto";
2
+ export class TwitterAdapter {
3
+ source = "twitter";
4
+ mapToRecords(result, query) {
5
+ const fetchedAt = result.fetchedAt;
6
+ return result.tweets.map((tweet) => ({
7
+ id: randomUUID(),
8
+ source: this.source,
9
+ sourceId: tweet.id,
10
+ query,
11
+ title: null,
12
+ text: tweet.text,
13
+ author: tweet.author,
14
+ url: tweet.url,
15
+ publishedAt: tweet.created,
16
+ fetchedAt,
17
+ engagement: {
18
+ score: tweet.likes,
19
+ replies: tweet.replies,
20
+ shares: tweet.retweets,
21
+ views: tweet.views,
22
+ },
23
+ sentiment: { score: 0, confidence: 0, method: "keyword", tickers: [] },
24
+ metadata: {},
25
+ }));
26
+ }
27
+ async fetch(_query, _options) {
28
+ // Actual fetching is done by the pipeline via the provider
29
+ throw new Error("Use pipeline.run() instead of adapter.fetch() directly");
30
+ }
31
+ }
32
+ //# sourceMappingURL=twitter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"twitter.js","sourceRoot":"","sources":["../../../src/sentiment/adapters/twitter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAIzC,MAAM,OAAO,cAAc;IAChB,MAAM,GAAG,SAAkB,CAAC;IAErC,YAAY,CAAC,MAA8B,EAAE,KAAa;QACxD,MAAM,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;QACnC,OAAO,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;YACnC,EAAE,EAAE,UAAU,EAAE;YAChB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,KAAK,CAAC,EAAE;YAClB,KAAK;YACL,KAAK,EAAE,IAAI;YACX,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,GAAG,EAAE,KAAK,CAAC,GAAG;YACd,WAAW,EAAE,KAAK,CAAC,OAAO;YAC1B,SAAS;YACT,UAAU,EAAE;gBACV,KAAK,EAAE,KAAK,CAAC,KAAK;gBAClB,OAAO,EAAE,KAAK,CAAC,OAAO;gBACtB,MAAM,EAAE,KAAK,CAAC,QAAQ;gBACtB,KAAK,EAAE,KAAK,CAAC,KAAK;aACnB;YACD,SAAS,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,EAAE,SAAkB,EAAE,OAAO,EAAE,EAAE,EAAE;YAC/E,QAAQ,EAAE,EAAE;SACb,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,MAAc,EAAE,QAA6B;QACvD,2DAA2D;QAC3D,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;IAC5E,CAAC;CACF"}
@@ -0,0 +1,9 @@
1
+ import type { SentinelRecord, SentimentAdapter } from "../types.js";
2
+ import type { WebSearchEnvelope } from "../../types/sentiment.js";
3
+ export declare class WebAdapter implements SentimentAdapter {
4
+ readonly source: "web";
5
+ mapToRecords(envelope: WebSearchEnvelope, query: string): SentinelRecord[];
6
+ fetch(_query: string, _options?: {
7
+ hours?: number;
8
+ }): Promise<SentinelRecord[]>;
9
+ }
@@ -0,0 +1,40 @@
1
+ import { randomUUID } from "node:crypto";
2
+ export class WebAdapter {
3
+ source = "web";
4
+ mapToRecords(envelope, query) {
5
+ const fetchedAt = envelope.fetchedAt;
6
+ return envelope.results.map((result) => ({
7
+ id: randomUUID(),
8
+ source: this.source,
9
+ sourceId: canonicalizeUrl(result.url),
10
+ query,
11
+ title: result.title,
12
+ text: result.snippet,
13
+ author: result.source,
14
+ url: result.url,
15
+ publishedAt: result.published,
16
+ fetchedAt,
17
+ engagement: {
18
+ score: 0,
19
+ replies: null,
20
+ shares: null,
21
+ views: null,
22
+ },
23
+ sentiment: { score: 0, confidence: 0, method: "keyword", tickers: [] },
24
+ metadata: { category: result.category, provider: envelope.provider },
25
+ }));
26
+ }
27
+ async fetch(_query, _options) {
28
+ throw new Error("Use pipeline.run() instead of adapter.fetch() directly");
29
+ }
30
+ }
31
+ function canonicalizeUrl(url) {
32
+ try {
33
+ const parsed = new URL(url);
34
+ return parsed.origin + parsed.pathname;
35
+ }
36
+ catch {
37
+ return url;
38
+ }
39
+ }
40
+ //# sourceMappingURL=web.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web.js","sourceRoot":"","sources":["../../../src/sentiment/adapters/web.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,aAAa,CAAC;AAIzC,MAAM,OAAO,UAAU;IACZ,MAAM,GAAG,KAAc,CAAC;IAEjC,YAAY,CAAC,QAA2B,EAAE,KAAa;QACrD,MAAM,SAAS,GAAG,QAAQ,CAAC,SAAS,CAAC;QACrC,OAAO,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;YACvC,EAAE,EAAE,UAAU,EAAE;YAChB,MAAM,EAAE,IAAI,CAAC,MAAM;YACnB,QAAQ,EAAE,eAAe,CAAC,MAAM,CAAC,GAAG,CAAC;YACrC,KAAK;YACL,KAAK,EAAE,MAAM,CAAC,KAAK;YACnB,IAAI,EAAE,MAAM,CAAC,OAAO;YACpB,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,GAAG,EAAE,MAAM,CAAC,GAAG;YACf,WAAW,EAAE,MAAM,CAAC,SAAS;YAC7B,SAAS;YACT,UAAU,EAAE;gBACV,KAAK,EAAE,CAAC;gBACR,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,IAAI;aACZ;YACD,SAAS,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,EAAE,MAAM,EAAE,SAAkB,EAAE,OAAO,EAAE,EAAE,EAAE;YAC/E,QAAQ,EAAE,EAAE,QAAQ,EAAE,MAAM,CAAC,QAAQ,EAAE,QAAQ,EAAE,QAAQ,CAAC,QAAQ,EAAE;SACrE,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,KAAK,CAAC,MAAc,EAAE,QAA6B;QACvD,MAAM,IAAI,KAAK,CAAC,wDAAwD,CAAC,CAAC;IAC5E,CAAC;CACF;AAED,SAAS,eAAe,CAAC,GAAW;IAClC,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;QAC5B,OAAO,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC,QAAQ,CAAC;IACzC,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,GAAG,CAAC;IACb,CAAC;AACH,CAAC"}
@@ -0,0 +1,16 @@
1
+ export type { SentinelRecord, SentinelEngagement, SentinelSentiment, SentimentAdapter, ScorerOptions, TrendBucket, TrendResult, DivergenceResult, SentimentSummary, SentimentSource, } from "./types.js";
2
+ export { isSentinelRecord, SENTIMENT_SOURCES } from "./types.js";
3
+ export { SentimentStore } from "./store.js";
4
+ export { scoreRecords, keywordScore } from "./scorer.js";
5
+ export { SentimentPipeline } from "./pipeline.js";
6
+ export { renderSparkline, computeTrend, computeDivergence } from "./trends.js";
7
+ export { BULLISH_TERMS, BEARISH_TERMS } from "./keywords.js";
8
+ export { TwitterAdapter } from "./adapters/twitter.js";
9
+ export { RedditAdapter } from "./adapters/reddit.js";
10
+ export { WebAdapter } from "./adapters/web.js";
11
+ import { SentimentStore } from "./store.js";
12
+ import { SentimentPipeline } from "./pipeline.js";
13
+ export declare function getSentimentStore(): SentimentStore;
14
+ export declare function getSentimentPipeline(): SentimentPipeline;
15
+ /** For testing: reset singletons */
16
+ export declare function _resetSentimentSingletons(): void;
@@ -0,0 +1,44 @@
1
+ export { isSentinelRecord, SENTIMENT_SOURCES } from "./types.js";
2
+ export { SentimentStore } from "./store.js";
3
+ export { scoreRecords, keywordScore } from "./scorer.js";
4
+ export { SentimentPipeline } from "./pipeline.js";
5
+ export { renderSparkline, computeTrend, computeDivergence } from "./trends.js";
6
+ export { BULLISH_TERMS, BEARISH_TERMS } from "./keywords.js";
7
+ export { TwitterAdapter } from "./adapters/twitter.js";
8
+ export { RedditAdapter } from "./adapters/reddit.js";
9
+ export { WebAdapter } from "./adapters/web.js";
10
+ import { SentimentStore } from "./store.js";
11
+ import { SentimentPipeline } from "./pipeline.js";
12
+ import { getConfig } from "../config.js";
13
+ import { resolveOpenCandlePath } from "../infra/opencandle-paths.js";
14
+ let _pipeline = null;
15
+ let _store = null;
16
+ export function getSentimentStore() {
17
+ if (!_store) {
18
+ const dbPath = resolveOpenCandlePath("sentinel.db");
19
+ _store = new SentimentStore(dbPath);
20
+ const config = getConfig();
21
+ _store.prune(config.sentiment?.retentionDays ?? 30);
22
+ }
23
+ return _store;
24
+ }
25
+ export function getSentimentPipeline() {
26
+ if (!_pipeline) {
27
+ const store = getSentimentStore();
28
+ const config = getConfig();
29
+ _pipeline = new SentimentPipeline(store, config.sentiment);
30
+ }
31
+ return _pipeline;
32
+ }
33
+ /** For testing: reset singletons */
34
+ export function _resetSentimentSingletons() {
35
+ if (_store) {
36
+ try {
37
+ _store.close();
38
+ }
39
+ catch { /* ignore */ }
40
+ }
41
+ _pipeline = null;
42
+ _store = null;
43
+ }
44
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sentiment/index.ts"],"names":[],"mappings":"AAaA,OAAO,EAAE,gBAAgB,EAAE,iBAAiB,EAAE,MAAM,YAAY,CAAC;AACjE,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,EAAE,YAAY,EAAE,YAAY,EAAE,MAAM,aAAa,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,eAAe,EAAE,YAAY,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAC/E,OAAO,EAAE,aAAa,EAAE,aAAa,EAAE,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAE,cAAc,EAAE,MAAM,uBAAuB,CAAC;AACvD,OAAO,EAAE,aAAa,EAAE,MAAM,sBAAsB,CAAC;AACrD,OAAO,EAAE,UAAU,EAAE,MAAM,mBAAmB,CAAC;AAE/C,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AACzC,OAAO,EAAE,qBAAqB,EAAE,MAAM,8BAA8B,CAAC;AAErE,IAAI,SAAS,GAA6B,IAAI,CAAC;AAC/C,IAAI,MAAM,GAA0B,IAAI,CAAC;AAEzC,MAAM,UAAU,iBAAiB;IAC/B,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,MAAM,GAAG,qBAAqB,CAAC,aAAa,CAAC,CAAC;QACpD,MAAM,GAAG,IAAI,cAAc,CAAC,MAAM,CAAC,CAAC;QACpC,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;QAC3B,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,EAAE,aAAa,IAAI,EAAE,CAAC,CAAC;IACtD,CAAC;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,UAAU,oBAAoB;IAClC,IAAI,CAAC,SAAS,EAAE,CAAC;QACf,MAAM,KAAK,GAAG,iBAAiB,EAAE,CAAC;QAClC,MAAM,MAAM,GAAG,SAAS,EAAE,CAAC;QAC3B,SAAS,GAAG,IAAI,iBAAiB,CAAC,KAAK,EAAE,MAAM,CAAC,SAAU,CAAC,CAAC;IAC9D,CAAC;IACD,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,oCAAoC;AACpC,MAAM,UAAU,yBAAyB;IACvC,IAAI,MAAM,EAAE,CAAC;QACX,IAAI,CAAC;YAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QAAC,CAAC;QAAC,MAAM,CAAC,CAAC,YAAY,CAAC,CAAC;IAChD,CAAC;IACD,SAAS,GAAG,IAAI,CAAC;IACjB,MAAM,GAAG,IAAI,CAAC;AAChB,CAAC"}
@@ -0,0 +1,2 @@
1
+ export declare const BULLISH_TERMS: readonly ["moon", "buy", "undervalued", "breakout", "calls", "bullish", "rocket", "diamond hands", "accumulate", "dip buy", "long", "rip", "squeeze"];
2
+ export declare const BEARISH_TERMS: readonly ["crash", "overvalued", "sell", "puts", "bearish", "bubble", "dump", "short", "bagholding", "exit", "drill", "tank", "rug"];
@@ -0,0 +1,9 @@
1
+ export const BULLISH_TERMS = [
2
+ "moon", "buy", "undervalued", "breakout", "calls", "bullish",
3
+ "rocket", "diamond hands", "accumulate", "dip buy", "long", "rip", "squeeze",
4
+ ];
5
+ export const BEARISH_TERMS = [
6
+ "crash", "overvalued", "sell", "puts", "bearish", "bubble",
7
+ "dump", "short", "bagholding", "exit", "drill", "tank", "rug",
8
+ ];
9
+ //# sourceMappingURL=keywords.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"keywords.js","sourceRoot":"","sources":["../../src/sentiment/keywords.ts"],"names":[],"mappings":"AAAA,MAAM,CAAC,MAAM,aAAa,GAAG;IAC3B,MAAM,EAAE,KAAK,EAAE,aAAa,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS;IAC5D,QAAQ,EAAE,eAAe,EAAE,YAAY,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,SAAS;CACpE,CAAC;AAEX,MAAM,CAAC,MAAM,aAAa,GAAG;IAC3B,OAAO,EAAE,YAAY,EAAE,MAAM,EAAE,MAAM,EAAE,SAAS,EAAE,QAAQ;IAC1D,MAAM,EAAE,OAAO,EAAE,YAAY,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,KAAK;CACrD,CAAC"}
@@ -0,0 +1,9 @@
1
+ import type { SentinelRecord, SentimentSummary } from "./types.js";
2
+ import type { SentimentConfig } from "../config.js";
3
+ import { SentimentStore } from "./store.js";
4
+ export declare class SentimentPipeline {
5
+ private store;
6
+ private config;
7
+ constructor(store: SentimentStore, config: SentimentConfig);
8
+ processRecords(records: SentinelRecord[], query: string): Promise<SentimentSummary>;
9
+ }
@@ -0,0 +1,57 @@
1
+ import { scoreRecords } from "./scorer.js";
2
+ import { computeTrend, computeDivergence } from "./trends.js";
3
+ export class SentimentPipeline {
4
+ store;
5
+ config;
6
+ constructor(store, config) {
7
+ this.store = store;
8
+ this.config = config;
9
+ }
10
+ async processRecords(records, query) {
11
+ const warnings = [];
12
+ if (records.length === 0) {
13
+ return { fresh: [], trend: null, divergence: null, warnings };
14
+ }
15
+ // Check if store had prior data before this fetch
16
+ const priorSeries = this.store.getTimeSeries(query, { days: 30, bucketHours: 24 });
17
+ const hadPriorData = priorSeries.length >= 2;
18
+ // Score all records
19
+ const scored = scoreRecords(records);
20
+ // Insert into store
21
+ this.store.insert(scored);
22
+ // Compute trend from historical data (only if we had prior data)
23
+ let trend = null;
24
+ if (hadPriorData) {
25
+ const series = this.store.getTimeSeries(query, { days: 7, bucketHours: 24 });
26
+ if (series.length >= 2) {
27
+ trend = [computeTrend(series, "aggregate")];
28
+ }
29
+ }
30
+ // Compute divergence from fresh records
31
+ let divergence = null;
32
+ const sourceGroups = groupBySource(scored);
33
+ const sourceStats = {};
34
+ for (const [source, recs] of Object.entries(sourceGroups)) {
35
+ // Exclude comments from divergence calculation
36
+ const postLevel = recs.filter((r) => !r.metadata.isComment);
37
+ if (postLevel.length > 0) {
38
+ const avg = postLevel.reduce((sum, r) => sum + r.sentiment.score, 0) / postLevel.length;
39
+ sourceStats[source] = { avg, count: postLevel.length };
40
+ }
41
+ }
42
+ if (Object.keys(sourceStats).length >= 2) {
43
+ divergence = computeDivergence(sourceStats, this.config.divergenceThreshold);
44
+ }
45
+ return { fresh: scored, trend, divergence, warnings };
46
+ }
47
+ }
48
+ function groupBySource(records) {
49
+ const groups = {};
50
+ for (const r of records) {
51
+ if (!groups[r.source])
52
+ groups[r.source] = [];
53
+ groups[r.source].push(r);
54
+ }
55
+ return groups;
56
+ }
57
+ //# sourceMappingURL=pipeline.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pipeline.js","sourceRoot":"","sources":["../../src/sentiment/pipeline.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,9 @@
1
+ import type { SentinelRecord } from "./types.js";
2
+ interface ScoreResult {
3
+ score: number;
4
+ confidence: number;
5
+ tickers: string[];
6
+ }
7
+ export declare function keywordScore(record: SentinelRecord): ScoreResult;
8
+ export declare function scoreRecords(records: SentinelRecord[]): SentinelRecord[];
9
+ export {};