Files
microdao-daarion/services/market-data-service/app/config.py
Apple 09dee24342 feat: MD pipeline — market-data-service hardening + SenpAI NATS consumer
Producer (market-data-service):
- Backpressure: smart drop policy (heartbeats→quotes→trades preserved)
- Heartbeat monitor: synthetic HeartbeatEvent on provider silence
- Graceful shutdown: WS→bus→storage→DB engine cleanup sequence
- Bybit V5 public WS provider (backup for Binance, no API key needed)
- FailoverManager: health-based provider switching with recovery
- NATS output adapter: md.events.{type}.{symbol} for SenpAI
- /bus-stats endpoint for backpressure monitoring
- Dockerfile + docker-compose.node1.yml integration
- 36 tests (parsing + bus + failover), requirements.lock

Consumer (senpai-md-consumer):
- NATSConsumer: subscribe md.events.>, queue group senpai-md, backpressure
- State store: LatestState + RollingWindow (deque, 60s)
- Feature engine: 11 features (mid, spread, VWAP, return, vol, latency)
- Rule-based signals: long/short on return+volume+spread conditions
- Publisher: rate-limited features + signals + alerts to NATS
- HTTP API: /health, /metrics, /state/latest, /features/latest, /stats
- 10 Prometheus metrics
- Dockerfile + docker-compose.senpai.yml
- 41 tests (parsing + state + features + rate-limit), requirements.lock

CI: ruff + pytest + smoke import for both services
Tests: 77 total passed, lint clean
Co-authored-by: Cursor <cursoragent@cursor.com>
2026-02-09 11:46:15 -08:00

66 lines
3.0 KiB
Python

"""
Configuration via pydantic-settings.
All secrets come from .env; no defaults for sensitive keys.
"""
from __future__ import annotations
from pydantic_settings import BaseSettings, SettingsConfigDict
class Settings(BaseSettings):
model_config = SettingsConfigDict(
env_file=".env",
env_file_encoding="utf-8",
extra="ignore",
)
# ── Binance (no key needed for public WS) ──────────────────────────
binance_ws_url: str = "wss://stream.binance.com:9443/ws"
binance_rest_url: str = "https://api.binance.com"
# ── Bybit (backup crypto — no key needed) ──────────────────────────
bybit_ws_url: str = "wss://stream.bybit.com/v5/public/spot"
# ── Alpaca (paper trading — free tier) ─────────────────────────────
alpaca_key: str = ""
alpaca_secret: str = ""
alpaca_base_url: str = "https://paper-api.alpaca.markets"
alpaca_data_ws_url: str = "wss://stream.data.alpaca.markets/v2/iex"
alpaca_dry_run: bool = True # True = skip real API calls if no keys
# ── Storage ────────────────────────────────────────────────────────
sqlite_url: str = "sqlite+aiosqlite:///market_data.db"
jsonl_path: str = "events.jsonl"
# ── Reliability ────────────────────────────────────────────────────
reconnect_max_retries: int = 20
reconnect_base_delay: float = 1.0 # seconds, exponential backoff
reconnect_max_delay: float = 60.0
heartbeat_timeout: float = 30.0 # no-message timeout → reconnect
# ── Metrics / HTTP ─────────────────────────────────────────────────
http_host: str = "0.0.0.0"
http_port: int = 8891
metrics_enabled: bool = True
# ── NATS output adapter ─────────────────────────────────────────────
nats_url: str = "" # e.g. "nats://localhost:4222"
nats_subject_prefix: str = "md.events" # → md.events.trade.BTCUSDT
nats_enabled: bool = False
# ── Logging ────────────────────────────────────────────────────────
log_level: str = "INFO"
log_sample_rate: int = 100 # PrintConsumer: log 1 out of N events
@property
def alpaca_configured(self) -> bool:
return bool(self.alpaca_key and self.alpaca_secret)
@property
def nats_configured(self) -> bool:
return bool(self.nats_url and self.nats_enabled)
settings = Settings()