opencandle 0.5.0 → 0.7.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +170 -186
- package/dist/analysts/contracts.d.ts +1 -3
- package/dist/analysts/contracts.js +1 -11
- package/dist/analysts/contracts.js.map +1 -1
- package/dist/analysts/orchestrator.d.ts +1 -3
- package/dist/analysts/orchestrator.js +1 -26
- package/dist/analysts/orchestrator.js.map +1 -1
- package/dist/cli.js +66 -7
- package/dist/cli.js.map +1 -1
- package/dist/config.d.ts +13 -3
- package/dist/config.js +25 -5
- 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 +8 -11
- package/dist/infra/cache.js +17 -15
- package/dist/infra/cache.js.map +1 -1
- package/dist/infra/http-client.d.ts +4 -1
- package/dist/infra/http-client.js +59 -6
- package/dist/infra/http-client.js.map +1 -1
- package/dist/infra/index.d.ts +2 -3
- package/dist/infra/index.js +2 -3
- package/dist/infra/index.js.map +1 -1
- package/dist/infra/native-dependencies.js +2 -2
- package/dist/infra/native-dependencies.js.map +1 -1
- package/dist/infra/node-version.js.map +1 -1
- package/dist/infra/opencandle-paths.d.ts +0 -3
- package/dist/infra/opencandle-paths.js +4 -11
- package/dist/infra/opencandle-paths.js.map +1 -1
- package/dist/infra/rate-limiter.js +12 -9
- package/dist/infra/rate-limiter.js.map +1 -1
- package/dist/market-state/alert-conditions.d.ts +34 -0
- package/dist/market-state/alert-conditions.js +23 -0
- package/dist/market-state/alert-conditions.js.map +1 -0
- package/dist/market-state/alert-runner.d.ts +55 -0
- package/dist/market-state/alert-runner.js +634 -0
- package/dist/market-state/alert-runner.js.map +1 -0
- package/dist/market-state/daily-report.d.ts +26 -0
- package/dist/market-state/daily-report.js +179 -0
- package/dist/market-state/daily-report.js.map +1 -0
- package/dist/market-state/local-automation-service.d.ts +25 -0
- package/dist/market-state/local-automation-service.js +119 -0
- package/dist/market-state/local-automation-service.js.map +1 -0
- package/dist/market-state/notification-delivery.d.ts +14 -0
- package/dist/market-state/notification-delivery.js +139 -0
- package/dist/market-state/notification-delivery.js.map +1 -0
- package/dist/market-state/resolve-for-mutation.d.ts +10 -0
- package/dist/market-state/resolve-for-mutation.js +15 -0
- package/dist/market-state/resolve-for-mutation.js.map +1 -0
- package/dist/market-state/resolve.d.ts +14 -0
- package/dist/market-state/resolve.js +89 -0
- package/dist/market-state/resolve.js.map +1 -0
- package/dist/market-state/service.d.ts +527 -0
- package/dist/market-state/service.js +1099 -0
- package/dist/market-state/service.js.map +1 -0
- package/dist/memory/index.d.ts +7 -7
- package/dist/memory/index.js +6 -6
- package/dist/memory/index.js.map +1 -1
- package/dist/memory/manager.js +11 -11
- package/dist/memory/manager.js.map +1 -1
- package/dist/memory/retrieval.js +7 -4
- package/dist/memory/retrieval.js.map +1 -1
- package/dist/memory/sqlite.js +385 -3
- package/dist/memory/sqlite.js.map +1 -1
- package/dist/memory/storage.js +1 -2
- package/dist/memory/storage.js.map +1 -1
- package/dist/memory/tool-defaults.js +64 -28
- package/dist/memory/tool-defaults.js.map +1 -1
- package/dist/memory/types.js.map +1 -1
- package/dist/monitor.d.ts +2 -0
- package/dist/monitor.js +104 -0
- package/dist/monitor.js.map +1 -0
- package/dist/onboarding/connect.d.ts +2 -2
- package/dist/onboarding/connect.js +13 -8
- package/dist/onboarding/connect.js.map +1 -1
- package/dist/onboarding/credential-interceptor.js +1 -1
- package/dist/onboarding/credential-interceptor.js.map +1 -1
- package/dist/onboarding/degradation-accumulator.js +1 -3
- package/dist/onboarding/degradation-accumulator.js.map +1 -1
- package/dist/onboarding/provider-status.d.ts +48 -0
- package/dist/onboarding/provider-status.js +285 -0
- package/dist/onboarding/provider-status.js.map +1 -0
- package/dist/onboarding/providers.d.ts +85 -8
- package/dist/onboarding/providers.js +83 -18
- package/dist/onboarding/providers.js.map +1 -1
- package/dist/onboarding/state.d.ts +1 -0
- package/dist/onboarding/state.js +5 -0
- package/dist/onboarding/state.js.map +1 -1
- package/dist/onboarding/tool-helpers.js +1 -1
- package/dist/onboarding/tool-helpers.js.map +1 -1
- package/dist/onboarding/tool-tags.d.ts +12 -1
- package/dist/onboarding/tool-tags.js +37 -5
- package/dist/onboarding/tool-tags.js.map +1 -1
- package/dist/onboarding/validation.d.ts +2 -2
- package/dist/onboarding/validation.js +1 -1
- package/dist/onboarding/validation.js.map +1 -1
- package/dist/pi/opencandle-extension.d.ts +8 -0
- package/dist/pi/opencandle-extension.js +502 -42
- package/dist/pi/opencandle-extension.js.map +1 -1
- package/dist/pi/session.d.ts +1 -1
- package/dist/pi/session.js +3 -1
- package/dist/pi/session.js.map +1 -1
- package/dist/pi/setup.js +8 -3
- package/dist/pi/setup.js.map +1 -1
- package/dist/pi/tool-adapter.d.ts +4 -1
- package/dist/pi/tool-adapter.js +10 -6
- package/dist/pi/tool-adapter.js.map +1 -1
- package/dist/prompts/context-builder.d.ts +1 -1
- package/dist/prompts/context-builder.js +20 -7
- package/dist/prompts/context-builder.js.map +1 -1
- package/dist/prompts/policy-cards.d.ts +1 -1
- package/dist/prompts/policy-cards.js +2 -2
- package/dist/prompts/policy-cards.js.map +1 -1
- package/dist/prompts/sections.d.ts +1 -1
- package/dist/prompts/symbol-preflight.d.ts +20 -0
- package/dist/prompts/symbol-preflight.js +49 -0
- package/dist/prompts/symbol-preflight.js.map +1 -0
- package/dist/prompts/workflow-prompts.d.ts +1 -1
- package/dist/prompts/workflow-prompts.js +54 -16
- package/dist/prompts/workflow-prompts.js.map +1 -1
- package/dist/providers/alpha-vantage.d.ts +1 -1
- package/dist/providers/alpha-vantage.js +26 -7
- package/dist/providers/alpha-vantage.js.map +1 -1
- package/dist/providers/coingecko.js +1 -1
- package/dist/providers/coingecko.js.map +1 -1
- package/dist/providers/errors.d.ts +5 -0
- package/dist/providers/errors.js +11 -0
- package/dist/providers/errors.js.map +1 -0
- package/dist/providers/exa-search.d.ts +2 -2
- package/dist/providers/exa-search.js +19 -11
- package/dist/providers/exa-search.js.map +1 -1
- package/dist/providers/external-tool-error.d.ts +10 -0
- package/dist/providers/external-tool-error.js +21 -0
- package/dist/providers/external-tool-error.js.map +1 -0
- package/dist/providers/fear-greed.js +1 -1
- package/dist/providers/fear-greed.js.map +1 -1
- package/dist/providers/finnhub.js +3 -5
- package/dist/providers/finnhub.js.map +1 -1
- package/dist/providers/fred.js +2 -2
- package/dist/providers/fred.js.map +1 -1
- package/dist/providers/index.d.ts +7 -6
- package/dist/providers/index.js +6 -5
- package/dist/providers/index.js.map +1 -1
- package/dist/providers/reddit-cli.d.ts +36 -0
- package/dist/providers/reddit-cli.js +201 -0
- package/dist/providers/reddit-cli.js.map +1 -0
- package/dist/providers/reddit.d.ts +1 -1
- package/dist/providers/reddit.js +9 -37
- package/dist/providers/reddit.js.map +1 -1
- package/dist/providers/sec-edgar.d.ts +1 -0
- package/dist/providers/sec-edgar.js +12 -4
- package/dist/providers/sec-edgar.js.map +1 -1
- package/dist/providers/tradingview.d.ts +47 -0
- package/dist/providers/tradingview.js +275 -0
- package/dist/providers/tradingview.js.map +1 -0
- package/dist/providers/twitter-cli.d.ts +40 -0
- package/dist/providers/twitter-cli.js +153 -0
- package/dist/providers/twitter-cli.js.map +1 -0
- package/dist/providers/twitter.d.ts +0 -8
- package/dist/providers/twitter.js +8 -60
- package/dist/providers/twitter.js.map +1 -1
- package/dist/providers/web-search.js +26 -12
- package/dist/providers/web-search.js.map +1 -1
- package/dist/providers/with-fallback.js +4 -2
- package/dist/providers/with-fallback.js.map +1 -1
- package/dist/providers/wrap-provider.d.ts +2 -3
- package/dist/providers/wrap-provider.js +44 -8
- package/dist/providers/wrap-provider.js.map +1 -1
- package/dist/providers/yahoo-finance.d.ts +1 -1
- package/dist/providers/yahoo-finance.js +153 -48
- package/dist/providers/yahoo-finance.js.map +1 -1
- package/dist/routing/classify-intent.d.ts +6 -0
- package/dist/routing/classify-intent.js +78 -7
- package/dist/routing/classify-intent.js.map +1 -1
- package/dist/routing/defaults.d.ts +1 -1
- package/dist/routing/entity-extractor.d.ts +1 -0
- package/dist/routing/entity-extractor.js +234 -29
- package/dist/routing/entity-extractor.js.map +1 -1
- package/dist/routing/fund-symbols.d.ts +2 -0
- package/dist/routing/fund-symbols.js +55 -0
- package/dist/routing/fund-symbols.js.map +1 -0
- package/dist/routing/horizon.d.ts +1 -0
- package/dist/routing/horizon.js +10 -0
- package/dist/routing/horizon.js.map +1 -0
- package/dist/routing/index.d.ts +10 -10
- package/dist/routing/index.js +6 -6
- package/dist/routing/index.js.map +1 -1
- package/dist/routing/planning.d.ts +2 -2
- package/dist/routing/planning.js +65 -34
- package/dist/routing/planning.js.map +1 -1
- package/dist/routing/route-manifest.d.ts +2 -2
- package/dist/routing/route-manifest.js +25 -4
- package/dist/routing/route-manifest.js.map +1 -1
- package/dist/routing/router-llm-client.js.map +1 -1
- package/dist/routing/router-prompt.js +7 -9
- package/dist/routing/router-prompt.js.map +1 -1
- package/dist/routing/router-types.d.ts +1 -0
- package/dist/routing/router.js +137 -22
- package/dist/routing/router.js.map +1 -1
- package/dist/routing/slot-resolver.d.ts +1 -1
- package/dist/routing/slot-resolver.js +2 -4
- package/dist/routing/slot-resolver.js.map +1 -1
- package/dist/routing/symbol-disambiguator.d.ts +11 -0
- package/dist/routing/symbol-disambiguator.js +52 -0
- package/dist/routing/symbol-disambiguator.js.map +1 -0
- package/dist/routing/turn-context.d.ts +1 -1
- package/dist/routing/turn-context.js +1 -1
- package/dist/routing/turn-context.js.map +1 -1
- package/dist/routing/types.d.ts +2 -0
- package/dist/runtime/answer-contracts.d.ts +1 -1
- package/dist/runtime/answer-contracts.js +48 -9
- package/dist/runtime/answer-contracts.js.map +1 -1
- package/dist/runtime/artifact-contracts.js.map +1 -1
- package/dist/runtime/planning-evidence.js +47 -26
- package/dist/runtime/planning-evidence.js.map +1 -1
- package/dist/runtime/prompt-step.d.ts +1 -9
- package/dist/runtime/prompt-step.js +0 -10
- package/dist/runtime/prompt-step.js.map +1 -1
- package/dist/runtime/run-context.d.ts +5 -2
- package/dist/runtime/run-context.js +8 -1
- package/dist/runtime/run-context.js.map +1 -1
- package/dist/runtime/session-coordinator.d.ts +13 -5
- package/dist/runtime/session-coordinator.js +160 -20
- package/dist/runtime/session-coordinator.js.map +1 -1
- package/dist/runtime/session-title.d.ts +14 -0
- package/dist/runtime/session-title.js +50 -0
- package/dist/runtime/session-title.js.map +1 -0
- package/dist/runtime/tool-defaults-wrapper.js +7 -5
- package/dist/runtime/tool-defaults-wrapper.js.map +1 -1
- package/dist/runtime/validation.js.map +1 -1
- package/dist/runtime/workflow-events.js.map +1 -1
- package/dist/runtime/workflow-runner.d.ts +3 -3
- package/dist/runtime/workflow-runner.js +1 -1
- package/dist/runtime/workflow-runner.js.map +1 -1
- package/dist/sentiment/adapters/finnhub.d.ts +1 -1
- package/dist/sentiment/adapters/finnhub.js +6 -1
- package/dist/sentiment/adapters/finnhub.js.map +1 -1
- package/dist/sentiment/adapters/reddit.d.ts +2 -2
- package/dist/sentiment/adapters/twitter.d.ts +1 -1
- package/dist/sentiment/adapters/web.d.ts +1 -1
- package/dist/sentiment/index.d.ts +10 -11
- package/dist/sentiment/index.js +10 -20
- package/dist/sentiment/index.js.map +1 -1
- package/dist/sentiment/insights.d.ts +17 -0
- package/dist/sentiment/insights.js +206 -0
- package/dist/sentiment/insights.js.map +1 -0
- package/dist/sentiment/keywords.js +26 -4
- package/dist/sentiment/keywords.js.map +1 -1
- package/dist/sentiment/pipeline.d.ts +2 -2
- package/dist/sentiment/pipeline.js +14 -2
- package/dist/sentiment/pipeline.js.map +1 -1
- package/dist/sentiment/scorer.d.ts +2 -0
- package/dist/sentiment/scorer.js +11 -2
- package/dist/sentiment/scorer.js.map +1 -1
- package/dist/sentiment/store.d.ts +1 -1
- package/dist/sentiment/store.js +1 -1
- package/dist/sentiment/store.js.map +1 -1
- package/dist/sentiment/trends.d.ts +1 -1
- package/dist/sentiment/trends.js.map +1 -1
- package/dist/sentiment/types.d.ts +2 -0
- package/dist/sentiment/types.js.map +1 -1
- package/dist/system-prompt.js +6 -9
- package/dist/system-prompt.js.map +1 -1
- package/dist/tool-kit.d.ts +7 -7
- package/dist/tool-kit.js +4 -4
- package/dist/tool-kit.js.map +1 -1
- package/dist/tools/fundamentals/company-overview.js +11 -6
- package/dist/tools/fundamentals/company-overview.js.map +1 -1
- package/dist/tools/fundamentals/comps.js +18 -9
- package/dist/tools/fundamentals/comps.js.map +1 -1
- package/dist/tools/fundamentals/dcf.js +23 -11
- package/dist/tools/fundamentals/dcf.js.map +1 -1
- package/dist/tools/fundamentals/earnings.js +8 -3
- package/dist/tools/fundamentals/earnings.js.map +1 -1
- package/dist/tools/fundamentals/financials.js +8 -3
- package/dist/tools/fundamentals/financials.js.map +1 -1
- package/dist/tools/fundamentals/sec-filings.js +21 -6
- package/dist/tools/fundamentals/sec-filings.js.map +1 -1
- package/dist/tools/index.d.ts +27 -20
- package/dist/tools/index.js +55 -43
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/interaction/ask-user.js +15 -3
- package/dist/tools/interaction/ask-user.js.map +1 -1
- package/dist/tools/macro/fear-greed.js.map +1 -1
- package/dist/tools/macro/fred-data.d.ts +1 -1
- package/dist/tools/macro/fred-data.js +17 -6
- package/dist/tools/macro/fred-data.js.map +1 -1
- package/dist/tools/market/crypto-history.js +3 -1
- package/dist/tools/market/crypto-history.js.map +1 -1
- package/dist/tools/market/crypto-price.js +3 -1
- package/dist/tools/market/crypto-price.js.map +1 -1
- package/dist/tools/market/screen-stocks.d.ts +18 -0
- package/dist/tools/market/screen-stocks.js +252 -0
- package/dist/tools/market/screen-stocks.js.map +1 -0
- package/dist/tools/market/search-ticker.js +160 -8
- package/dist/tools/market/search-ticker.js.map +1 -1
- package/dist/tools/market/stock-history.d.ts +2 -2
- package/dist/tools/market/stock-history.js +26 -7
- package/dist/tools/market/stock-history.js.map +1 -1
- package/dist/tools/market/stock-quote.js +5 -3
- package/dist/tools/market/stock-quote.js.map +1 -1
- package/dist/tools/options/greeks.js +1 -1
- package/dist/tools/options/greeks.js.map +1 -1
- package/dist/tools/options/option-chain.js +19 -6
- package/dist/tools/options/option-chain.js.map +1 -1
- package/dist/tools/portfolio/alerts.d.ts +15 -0
- package/dist/tools/portfolio/alerts.js +357 -0
- package/dist/tools/portfolio/alerts.js.map +1 -0
- package/dist/tools/portfolio/correlation.d.ts +1 -1
- package/dist/tools/portfolio/correlation.js +33 -13
- package/dist/tools/portfolio/correlation.js.map +1 -1
- package/dist/tools/portfolio/daily-report.d.ts +8 -0
- package/dist/tools/portfolio/daily-report.js +83 -0
- package/dist/tools/portfolio/daily-report.js.map +1 -0
- package/dist/tools/portfolio/holdings-overlap.js +10 -3
- package/dist/tools/portfolio/holdings-overlap.js.map +1 -1
- package/dist/tools/portfolio/notifications.d.ts +7 -0
- package/dist/tools/portfolio/notifications.js +43 -0
- package/dist/tools/portfolio/notifications.js.map +1 -0
- package/dist/tools/portfolio/predictions.d.ts +12 -6
- package/dist/tools/portfolio/predictions.js +337 -87
- package/dist/tools/portfolio/predictions.js.map +1 -1
- package/dist/tools/portfolio/risk-analysis.d.ts +1 -1
- package/dist/tools/portfolio/risk-analysis.js +45 -6
- package/dist/tools/portfolio/risk-analysis.js.map +1 -1
- package/dist/tools/portfolio/tracker.d.ts +4 -3
- package/dist/tools/portfolio/tracker.js +246 -101
- package/dist/tools/portfolio/tracker.js.map +1 -1
- package/dist/tools/portfolio/watchlist.d.ts +6 -4
- package/dist/tools/portfolio/watchlist.js +208 -108
- package/dist/tools/portfolio/watchlist.js.map +1 -1
- package/dist/tools/sentiment/insight-format.d.ts +2 -0
- package/dist/tools/sentiment/insight-format.js +36 -0
- package/dist/tools/sentiment/insight-format.js.map +1 -0
- package/dist/tools/sentiment/query-match.d.ts +3 -0
- package/dist/tools/sentiment/query-match.js +113 -0
- package/dist/tools/sentiment/query-match.js.map +1 -0
- package/dist/tools/sentiment/reddit-sentiment.d.ts +12 -1
- package/dist/tools/sentiment/reddit-sentiment.js +266 -107
- package/dist/tools/sentiment/reddit-sentiment.js.map +1 -1
- package/dist/tools/sentiment/sentiment-summary.d.ts +9 -1
- package/dist/tools/sentiment/sentiment-summary.js +223 -205
- package/dist/tools/sentiment/sentiment-summary.js.map +1 -1
- package/dist/tools/sentiment/sentiment-trend.d.ts +1 -1
- package/dist/tools/sentiment/sentiment-trend.js +12 -2
- package/dist/tools/sentiment/sentiment-trend.js.map +1 -1
- package/dist/tools/sentiment/twitter-sentiment.d.ts +11 -1
- package/dist/tools/sentiment/twitter-sentiment.js +188 -58
- package/dist/tools/sentiment/twitter-sentiment.js.map +1 -1
- package/dist/tools/sentiment/untrusted-text.d.ts +2 -0
- package/dist/tools/sentiment/untrusted-text.js +17 -0
- package/dist/tools/sentiment/untrusted-text.js.map +1 -0
- package/dist/tools/sentiment/web-search.js +9 -13
- package/dist/tools/sentiment/web-search.js.map +1 -1
- package/dist/tools/sentiment/web-sentiment.js +19 -3
- package/dist/tools/sentiment/web-sentiment.js.map +1 -1
- package/dist/tools/technical/backtest.d.ts +1 -1
- package/dist/tools/technical/backtest.js +27 -20
- package/dist/tools/technical/backtest.js.map +1 -1
- package/dist/tools/technical/indicators.js +23 -5
- package/dist/tools/technical/indicators.js.map +1 -1
- package/dist/types/index.d.ts +3 -3
- package/dist/types/index.js.map +1 -1
- package/dist/types/market.d.ts +1 -0
- package/dist/types/portfolio.d.ts +14 -4
- package/dist/types/sentiment.d.ts +52 -0
- package/dist/workflows/compare-assets.d.ts +0 -3
- package/dist/workflows/compare-assets.js +20 -11
- package/dist/workflows/compare-assets.js.map +1 -1
- package/dist/workflows/index.d.ts +3 -4
- package/dist/workflows/index.js +3 -3
- package/dist/workflows/index.js.map +1 -1
- package/dist/workflows/options-screener.d.ts +0 -3
- package/dist/workflows/options-screener.js +4 -11
- package/dist/workflows/options-screener.js.map +1 -1
- package/dist/workflows/portfolio-builder.d.ts +0 -3
- package/dist/workflows/portfolio-builder.js +0 -8
- package/dist/workflows/portfolio-builder.js.map +1 -1
- package/gui/server/ask-user-bridge.ts +1 -1
- package/gui/server/automation-heartbeat.ts +97 -0
- package/gui/server/background-quotes.ts +97 -1
- package/gui/server/chat-event-adapter.ts +32 -10
- package/gui/server/chat-run-session.ts +16 -0
- package/gui/server/invoke-tool.ts +160 -3
- package/gui/server/live-chat-event-adapter.ts +21 -6
- package/gui/server/market-state-api.ts +315 -0
- package/gui/server/model-setup.ts +156 -2
- package/gui/server/private-api-access.ts +62 -0
- package/gui/server/projector.ts +18 -9
- package/gui/server/prompt-observation.ts +4 -7
- package/gui/server/quote-snapshot-store.ts +50 -0
- package/gui/server/server.ts +218 -451
- package/gui/server/session-actions.ts +186 -1
- package/gui/server/shutdown.ts +47 -0
- package/gui/server/tool-invoke-ack.ts +49 -0
- package/gui/server/tool-metadata.ts +101 -24
- package/gui/server/websocket.ts +13 -3
- package/gui/server/writer-lock.ts +6 -2
- package/gui/server/ws-hub.ts +311 -0
- package/gui/shared/chat-events.ts +16 -1
- package/gui/shared/event-reducer.ts +24 -6
- package/gui/web/dist/assets/CatalogOverlay-CgeY5Pkp.js +1 -0
- package/gui/web/dist/assets/index-C6W_2eAn.js +69 -0
- package/gui/web/dist/assets/index-hwbx24a5.css +1 -0
- package/gui/web/dist/index.html +2 -2
- package/package.json +9 -6
- package/src/analysts/contracts.ts +10 -23
- package/src/analysts/orchestrator.ts +8 -43
- package/src/cli.ts +76 -12
- package/src/config.ts +44 -9
- package/src/index.ts +1 -1
- package/src/infra/cache.ts +41 -30
- package/src/infra/http-client.ts +72 -6
- package/src/infra/index.ts +6 -10
- package/src/infra/native-dependencies.ts +8 -3
- package/src/infra/node-version.ts +3 -1
- package/src/infra/opencandle-paths.ts +3 -14
- package/src/infra/rate-limiter.ts +22 -19
- package/src/market-state/alert-conditions.ts +82 -0
- package/src/market-state/alert-runner.ts +863 -0
- package/src/market-state/daily-report.ts +247 -0
- package/src/market-state/local-automation-service.ts +162 -0
- package/src/market-state/notification-delivery.ts +158 -0
- package/src/market-state/resolve-for-mutation.ts +24 -0
- package/src/market-state/resolve.ts +112 -0
- package/src/market-state/service.ts +2344 -0
- package/src/memory/index.ts +7 -7
- package/src/memory/manager.ts +14 -16
- package/src/memory/retrieval.ts +8 -7
- package/src/memory/sqlite.ts +407 -6
- package/src/memory/storage.ts +5 -15
- package/src/memory/tool-defaults.ts +60 -39
- package/src/memory/types.ts +3 -3
- package/src/monitor.ts +121 -0
- package/src/onboarding/connect.ts +24 -31
- package/src/onboarding/credential-interceptor.ts +3 -15
- package/src/onboarding/degradation-accumulator.ts +1 -3
- package/src/onboarding/provider-status.ts +410 -0
- package/src/onboarding/providers.ts +144 -45
- package/src/onboarding/state.ts +13 -15
- package/src/onboarding/tool-helpers.ts +2 -9
- package/src/onboarding/tool-tags.ts +51 -8
- package/src/onboarding/validation.ts +16 -22
- package/src/pi/opencandle-extension.ts +643 -101
- package/src/pi/session.ts +7 -5
- package/src/pi/setup.ts +61 -43
- package/src/pi/tool-adapter.ts +19 -6
- package/src/prompts/context-builder.ts +24 -13
- package/src/prompts/policy-cards.ts +3 -3
- package/src/prompts/sections.ts +1 -1
- package/src/prompts/symbol-preflight.ts +80 -0
- package/src/prompts/workflow-prompts.ts +77 -28
- package/src/providers/alpha-vantage.ts +58 -39
- package/src/providers/coingecko.ts +2 -5
- package/src/providers/errors.ts +9 -0
- package/src/providers/exa-search.ts +24 -22
- package/src/providers/external-tool-error.ts +20 -0
- package/src/providers/fear-greed.ts +1 -1
- package/src/providers/finnhub.ts +7 -6
- package/src/providers/fred.ts +3 -3
- package/src/providers/index.ts +14 -6
- package/src/providers/reddit-cli.ts +317 -0
- package/src/providers/reddit.ts +14 -59
- package/src/providers/sec-edgar.ts +20 -6
- package/src/providers/tradingview.ts +399 -0
- package/src/providers/twitter-cli.ts +233 -0
- package/src/providers/twitter.ts +8 -79
- package/src/providers/web-search.ts +30 -20
- package/src/providers/with-fallback.ts +8 -7
- package/src/providers/wrap-provider.ts +49 -10
- package/src/providers/yahoo-finance.ts +204 -66
- package/src/routing/classify-intent.ts +101 -10
- package/src/routing/defaults.ts +1 -1
- package/src/routing/entity-extractor.ts +287 -38
- package/src/routing/fund-symbols.ts +58 -0
- package/src/routing/horizon.ts +7 -0
- package/src/routing/index.ts +48 -48
- package/src/routing/planning.ts +145 -53
- package/src/routing/route-manifest.ts +37 -15
- package/src/routing/router-llm-client.ts +4 -4
- package/src/routing/router-prompt.ts +15 -19
- package/src/routing/router-types.ts +2 -5
- package/src/routing/router.ts +251 -53
- package/src/routing/slot-resolver.ts +34 -11
- package/src/routing/symbol-disambiguator.ts +72 -0
- package/src/routing/turn-context.ts +6 -9
- package/src/routing/types.ts +2 -0
- package/src/runtime/answer-contracts.ts +105 -45
- package/src/runtime/artifact-contracts.ts +2 -1
- package/src/runtime/planning-evidence.ts +157 -66
- package/src/runtime/prompt-step.ts +1 -16
- package/src/runtime/run-context.ts +12 -2
- package/src/runtime/session-coordinator.ts +238 -63
- package/src/runtime/session-title.ts +60 -0
- package/src/runtime/tool-defaults-wrapper.ts +13 -5
- package/src/runtime/validation.ts +1 -4
- package/src/runtime/workflow-events.ts +7 -7
- package/src/runtime/workflow-runner.ts +5 -11
- package/src/sentiment/adapters/finnhub.ts +7 -2
- package/src/sentiment/adapters/reddit.ts +2 -2
- package/src/sentiment/adapters/twitter.ts +1 -1
- package/src/sentiment/adapters/web.ts +1 -1
- package/src/sentiment/index.ts +17 -26
- package/src/sentiment/insights.ts +269 -0
- package/src/sentiment/keywords.ts +26 -4
- package/src/sentiment/pipeline.ts +28 -5
- package/src/sentiment/scorer.ts +13 -2
- package/src/sentiment/store.ts +2 -2
- package/src/sentiment/trends.ts +9 -3
- package/src/sentiment/types.ts +8 -4
- package/src/system-prompt.ts +6 -9
- package/src/tool-kit.ts +10 -9
- package/src/tools/fundamentals/company-overview.ts +19 -9
- package/src/tools/fundamentals/comps.ts +68 -55
- package/src/tools/fundamentals/dcf.ts +145 -95
- package/src/tools/fundamentals/earnings.ts +16 -6
- package/src/tools/fundamentals/financials.ts +16 -7
- package/src/tools/fundamentals/sec-filings.ts +37 -16
- package/src/tools/index.ts +56 -43
- package/src/tools/interaction/ask-user.ts +22 -10
- package/src/tools/macro/fear-greed.ts +1 -1
- package/src/tools/macro/fred-data.ts +58 -46
- package/src/tools/market/crypto-history.ts +8 -3
- package/src/tools/market/crypto-price.ts +6 -6
- package/src/tools/market/screen-stocks.ts +279 -0
- package/src/tools/market/search-ticker.ts +218 -17
- package/src/tools/market/stock-history.ts +37 -12
- package/src/tools/market/stock-quote.ts +10 -7
- package/src/tools/options/greeks.ts +5 -5
- package/src/tools/options/option-chain.ts +41 -17
- package/src/tools/portfolio/alerts.ts +457 -0
- package/src/tools/portfolio/correlation.ts +47 -20
- package/src/tools/portfolio/daily-report.ts +101 -0
- package/src/tools/portfolio/holdings-overlap.ts +31 -15
- package/src/tools/portfolio/notifications.ts +45 -0
- package/src/tools/portfolio/predictions.ts +406 -106
- package/src/tools/portfolio/risk-analysis.ts +46 -7
- package/src/tools/portfolio/tracker.ts +270 -109
- package/src/tools/portfolio/watchlist.ts +250 -121
- package/src/tools/sentiment/insight-format.ts +50 -0
- package/src/tools/sentiment/query-match.ts +117 -0
- package/src/tools/sentiment/reddit-sentiment.ts +360 -121
- package/src/tools/sentiment/sentiment-summary.ts +302 -235
- package/src/tools/sentiment/sentiment-trend.ts +24 -7
- package/src/tools/sentiment/twitter-sentiment.ts +264 -73
- package/src/tools/sentiment/untrusted-text.ts +21 -0
- package/src/tools/sentiment/web-search.ts +21 -18
- package/src/tools/sentiment/web-sentiment.ts +30 -10
- package/src/tools/technical/backtest.ts +32 -22
- package/src/tools/technical/indicators.ts +39 -14
- package/src/types/index.ts +8 -3
- package/src/types/market.ts +1 -0
- package/src/types/portfolio.ts +14 -4
- package/src/types/sentiment.ts +61 -2
- package/src/workflows/compare-assets.ts +33 -21
- package/src/workflows/index.ts +3 -4
- package/src/workflows/options-screener.ts +27 -29
- package/src/workflows/portfolio-builder.ts +34 -27
- package/dist/infra/browser.d.ts +0 -35
- package/dist/infra/browser.js +0 -103
- package/dist/infra/browser.js.map +0 -1
- package/dist/tools/interaction/twitter-login.d.ts +0 -8
- package/dist/tools/interaction/twitter-login.js +0 -77
- package/dist/tools/interaction/twitter-login.js.map +0 -1
- package/dist/workflows/types.d.ts +0 -4
- package/dist/workflows/types.js +0 -2
- package/dist/workflows/types.js.map +0 -1
- package/gui/web/dist/assets/CatalogOverlay-Bmp6Knu7.js +0 -1
- package/gui/web/dist/assets/index-Bxt9QpLX.css +0 -1
- package/gui/web/dist/assets/index-CZ9DHZYy.js +0 -67
- package/src/infra/browser.ts +0 -111
- package/src/tools/interaction/twitter-login.ts +0 -93
- package/src/workflows/types.ts +0 -4
|
@@ -1,21 +1,24 @@
|
|
|
1
|
-
import { Type } from "@sinclair/typebox";
|
|
2
1
|
import type { AgentTool } from "@earendil-works/pi-agent-core";
|
|
3
|
-
import {
|
|
2
|
+
import { Type } from "@sinclair/typebox";
|
|
4
3
|
import { wrapProvider } from "../../providers/wrap-provider.js";
|
|
5
|
-
import
|
|
4
|
+
import { getOptionsChain } from "../../providers/yahoo-finance.js";
|
|
5
|
+
import type { OptionContract, OptionsChain } from "../../types/options.js";
|
|
6
6
|
|
|
7
7
|
const params = Type.Object({
|
|
8
8
|
symbol: Type.String({ description: "Stock ticker symbol (e.g. AAPL, TSLA, SPY, MSFT)" }),
|
|
9
9
|
expiration: Type.Optional(
|
|
10
10
|
Type.String({
|
|
11
|
-
|
|
12
|
-
|
|
11
|
+
pattern: "^\\d{4}-\\d{2}-\\d{2}$",
|
|
12
|
+
description: "Expiration date as YYYY-MM-DD. If omitted, uses the nearest expiration.",
|
|
13
13
|
}),
|
|
14
14
|
),
|
|
15
15
|
type: Type.Optional(
|
|
16
|
-
Type.Union(
|
|
17
|
-
|
|
18
|
-
|
|
16
|
+
Type.Union(
|
|
17
|
+
[Type.Literal("call"), Type.Literal("put"), Type.Literal("CALL"), Type.Literal("PUT")],
|
|
18
|
+
{
|
|
19
|
+
description: "Filter by option type. Omit for both calls and puts.",
|
|
20
|
+
},
|
|
21
|
+
),
|
|
19
22
|
),
|
|
20
23
|
});
|
|
21
24
|
|
|
@@ -28,14 +31,14 @@ export const optionChainTool: AgentTool<typeof params, OptionsChain> = {
|
|
|
28
31
|
async execute(_toolCallId, args) {
|
|
29
32
|
const symbol = args.symbol.toUpperCase();
|
|
30
33
|
const normalizedType = args.type?.toLowerCase();
|
|
31
|
-
const expirationTs = args.expiration
|
|
32
|
-
? Math.floor(new Date(args.expiration).getTime() / 1000)
|
|
33
|
-
: undefined;
|
|
34
|
+
const expirationTs = args.expiration ? parseExpiration(args.expiration) : undefined;
|
|
34
35
|
|
|
35
36
|
const result = await wrapProvider("yahoo", () => getOptionsChain(symbol, expirationTs));
|
|
36
37
|
if (result.status === "unavailable") {
|
|
37
38
|
return {
|
|
38
|
-
content: [
|
|
39
|
+
content: [
|
|
40
|
+
{ type: "text", text: `⚠ Options chain unavailable for ${symbol} (${result.reason}).` },
|
|
41
|
+
],
|
|
39
42
|
details: null as any,
|
|
40
43
|
};
|
|
41
44
|
}
|
|
@@ -47,7 +50,9 @@ export const optionChainTool: AgentTool<typeof params, OptionsChain> = {
|
|
|
47
50
|
`Quote status: ${chain.quoteStatus.marketSession} / ${chain.quoteStatus.bidAskState}`,
|
|
48
51
|
"Option bid/ask and last prices are quoted per share; multiply by 100 for one standard contract premium.",
|
|
49
52
|
...(chain.quoteStatus.warning
|
|
50
|
-
? [
|
|
53
|
+
? [
|
|
54
|
+
`⚠ ${chain.quoteStatus.warning} do not treat zero bid/ask as confirmed live illiquidity without broker verification; do not stop at the stale quote caveat. Disclose the gap, avoid naming tradable live premiums, and finish the strategy explanation with mechanics, assignment outcomes, labeled hypotheticals, and what live broker quotes would change.`,
|
|
55
|
+
]
|
|
51
56
|
: []),
|
|
52
57
|
`Available expirations: ${formatAvailableExpirations(chain.expirationDates)}`,
|
|
53
58
|
"",
|
|
@@ -57,8 +62,12 @@ export const optionChainTool: AgentTool<typeof params, OptionsChain> = {
|
|
|
57
62
|
const showPuts = !normalizedType || normalizedType === "put";
|
|
58
63
|
|
|
59
64
|
if (showCalls && chain.calls.length > 0) {
|
|
60
|
-
lines.push(
|
|
61
|
-
|
|
65
|
+
lines.push(
|
|
66
|
+
`**CALLS** (${chain.calls.length} contracts, volume: ${chain.totalCallVolume.toLocaleString()})`,
|
|
67
|
+
);
|
|
68
|
+
lines.push(
|
|
69
|
+
"Strike | Bid/Ask (per share) | Last (per share) | Vol | OI | IV | Delta | Gamma | Theta | Vega | Rho",
|
|
70
|
+
);
|
|
62
71
|
const topCalls = sortByVolume(chain.calls).slice(0, 10);
|
|
63
72
|
for (const c of topCalls) {
|
|
64
73
|
lines.push(formatContract(c));
|
|
@@ -67,8 +76,12 @@ export const optionChainTool: AgentTool<typeof params, OptionsChain> = {
|
|
|
67
76
|
}
|
|
68
77
|
|
|
69
78
|
if (showPuts && chain.puts.length > 0) {
|
|
70
|
-
lines.push(
|
|
71
|
-
|
|
79
|
+
lines.push(
|
|
80
|
+
`**PUTS** (${chain.puts.length} contracts, volume: ${chain.totalPutVolume.toLocaleString()})`,
|
|
81
|
+
);
|
|
82
|
+
lines.push(
|
|
83
|
+
"Strike | Bid/Ask (per share) | Last (per share) | Vol | OI | IV | Delta | Gamma | Theta | Vega | Rho",
|
|
84
|
+
);
|
|
72
85
|
const topPuts = sortByVolume(chain.puts).slice(0, 10);
|
|
73
86
|
for (const c of topPuts) {
|
|
74
87
|
lines.push(formatContract(c));
|
|
@@ -82,6 +95,17 @@ export const optionChainTool: AgentTool<typeof params, OptionsChain> = {
|
|
|
82
95
|
},
|
|
83
96
|
};
|
|
84
97
|
|
|
98
|
+
function parseExpiration(expiration: string): number {
|
|
99
|
+
if (!/^\d{4}-\d{2}-\d{2}$/.test(expiration)) {
|
|
100
|
+
throw new Error("expiration must be a valid YYYY-MM-DD date.");
|
|
101
|
+
}
|
|
102
|
+
const parsed = new Date(`${expiration}T00:00:00.000Z`);
|
|
103
|
+
if (Number.isNaN(parsed.getTime()) || parsed.toISOString().slice(0, 10) !== expiration) {
|
|
104
|
+
throw new Error("expiration must be a valid YYYY-MM-DD date.");
|
|
105
|
+
}
|
|
106
|
+
return Math.floor(parsed.getTime() / 1000);
|
|
107
|
+
}
|
|
108
|
+
|
|
85
109
|
function sortByVolume(contracts: OptionContract[]): OptionContract[] {
|
|
86
110
|
return [...contracts].sort((a, b) => b.volume - a.volume);
|
|
87
111
|
}
|
|
@@ -0,0 +1,457 @@
|
|
|
1
|
+
import type { AgentTool } from "@earendil-works/pi-agent-core";
|
|
2
|
+
import { Type } from "@sinclair/typebox";
|
|
3
|
+
import {
|
|
4
|
+
ALERT_CONDITION_VERSION,
|
|
5
|
+
percentMove,
|
|
6
|
+
priceCrossesAbove,
|
|
7
|
+
priceCrossesBelow,
|
|
8
|
+
priceCrossesSma,
|
|
9
|
+
rsiThreshold,
|
|
10
|
+
smaCross,
|
|
11
|
+
volumeSpike,
|
|
12
|
+
} from "../../market-state/alert-conditions.js";
|
|
13
|
+
import { defaultAlertRunnerProviders, runAlertChecks } from "../../market-state/alert-runner.js";
|
|
14
|
+
import { deliverPendingNotifications } from "../../market-state/notification-delivery.js";
|
|
15
|
+
import { resolveInstrumentForMutation } from "../../market-state/resolve-for-mutation.js";
|
|
16
|
+
import { type AlertRuleRecord, MarketStateService } from "../../market-state/service.js";
|
|
17
|
+
import { initDefaultDatabase } from "../../memory/sqlite.js";
|
|
18
|
+
|
|
19
|
+
const ACTION_DESCRIPTION = [
|
|
20
|
+
"One of: create_price_above, create_price_below, create_price_above_sma,",
|
|
21
|
+
"create_price_below_sma, create_rsi_above, create_rsi_below,",
|
|
22
|
+
"create_volume_spike, create_percent_move_up, create_percent_move_down,",
|
|
23
|
+
"create_sma_cross_above, create_sma_cross_below, set_enabled, list, status, check.",
|
|
24
|
+
"Use create_price_above/create_price_below for price alerts,",
|
|
25
|
+
"create_price_above_sma/create_price_below_sma for SMA crossing alerts,",
|
|
26
|
+
"create_percent_move_up/create_percent_move_down for one-day percent move alerts,",
|
|
27
|
+
"create_sma_cross_above/create_sma_cross_below for fast/slow SMA crossing alerts,",
|
|
28
|
+
"create_rsi_above/create_rsi_below for RSI alerts,",
|
|
29
|
+
"create_volume_spike for volume alerts, set_enabled to enable or disable an alert,",
|
|
30
|
+
"and status to inspect local runner/check history.",
|
|
31
|
+
"When the user asks to create an alert and check it now in the same request,",
|
|
32
|
+
"set check_after_create to true on the create action.",
|
|
33
|
+
].join(" ");
|
|
34
|
+
|
|
35
|
+
const params = Type.Object({
|
|
36
|
+
action: Type.Union(
|
|
37
|
+
[
|
|
38
|
+
Type.Literal("create_price_above"),
|
|
39
|
+
Type.Literal("create_price_below"),
|
|
40
|
+
Type.Literal("create_price_above_sma"),
|
|
41
|
+
Type.Literal("create_price_below_sma"),
|
|
42
|
+
Type.Literal("create_rsi_above"),
|
|
43
|
+
Type.Literal("create_rsi_below"),
|
|
44
|
+
Type.Literal("create_volume_spike"),
|
|
45
|
+
Type.Literal("create_percent_move_up"),
|
|
46
|
+
Type.Literal("create_percent_move_down"),
|
|
47
|
+
Type.Literal("create_sma_cross_above"),
|
|
48
|
+
Type.Literal("create_sma_cross_below"),
|
|
49
|
+
Type.Literal("set_enabled"),
|
|
50
|
+
Type.Literal("list"),
|
|
51
|
+
Type.Literal("status"),
|
|
52
|
+
Type.Literal("check"),
|
|
53
|
+
],
|
|
54
|
+
{ description: ACTION_DESCRIPTION },
|
|
55
|
+
),
|
|
56
|
+
symbol: Type.Optional(Type.String({ description: "Ticker symbol for create actions" })),
|
|
57
|
+
threshold: Type.Optional(
|
|
58
|
+
Type.Number({ description: "Price or indicator threshold for create actions" }),
|
|
59
|
+
),
|
|
60
|
+
period: Type.Optional(
|
|
61
|
+
Type.Integer({
|
|
62
|
+
minimum: 1,
|
|
63
|
+
description:
|
|
64
|
+
"Indicator lookback period for SMA/RSI alerts. Max: 200 for price-SMA, 100 for RSI/volume alerts",
|
|
65
|
+
}),
|
|
66
|
+
),
|
|
67
|
+
fast_period: Type.Optional(
|
|
68
|
+
Type.Integer({
|
|
69
|
+
minimum: 1,
|
|
70
|
+
description: "Fast SMA lookback period for SMA-cross alerts. Default: 50",
|
|
71
|
+
}),
|
|
72
|
+
),
|
|
73
|
+
slow_period: Type.Optional(
|
|
74
|
+
Type.Integer({
|
|
75
|
+
minimum: 1,
|
|
76
|
+
maximum: 400,
|
|
77
|
+
description: "Slow SMA lookback period for SMA-cross alerts. Default: 200, max: 400",
|
|
78
|
+
}),
|
|
79
|
+
),
|
|
80
|
+
cooldown_seconds: Type.Optional(
|
|
81
|
+
Type.Integer({
|
|
82
|
+
minimum: 0,
|
|
83
|
+
description: "Cooldown between repeated trigger events. Default: 3600",
|
|
84
|
+
}),
|
|
85
|
+
),
|
|
86
|
+
id: Type.Optional(Type.Number({ description: "Alert rule id for update actions" })),
|
|
87
|
+
enabled: Type.Optional(Type.Boolean({ description: "Whether an alert rule is enabled" })),
|
|
88
|
+
check_after_create: Type.Optional(
|
|
89
|
+
Type.Boolean({
|
|
90
|
+
description:
|
|
91
|
+
"Set true only when the user asks to check/run alerts immediately after creating this alert in the same request.",
|
|
92
|
+
}),
|
|
93
|
+
),
|
|
94
|
+
});
|
|
95
|
+
|
|
96
|
+
export const alertsTool: AgentTool<typeof params> = {
|
|
97
|
+
name: "manage_alerts",
|
|
98
|
+
label: "Alerts",
|
|
99
|
+
description:
|
|
100
|
+
"Create, pause/resume, list, check, and inspect status for durable local alerts. Actions include create_price_above, create_price_below, create_price_above_sma, create_price_below_sma, create_rsi_above, create_rsi_below, create_volume_spike, create_percent_move_up, create_percent_move_down, create_sma_cross_above, create_sma_cross_below, set_enabled, list, status, and check. Local background monitoring runs only while an OpenCandle writer/monitor process is active; manual checks are always available. If the user asks to create an alert and check it now, use the create action with check_after_create=true.",
|
|
101
|
+
parameters: params,
|
|
102
|
+
async execute(_toolCallId, args) {
|
|
103
|
+
const db = initDefaultDatabase();
|
|
104
|
+
const service = new MarketStateService(db);
|
|
105
|
+
|
|
106
|
+
try {
|
|
107
|
+
const cooldownSeconds = validateCooldownSeconds(args.cooldown_seconds);
|
|
108
|
+
|
|
109
|
+
if (args.action === "create_price_above" || args.action === "create_price_below") {
|
|
110
|
+
if (!args.symbol || args.threshold == null) {
|
|
111
|
+
throw new Error("symbol and threshold are required for create alert actions.");
|
|
112
|
+
}
|
|
113
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
114
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
115
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
116
|
+
const isAbove = args.action === "create_price_above";
|
|
117
|
+
const rule = service.createAlertRule({
|
|
118
|
+
scopeType: "instrument",
|
|
119
|
+
instrumentId: instrument.id,
|
|
120
|
+
conditionType: isAbove ? "price_crosses_above" : "price_crosses_below",
|
|
121
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
122
|
+
condition: isAbove
|
|
123
|
+
? priceCrossesAbove(args.threshold)
|
|
124
|
+
: priceCrossesBelow(args.threshold),
|
|
125
|
+
timeframe: "quote",
|
|
126
|
+
cooldownSeconds,
|
|
127
|
+
});
|
|
128
|
+
return await createResultMaybeChecked(
|
|
129
|
+
service,
|
|
130
|
+
rule,
|
|
131
|
+
`Created local alert ${rule.conditionType} for ${instrument.symbol} at $${args.threshold}.`,
|
|
132
|
+
args.check_after_create,
|
|
133
|
+
);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
if (args.action === "create_price_above_sma" || args.action === "create_price_below_sma") {
|
|
137
|
+
if (!args.symbol) {
|
|
138
|
+
throw new Error("symbol is required for SMA alert actions.");
|
|
139
|
+
}
|
|
140
|
+
// Runner evaluates price_crosses_sma against 1y of daily bars (~252).
|
|
141
|
+
const period = validateLookbackPeriod(args.period ?? 50, 200);
|
|
142
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
143
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
144
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
145
|
+
const direction = args.action === "create_price_above_sma" ? "above" : "below";
|
|
146
|
+
const rule = service.createAlertRule({
|
|
147
|
+
scopeType: "instrument",
|
|
148
|
+
instrumentId: instrument.id,
|
|
149
|
+
conditionType: "price_crosses_sma",
|
|
150
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
151
|
+
condition: priceCrossesSma(period, direction),
|
|
152
|
+
timeframe: "1d",
|
|
153
|
+
cooldownSeconds,
|
|
154
|
+
});
|
|
155
|
+
return await createResultMaybeChecked(
|
|
156
|
+
service,
|
|
157
|
+
rule,
|
|
158
|
+
`Created local alert price_crosses_sma for ${instrument.symbol} using SMA(${period}).`,
|
|
159
|
+
args.check_after_create,
|
|
160
|
+
);
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
if (args.action === "create_rsi_above" || args.action === "create_rsi_below") {
|
|
164
|
+
if (!args.symbol || args.threshold == null) {
|
|
165
|
+
throw new Error("symbol and threshold are required for RSI alert actions.");
|
|
166
|
+
}
|
|
167
|
+
// Runner evaluates rsi_threshold against 6mo of daily bars (~126).
|
|
168
|
+
const period = validateLookbackPeriod(args.period ?? 14, 100);
|
|
169
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
170
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
171
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
172
|
+
const direction = args.action === "create_rsi_above" ? "above" : "below";
|
|
173
|
+
const rule = service.createAlertRule({
|
|
174
|
+
scopeType: "instrument",
|
|
175
|
+
instrumentId: instrument.id,
|
|
176
|
+
conditionType: "rsi_threshold",
|
|
177
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
178
|
+
condition: rsiThreshold(period, args.threshold, direction),
|
|
179
|
+
timeframe: "1d",
|
|
180
|
+
cooldownSeconds,
|
|
181
|
+
});
|
|
182
|
+
return await createResultMaybeChecked(
|
|
183
|
+
service,
|
|
184
|
+
rule,
|
|
185
|
+
`Created local alert rsi_threshold for ${instrument.symbol}: RSI(${period}) ${direction} ${args.threshold}.`,
|
|
186
|
+
args.check_after_create,
|
|
187
|
+
);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
if (args.action === "create_volume_spike") {
|
|
191
|
+
if (!args.symbol) {
|
|
192
|
+
throw new Error("symbol is required for volume-spike alert actions.");
|
|
193
|
+
}
|
|
194
|
+
// Runner evaluates volume_spike against 6mo of daily bars (~126).
|
|
195
|
+
const period = validateLookbackPeriod(args.period ?? 20, 100);
|
|
196
|
+
const multiplier = args.threshold ?? 2;
|
|
197
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
198
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
199
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
200
|
+
const rule = service.createAlertRule({
|
|
201
|
+
scopeType: "instrument",
|
|
202
|
+
instrumentId: instrument.id,
|
|
203
|
+
conditionType: "volume_spike",
|
|
204
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
205
|
+
condition: volumeSpike(period, multiplier),
|
|
206
|
+
timeframe: "1d",
|
|
207
|
+
cooldownSeconds,
|
|
208
|
+
});
|
|
209
|
+
return await createResultMaybeChecked(
|
|
210
|
+
service,
|
|
211
|
+
rule,
|
|
212
|
+
`Created local alert volume_spike for ${instrument.symbol}: volume > ${multiplier}x ${period}-bar average.`,
|
|
213
|
+
args.check_after_create,
|
|
214
|
+
);
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
if (args.action === "create_percent_move_up" || args.action === "create_percent_move_down") {
|
|
218
|
+
if (!args.symbol || args.threshold == null) {
|
|
219
|
+
throw new Error("symbol and threshold are required for percent-move alert actions.");
|
|
220
|
+
}
|
|
221
|
+
if (args.threshold <= 0) {
|
|
222
|
+
throw new Error("threshold must be greater than 0 for percent-move alert actions.");
|
|
223
|
+
}
|
|
224
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
225
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
226
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
227
|
+
const direction = args.action === "create_percent_move_up" ? "up" : "down";
|
|
228
|
+
const rule = service.createAlertRule({
|
|
229
|
+
scopeType: "instrument",
|
|
230
|
+
instrumentId: instrument.id,
|
|
231
|
+
conditionType: "percent_move",
|
|
232
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
233
|
+
condition: percentMove(direction, args.threshold),
|
|
234
|
+
timeframe: "1d",
|
|
235
|
+
cooldownSeconds,
|
|
236
|
+
});
|
|
237
|
+
return await createResultMaybeChecked(
|
|
238
|
+
service,
|
|
239
|
+
rule,
|
|
240
|
+
`Created local alert percent_move for ${instrument.symbol}: ${direction} ${args.threshold}%.`,
|
|
241
|
+
args.check_after_create,
|
|
242
|
+
);
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
if (args.action === "create_sma_cross_above" || args.action === "create_sma_cross_below") {
|
|
246
|
+
if (!args.symbol) {
|
|
247
|
+
throw new Error("symbol is required for SMA-cross alert actions.");
|
|
248
|
+
}
|
|
249
|
+
const fastPeriod = args.fast_period ?? 50;
|
|
250
|
+
const slowPeriod = args.slow_period ?? 200;
|
|
251
|
+
if (!Number.isInteger(fastPeriod) || !Number.isInteger(slowPeriod)) {
|
|
252
|
+
throw new Error(
|
|
253
|
+
"fast_period and slow_period must be whole-number lookback periods for SMA-cross alert actions.",
|
|
254
|
+
);
|
|
255
|
+
}
|
|
256
|
+
if (fastPeriod <= 0 || slowPeriod <= 0) {
|
|
257
|
+
throw new Error(
|
|
258
|
+
"fast_period and slow_period must be greater than 0 for SMA-cross alert actions.",
|
|
259
|
+
);
|
|
260
|
+
}
|
|
261
|
+
if (fastPeriod >= slowPeriod) {
|
|
262
|
+
throw new Error("fast_period must be less than slow_period for SMA-cross alert actions.");
|
|
263
|
+
}
|
|
264
|
+
// Runner evaluates sma_cross against 2y of daily bars (~504).
|
|
265
|
+
if (slowPeriod > 400) {
|
|
266
|
+
throw new Error(
|
|
267
|
+
"slow_period must be at most 400 so alert checks can evaluate it against the runner's 2y daily history window.",
|
|
268
|
+
);
|
|
269
|
+
}
|
|
270
|
+
const resolution = await resolveInstrumentForMutation(args.symbol);
|
|
271
|
+
if (resolution.status === "needs_selection") return candidateResult(resolution, "alert");
|
|
272
|
+
const instrument = service.upsertInstrumentRecord(resolution.instrument);
|
|
273
|
+
const direction = args.action === "create_sma_cross_above" ? "above" : "below";
|
|
274
|
+
const rule = service.createAlertRule({
|
|
275
|
+
scopeType: "instrument",
|
|
276
|
+
instrumentId: instrument.id,
|
|
277
|
+
conditionType: "sma_cross",
|
|
278
|
+
conditionVersion: ALERT_CONDITION_VERSION,
|
|
279
|
+
condition: smaCross(fastPeriod, slowPeriod, direction),
|
|
280
|
+
timeframe: "1d",
|
|
281
|
+
cooldownSeconds,
|
|
282
|
+
});
|
|
283
|
+
return await createResultMaybeChecked(
|
|
284
|
+
service,
|
|
285
|
+
rule,
|
|
286
|
+
`Created local alert sma_cross for ${instrument.symbol}: SMA(${fastPeriod}) crosses ${direction} SMA(${slowPeriod}).`,
|
|
287
|
+
args.check_after_create,
|
|
288
|
+
);
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
if (args.action === "list") {
|
|
292
|
+
const rules = service.listAlertRules();
|
|
293
|
+
if (rules.length === 0) {
|
|
294
|
+
return { content: [{ type: "text", text: "No alert rules created yet." }], details: [] };
|
|
295
|
+
}
|
|
296
|
+
const lines = ["**Alerts** — local runner eligible; manual checks available", ""];
|
|
297
|
+
for (const rule of rules) {
|
|
298
|
+
const instrument =
|
|
299
|
+
rule.instrumentId == null ? null : service.getInstrument(rule.instrumentId);
|
|
300
|
+
lines.push(
|
|
301
|
+
` #${rule.id} ${instrument?.symbol ?? "watchlist"} ${rule.conditionType} (${rule.enabled ? "enabled" : "disabled"})`,
|
|
302
|
+
);
|
|
303
|
+
}
|
|
304
|
+
return { content: [{ type: "text", text: lines.join("\n") }], details: rules };
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
if (args.action === "status") {
|
|
308
|
+
const runnerLease = service.getAutomationRunnerLease();
|
|
309
|
+
const recentCheckRuns = service.listAlertCheckRuns().slice(0, 10);
|
|
310
|
+
const stateLine = runnerLease
|
|
311
|
+
? `Running locally — ${runnerLease.ownerKind} ${runnerLease.ownerId}; heartbeat ${runnerLease.heartbeatAt}; expires ${runnerLease.expiresAt}.`
|
|
312
|
+
: "Manual only — no active local runner lease. Keep the GUI/TUI writer or `opencandle monitor` open for background checks.";
|
|
313
|
+
const lines = ["**Alert Automation Status**", "", stateLine];
|
|
314
|
+
if (recentCheckRuns.length === 0) {
|
|
315
|
+
lines.push("", "No alert check runs yet.");
|
|
316
|
+
} else {
|
|
317
|
+
lines.push("", "Recent checks:");
|
|
318
|
+
for (const run of recentCheckRuns) {
|
|
319
|
+
lines.push(
|
|
320
|
+
` #${run.id} ${run.triggerType} ${run.status} checked=${run.checkedCount} triggered=${run.triggeredCount} unavailable=${run.unavailableCount}`,
|
|
321
|
+
);
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
return {
|
|
325
|
+
content: [{ type: "text", text: lines.join("\n") }],
|
|
326
|
+
details: { runnerLease, recentCheckRuns },
|
|
327
|
+
};
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
if (args.action === "set_enabled") {
|
|
331
|
+
if (args.id == null || args.enabled == null) {
|
|
332
|
+
throw new Error("id and enabled are required for set_enabled.");
|
|
333
|
+
}
|
|
334
|
+
const rule = service.setAlertRuleEnabled(args.id, args.enabled);
|
|
335
|
+
if (rule == null) {
|
|
336
|
+
return {
|
|
337
|
+
content: [
|
|
338
|
+
{
|
|
339
|
+
type: "text",
|
|
340
|
+
text: `Alert #${args.id} not found. Use the list action to see alert ids.`,
|
|
341
|
+
},
|
|
342
|
+
],
|
|
343
|
+
details: null,
|
|
344
|
+
};
|
|
345
|
+
}
|
|
346
|
+
return {
|
|
347
|
+
content: [
|
|
348
|
+
{
|
|
349
|
+
type: "text",
|
|
350
|
+
text: `${rule.enabled ? "Enabled" : "Disabled"} alert #${rule.id}.`,
|
|
351
|
+
},
|
|
352
|
+
],
|
|
353
|
+
details: rule,
|
|
354
|
+
};
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
const result = await checkAlerts(service);
|
|
358
|
+
return {
|
|
359
|
+
content: [{ type: "text", text: result.lines.join("\n") }],
|
|
360
|
+
details: result,
|
|
361
|
+
};
|
|
362
|
+
} finally {
|
|
363
|
+
db.close();
|
|
364
|
+
}
|
|
365
|
+
},
|
|
366
|
+
};
|
|
367
|
+
|
|
368
|
+
function validateLookbackPeriod(period: number, maxPeriod: number): number {
|
|
369
|
+
if (!Number.isInteger(period) || period <= 0) {
|
|
370
|
+
throw new Error(
|
|
371
|
+
"period must be a whole-number lookback period greater than 0 for indicator alert actions.",
|
|
372
|
+
);
|
|
373
|
+
}
|
|
374
|
+
if (period > maxPeriod) {
|
|
375
|
+
throw new Error(
|
|
376
|
+
`period must be at most ${maxPeriod} so alert checks can evaluate it against the runner's daily history window.`,
|
|
377
|
+
);
|
|
378
|
+
}
|
|
379
|
+
return period;
|
|
380
|
+
}
|
|
381
|
+
|
|
382
|
+
function validateCooldownSeconds(cooldownSeconds: number | undefined): number {
|
|
383
|
+
const resolved = cooldownSeconds ?? 3600;
|
|
384
|
+
if (!Number.isInteger(resolved) || resolved < 0) {
|
|
385
|
+
throw new Error("cooldown_seconds must be a whole number greater than or equal to 0.");
|
|
386
|
+
}
|
|
387
|
+
return resolved;
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
async function checkAlerts(service: MarketStateService): Promise<{
|
|
391
|
+
checked: number;
|
|
392
|
+
triggered: number;
|
|
393
|
+
lines: string[];
|
|
394
|
+
}> {
|
|
395
|
+
const enabled = service
|
|
396
|
+
.listAlertRules()
|
|
397
|
+
.filter((rule) => rule.enabled && rule.status === "active");
|
|
398
|
+
if (enabled.length === 0)
|
|
399
|
+
return { checked: 0, triggered: 0, lines: ["No enabled alert rules to check."] };
|
|
400
|
+
|
|
401
|
+
const result = await runAlertChecks(service, {
|
|
402
|
+
ownerId: "manual-tool",
|
|
403
|
+
triggerType: "manual",
|
|
404
|
+
providers: defaultAlertRunnerProviders,
|
|
405
|
+
});
|
|
406
|
+
await deliverPendingNotifications(service);
|
|
407
|
+
return {
|
|
408
|
+
checked: result.checked,
|
|
409
|
+
triggered: result.triggered,
|
|
410
|
+
lines: [
|
|
411
|
+
`**Manual Alert Check** — ${enabled.length} rule(s)`,
|
|
412
|
+
"",
|
|
413
|
+
...result.lines.map((line) => ` ${line}`),
|
|
414
|
+
],
|
|
415
|
+
};
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
function candidateResult(
|
|
419
|
+
resolution: Extract<
|
|
420
|
+
Awaited<ReturnType<typeof resolveInstrumentForMutation>>,
|
|
421
|
+
{ status: "needs_selection" }
|
|
422
|
+
>,
|
|
423
|
+
target: string,
|
|
424
|
+
) {
|
|
425
|
+
return {
|
|
426
|
+
content: [
|
|
427
|
+
{
|
|
428
|
+
type: "text" as const,
|
|
429
|
+
text: `Could not verify ${resolution.query}. Choose one of the returned candidates before creating the ${target}.`,
|
|
430
|
+
},
|
|
431
|
+
],
|
|
432
|
+
details: resolution,
|
|
433
|
+
};
|
|
434
|
+
}
|
|
435
|
+
|
|
436
|
+
async function createResultMaybeChecked(
|
|
437
|
+
service: MarketStateService,
|
|
438
|
+
rule: AlertRuleRecord,
|
|
439
|
+
createdText: string,
|
|
440
|
+
checkAfterCreate?: boolean,
|
|
441
|
+
) {
|
|
442
|
+
if (!checkAfterCreate) {
|
|
443
|
+
return {
|
|
444
|
+
content: [{ type: "text" as const, text: createdText }],
|
|
445
|
+
details: rule,
|
|
446
|
+
};
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
const check = await checkAlerts(service);
|
|
450
|
+
return {
|
|
451
|
+
content: [{ type: "text" as const, text: `${createdText}\n\n${check.lines.join("\n")}` }],
|
|
452
|
+
details: {
|
|
453
|
+
created: rule,
|
|
454
|
+
check,
|
|
455
|
+
},
|
|
456
|
+
};
|
|
457
|
+
}
|