From 7d7ecadca4f7416eb252afce750e75e696d54a94 Mon Sep 17 00:00:00 2001 From: TheSiahxyz <164138827+TheSiahxyz@users.noreply.github.com> Date: Wed, 1 Apr 2026 17:26:03 +0900 Subject: feat(strategy): add combined strategy with weighted signal voting --- .../strategies/combined_strategy.py | 88 ++++++++++++++++++++++ .../strategies/config/combined_strategy.yaml | 2 + 2 files changed, 90 insertions(+) create mode 100644 services/strategy-engine/strategies/combined_strategy.py create mode 100644 services/strategy-engine/strategies/config/combined_strategy.yaml (limited to 'services/strategy-engine/strategies') diff --git a/services/strategy-engine/strategies/combined_strategy.py b/services/strategy-engine/strategies/combined_strategy.py new file mode 100644 index 0000000..e99dfdf --- /dev/null +++ b/services/strategy-engine/strategies/combined_strategy.py @@ -0,0 +1,88 @@ +"""Combined strategy that aggregates signals from multiple sub-strategies.""" +from decimal import Decimal +from typing import Optional + +from shared.models import Candle, Signal, OrderSide +from strategies.base import BaseStrategy + + +class CombinedStrategy(BaseStrategy): + """Combines multiple strategies using weighted signal voting. + + Each sub-strategy votes BUY (+weight), SELL (-weight), or HOLD (0). + The combined signal fires when the weighted sum exceeds a threshold. + """ + name: str = "combined" + + def __init__(self) -> None: + self._strategies: list[tuple[BaseStrategy, float]] = [] # (strategy, weight) + self._threshold: float = 0.5 + self._quantity: Decimal = Decimal("0.01") + + @property + def warmup_period(self) -> int: + if not self._strategies: + return 0 + return max(s.warmup_period for s, _ in self._strategies) + + def configure(self, params: dict) -> None: + self._threshold = float(params.get("threshold", 0.5)) + self._quantity = Decimal(str(params.get("quantity", "0.01"))) + if self._threshold <= 0: + raise ValueError(f"Threshold must be positive, got {self._threshold}") + if self._quantity <= 0: + raise ValueError(f"Quantity must be positive, got {self._quantity}") + + def add_strategy(self, strategy: BaseStrategy, weight: float = 1.0) -> None: + """Add a sub-strategy with a weight.""" + if weight <= 0: + raise ValueError(f"Weight must be positive, got {weight}") + self._strategies.append((strategy, weight)) + + def reset(self) -> None: + for strategy, _ in self._strategies: + strategy.reset() + + def on_candle(self, candle: Candle) -> Signal | None: + if not self._strategies: + return None + + total_weight = sum(w for _, w in self._strategies) + if total_weight == 0: + return None + + score = 0.0 + reasons = [] + + for strategy, weight in self._strategies: + signal = strategy.on_candle(candle) + if signal is not None: + if signal.side == OrderSide.BUY: + score += weight + reasons.append(f"{strategy.name}:BUY({weight})") + elif signal.side == OrderSide.SELL: + score -= weight + reasons.append(f"{strategy.name}:SELL({weight})") + + normalized = score / total_weight # Range: -1.0 to 1.0 + + if normalized >= self._threshold: + return Signal( + strategy=self.name, + symbol=candle.symbol, + side=OrderSide.BUY, + price=candle.close, + quantity=self._quantity, + reason=f"Combined score {normalized:.2f} >= {self._threshold} [{', '.join(reasons)}]", + ) + elif normalized <= -self._threshold: + return Signal( + strategy=self.name, + symbol=candle.symbol, + side=OrderSide.SELL, + price=candle.close, + quantity=self._quantity, + reason=f"Combined score {normalized:.2f} <= -{self._threshold} [{', '.join(reasons)}]", + ) + + return None diff --git a/services/strategy-engine/strategies/config/combined_strategy.yaml b/services/strategy-engine/strategies/config/combined_strategy.yaml new file mode 100644 index 0000000..9b5a575 --- /dev/null +++ b/services/strategy-engine/strategies/config/combined_strategy.yaml @@ -0,0 +1,2 @@ +threshold: 0.5 +quantity: "0.01" -- cgit v1.2.3