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.
- package/README.md +111 -79
- package/dist/analysts/contracts.d.ts +31 -0
- package/dist/analysts/contracts.js +158 -0
- package/dist/analysts/contracts.js.map +1 -0
- package/dist/analysts/orchestrator.d.ts +11 -2
- package/dist/analysts/orchestrator.js +156 -9
- package/dist/analysts/orchestrator.js.map +1 -1
- package/dist/cli.js +26 -10
- package/dist/cli.js.map +1 -1
- package/dist/config.d.ts +29 -5
- package/dist/config.js +18 -8
- package/dist/config.js.map +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +1 -1
- package/dist/index.js.map +1 -1
- package/dist/infra/cache.d.ts +34 -0
- package/dist/infra/cache.js +44 -3
- package/dist/infra/cache.js.map +1 -1
- package/dist/infra/index.d.ts +1 -1
- package/dist/infra/index.js +1 -1
- package/dist/infra/index.js.map +1 -1
- package/dist/infra/opencandle-paths.d.ts +1 -0
- package/dist/infra/opencandle-paths.js +3 -0
- package/dist/infra/opencandle-paths.js.map +1 -1
- package/dist/infra/rate-limiter.js +7 -0
- package/dist/infra/rate-limiter.js.map +1 -1
- package/dist/memory/index.d.ts +3 -0
- package/dist/memory/index.js +2 -0
- package/dist/memory/index.js.map +1 -1
- package/dist/memory/manager.d.ts +19 -0
- package/dist/memory/manager.js +132 -0
- package/dist/memory/manager.js.map +1 -0
- package/dist/memory/sqlite.js +12 -1
- package/dist/memory/sqlite.js.map +1 -1
- package/dist/memory/types.d.ts +21 -0
- package/dist/memory/types.js +47 -0
- package/dist/memory/types.js.map +1 -0
- package/dist/onboarding/connect.d.ts +23 -0
- package/dist/onboarding/connect.js +107 -0
- package/dist/onboarding/connect.js.map +1 -0
- package/dist/onboarding/credential-interceptor.d.ts +44 -0
- package/dist/onboarding/credential-interceptor.js +72 -0
- package/dist/onboarding/credential-interceptor.js.map +1 -0
- package/dist/onboarding/degradation-accumulator.d.ts +21 -0
- package/dist/onboarding/degradation-accumulator.js +55 -0
- package/dist/onboarding/degradation-accumulator.js.map +1 -0
- package/dist/onboarding/prompt-user.d.ts +23 -0
- package/dist/onboarding/prompt-user.js +61 -0
- package/dist/onboarding/prompt-user.js.map +1 -0
- package/dist/onboarding/providers.d.ts +109 -0
- package/dist/onboarding/providers.js +236 -0
- package/dist/onboarding/providers.js.map +1 -0
- package/dist/onboarding/state.d.ts +31 -2
- package/dist/onboarding/state.js +141 -13
- package/dist/onboarding/state.js.map +1 -1
- package/dist/onboarding/tool-helpers.d.ts +34 -0
- package/dist/onboarding/tool-helpers.js +80 -0
- package/dist/onboarding/tool-helpers.js.map +1 -0
- package/dist/onboarding/tool-tags.d.ts +37 -0
- package/dist/onboarding/tool-tags.js +149 -0
- package/dist/onboarding/tool-tags.js.map +1 -0
- package/dist/onboarding/validation.d.ts +19 -0
- package/dist/onboarding/validation.js +117 -0
- package/dist/onboarding/validation.js.map +1 -0
- package/dist/pi/opencandle-extension.d.ts +5 -1
- package/dist/pi/opencandle-extension.js +345 -143
- package/dist/pi/opencandle-extension.js.map +1 -1
- package/dist/pi/session.d.ts +2 -0
- package/dist/pi/session.js +1 -1
- package/dist/pi/session.js.map +1 -1
- package/dist/pi/setup.d.ts +0 -1
- package/dist/pi/setup.js +66 -119
- package/dist/pi/setup.js.map +1 -1
- package/dist/prompts/context-builder.d.ts +26 -0
- package/dist/prompts/context-builder.js +127 -0
- package/dist/prompts/context-builder.js.map +1 -0
- package/dist/prompts/sections.d.ts +13 -0
- package/dist/prompts/sections.js +35 -0
- package/dist/prompts/sections.js.map +1 -0
- package/dist/providers/alpha-vantage.d.ts +3 -0
- package/dist/providers/alpha-vantage.js +204 -77
- package/dist/providers/alpha-vantage.js.map +1 -1
- package/dist/providers/coingecko.js +53 -37
- package/dist/providers/coingecko.js.map +1 -1
- package/dist/providers/exa-search.d.ts +39 -0
- package/dist/providers/exa-search.js +276 -0
- package/dist/providers/exa-search.js.map +1 -0
- package/dist/providers/fear-greed.js +23 -15
- package/dist/providers/fear-greed.js.map +1 -1
- package/dist/providers/finnhub.d.ts +17 -0
- package/dist/providers/finnhub.js +94 -0
- package/dist/providers/finnhub.js.map +1 -0
- package/dist/providers/fred.js +48 -28
- package/dist/providers/fred.js.map +1 -1
- package/dist/providers/index.d.ts +2 -0
- package/dist/providers/index.js +1 -0
- package/dist/providers/index.js.map +1 -1
- package/dist/providers/provider-credential-error.d.ts +8 -0
- package/dist/providers/provider-credential-error.js +22 -0
- package/dist/providers/provider-credential-error.js.map +1 -0
- package/dist/providers/reddit.d.ts +8 -0
- package/dist/providers/reddit.js +78 -43
- package/dist/providers/reddit.js.map +1 -1
- package/dist/providers/twitter.d.ts +20 -0
- package/dist/providers/twitter.js +137 -0
- package/dist/providers/twitter.js.map +1 -0
- package/dist/providers/web-search.d.ts +17 -0
- package/dist/providers/web-search.js +224 -0
- package/dist/providers/web-search.js.map +1 -0
- package/dist/providers/with-fallback.d.ts +15 -0
- package/dist/providers/with-fallback.js +32 -0
- package/dist/providers/with-fallback.js.map +1 -0
- package/dist/providers/wrap-provider.d.ts +20 -0
- package/dist/providers/wrap-provider.js +58 -0
- package/dist/providers/wrap-provider.js.map +1 -0
- package/dist/providers/yahoo-finance.js +77 -57
- package/dist/providers/yahoo-finance.js.map +1 -1
- package/dist/routing/classify-intent.js +22 -0
- package/dist/routing/classify-intent.js.map +1 -1
- package/dist/runtime/evidence.d.ts +35 -0
- package/dist/runtime/evidence.js +29 -0
- package/dist/runtime/evidence.js.map +1 -0
- package/dist/runtime/index.d.ts +16 -0
- package/dist/runtime/index.js +10 -0
- package/dist/runtime/index.js.map +1 -0
- package/dist/runtime/prompt-step.d.ts +41 -0
- package/dist/runtime/prompt-step.js +42 -0
- package/dist/runtime/prompt-step.js.map +1 -0
- package/dist/runtime/provider-ids.d.ts +14 -0
- package/dist/runtime/provider-ids.js +14 -0
- package/dist/runtime/provider-ids.js.map +1 -0
- package/dist/runtime/provider-tracker.d.ts +20 -0
- package/dist/runtime/provider-tracker.js +36 -0
- package/dist/runtime/provider-tracker.js.map +1 -0
- package/dist/runtime/run-context.d.ts +11 -0
- package/dist/runtime/run-context.js +14 -0
- package/dist/runtime/run-context.js.map +1 -0
- package/dist/runtime/session-coordinator.d.ts +47 -0
- package/dist/runtime/session-coordinator.js +171 -0
- package/dist/runtime/session-coordinator.js.map +1 -0
- package/dist/runtime/validation.d.ts +44 -0
- package/dist/runtime/validation.js +157 -0
- package/dist/runtime/validation.js.map +1 -0
- package/dist/runtime/workflow-events.d.ts +21 -0
- package/dist/runtime/workflow-events.js +31 -0
- package/dist/runtime/workflow-events.js.map +1 -0
- package/dist/runtime/workflow-runner.d.ts +36 -0
- package/dist/runtime/workflow-runner.js +129 -0
- package/dist/runtime/workflow-runner.js.map +1 -0
- package/dist/runtime/workflow-types.d.ts +60 -0
- package/dist/runtime/workflow-types.js +32 -0
- package/dist/runtime/workflow-types.js.map +1 -0
- package/dist/sentiment/adapters/finnhub.d.ts +7 -0
- package/dist/sentiment/adapters/finnhub.js +39 -0
- package/dist/sentiment/adapters/finnhub.js.map +1 -0
- package/dist/sentiment/adapters/reddit.d.ts +11 -0
- package/dist/sentiment/adapters/reddit.js +54 -0
- package/dist/sentiment/adapters/reddit.js.map +1 -0
- package/dist/sentiment/adapters/twitter.d.ts +9 -0
- package/dist/sentiment/adapters/twitter.js +32 -0
- package/dist/sentiment/adapters/twitter.js.map +1 -0
- package/dist/sentiment/adapters/web.d.ts +9 -0
- package/dist/sentiment/adapters/web.js +40 -0
- package/dist/sentiment/adapters/web.js.map +1 -0
- package/dist/sentiment/index.d.ts +16 -0
- package/dist/sentiment/index.js +44 -0
- package/dist/sentiment/index.js.map +1 -0
- package/dist/sentiment/keywords.d.ts +2 -0
- package/dist/sentiment/keywords.js +9 -0
- package/dist/sentiment/keywords.js.map +1 -0
- package/dist/sentiment/pipeline.d.ts +9 -0
- package/dist/sentiment/pipeline.js +57 -0
- package/dist/sentiment/pipeline.js.map +1 -0
- package/dist/sentiment/scorer.d.ts +9 -0
- package/dist/sentiment/scorer.js +64 -0
- package/dist/sentiment/scorer.js.map +1 -0
- package/dist/sentiment/store.d.ts +24 -0
- package/dist/sentiment/store.js +177 -0
- package/dist/sentiment/store.js.map +1 -0
- package/dist/sentiment/trends.d.ts +13 -0
- package/dist/sentiment/trends.js +73 -0
- package/dist/sentiment/trends.js.map +1 -0
- package/dist/sentiment/types.d.ts +66 -0
- package/dist/sentiment/types.js +54 -0
- package/dist/sentiment/types.js.map +1 -0
- package/dist/system-prompt.js +60 -2
- package/dist/system-prompt.js.map +1 -1
- package/dist/tool-kit.d.ts +9 -5
- package/dist/tool-kit.js +29 -6
- package/dist/tool-kit.js.map +1 -1
- package/dist/tools/fundamentals/company-overview.d.ts +3 -1
- package/dist/tools/fundamentals/company-overview.js +28 -17
- package/dist/tools/fundamentals/company-overview.js.map +1 -1
- package/dist/tools/fundamentals/comps.js +47 -39
- package/dist/tools/fundamentals/comps.js.map +1 -1
- package/dist/tools/fundamentals/dcf.js +83 -65
- package/dist/tools/fundamentals/dcf.js.map +1 -1
- package/dist/tools/fundamentals/earnings.d.ts +3 -1
- package/dist/tools/fundamentals/earnings.js +26 -18
- package/dist/tools/fundamentals/earnings.js.map +1 -1
- package/dist/tools/fundamentals/financials.d.ts +3 -1
- package/dist/tools/fundamentals/financials.js +24 -16
- package/dist/tools/fundamentals/financials.js.map +1 -1
- package/dist/tools/fundamentals/sec-filings.js +9 -1
- package/dist/tools/fundamentals/sec-filings.js.map +1 -1
- package/dist/tools/index.js +10 -2
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/interaction/ask-user.d.ts +3 -0
- package/dist/tools/interaction/ask-user.js +51 -0
- package/dist/tools/interaction/ask-user.js.map +1 -0
- package/dist/tools/interaction/twitter-login.d.ts +8 -0
- package/dist/tools/interaction/twitter-login.js +77 -0
- package/dist/tools/interaction/twitter-login.js.map +1 -0
- package/dist/tools/macro/fear-greed.js +9 -1
- package/dist/tools/macro/fear-greed.js.map +1 -1
- package/dist/tools/macro/fred-data.d.ts +3 -1
- package/dist/tools/macro/fred-data.js +27 -16
- package/dist/tools/macro/fred-data.js.map +1 -1
- package/dist/tools/market/crypto-history.js +9 -1
- package/dist/tools/market/crypto-history.js.map +1 -1
- package/dist/tools/market/crypto-price.js +9 -1
- package/dist/tools/market/crypto-price.js.map +1 -1
- package/dist/tools/market/stock-history.js +28 -1
- package/dist/tools/market/stock-history.js.map +1 -1
- package/dist/tools/market/stock-quote.js +29 -4
- package/dist/tools/market/stock-quote.js.map +1 -1
- package/dist/tools/options/option-chain.js +9 -1
- package/dist/tools/options/option-chain.js.map +1 -1
- package/dist/tools/portfolio/correlation.js +15 -3
- package/dist/tools/portfolio/correlation.js.map +1 -1
- package/dist/tools/portfolio/predictions.js +6 -5
- package/dist/tools/portfolio/predictions.js.map +1 -1
- package/dist/tools/portfolio/risk-analysis.js +9 -1
- package/dist/tools/portfolio/risk-analysis.js.map +1 -1
- package/dist/tools/portfolio/tracker.js +6 -3
- package/dist/tools/portfolio/tracker.js.map +1 -1
- package/dist/tools/portfolio/watchlist.js +6 -1
- package/dist/tools/portfolio/watchlist.js.map +1 -1
- package/dist/tools/sentiment/reddit-sentiment.d.ts +3 -1
- package/dist/tools/sentiment/reddit-sentiment.js +112 -19
- package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
- package/dist/tools/sentiment/sentiment-summary.d.ts +7 -0
- package/dist/tools/sentiment/sentiment-summary.js +230 -0
- package/dist/tools/sentiment/sentiment-summary.js.map +1 -0
- package/dist/tools/sentiment/sentiment-trend.d.ts +22 -0
- package/dist/tools/sentiment/sentiment-trend.js +39 -0
- package/dist/tools/sentiment/sentiment-trend.js.map +1 -0
- package/dist/tools/sentiment/twitter-sentiment.d.ts +9 -0
- package/dist/tools/sentiment/twitter-sentiment.js +75 -0
- package/dist/tools/sentiment/twitter-sentiment.js.map +1 -0
- package/dist/tools/sentiment/web-search.d.ts +11 -0
- package/dist/tools/sentiment/web-search.js +115 -0
- package/dist/tools/sentiment/web-search.js.map +1 -0
- package/dist/tools/sentiment/web-sentiment.d.ts +8 -0
- package/dist/tools/sentiment/web-sentiment.js +66 -0
- package/dist/tools/sentiment/web-sentiment.js.map +1 -0
- package/dist/tools/technical/backtest.js +9 -1
- package/dist/tools/technical/backtest.js.map +1 -1
- package/dist/tools/technical/indicators.js +9 -1
- package/dist/tools/technical/indicators.js.map +1 -1
- package/dist/types/index.d.ts +16 -1
- package/dist/types/sentiment.d.ts +41 -0
- package/dist/workflows/compare-assets.d.ts +3 -0
- package/dist/workflows/compare-assets.js +21 -5
- package/dist/workflows/compare-assets.js.map +1 -1
- package/dist/workflows/index.d.ts +3 -3
- package/dist/workflows/index.js +3 -3
- package/dist/workflows/index.js.map +1 -1
- package/dist/workflows/options-screener.d.ts +3 -0
- package/dist/workflows/options-screener.js +24 -7
- package/dist/workflows/options-screener.js.map +1 -1
- package/dist/workflows/portfolio-builder.d.ts +3 -0
- package/dist/workflows/portfolio-builder.js +30 -9
- package/dist/workflows/portfolio-builder.js.map +1 -1
- package/package.json +27 -7
- package/dist/tools/sentiment/news-sentiment.d.ts +0 -7
- package/dist/tools/sentiment/news-sentiment.js +0 -57
- 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 {};
|