Разработка системы бэктестинга портфельных стратегий

Проектируем и разрабатываем блокчейн-решения полного цикла: от архитектуры смарт-контрактов до запуска DeFi-протоколов, NFT-маркетплейсов и криптобирж. Аудит безопасности, токеномика, интеграция с существующей инфраструктурой.
Показано 1 из 1 услугВсе 1306 услуг
Разработка системы бэктестинга портфельных стратегий
Сложная
~1-2 недели
Часто задаваемые вопросы
Направления блокчейн-разработки
Этапы блокчейн-разработки
Последние работы
  • image_website-b2b-advance_0.png
    Разработка сайта компании B2B ADVANCE
    1221
  • image_web-applications_feedme_466_0.webp
    Разработка веб-приложения для компании FEEDME
    1163
  • image_websites_belfingroup_462_0.webp
    Разработка веб-сайта для компании БЕЛФИНГРУПП
    855
  • image_ecommerce_furnoro_435_0.webp
    Разработка интернет магазина для компании FURNORO
    1062
  • image_logo-advance_0.png
    Разработка логотипа компании B2B Advance
    561
  • image_crm_enviok_479_0.webp
    Разработка веб-приложения для компании Enviok
    828

Разработка системы бэктестинга портфельных стратегий

Портфельный бэктестинг отличается от одноинструментального: вместо анализа одной пары отслеживается распределение капитала между множеством активов, корреляции между позициями и совокупный риск портфеля.

Ключевые отличия от single-asset бэктеста

Capital allocation — как распределяется капитал между сигналами? Equal weight, risk-parity, momentum-weighted? Решение принимается динамически каждый ребалансировочный период.

Correlated positions — BTC и ETH движутся вместе. Держать обе позиции — не диверсификация. Нужно учитывать корреляцию при расчёте реального риска.

Transaction costs — при ребалансировке портфеля из 20 активов каждые 2 недели комиссии существенно съедают прибыль.

Slippage при масштабировании — крупный портфель не может войти в малоликвидный альткоин по рыночной цене без значительного impact.

Структура данных портфеля

from dataclasses import dataclass, field
from typing import Optional
import numpy as np
import pandas as pd

@dataclass
class PortfolioPosition:
    symbol: str
    quantity: float
    avg_entry_price: float
    entry_time: int
    unrealized_pnl: float = 0.0
    realized_pnl: float = 0.0

@dataclass
class PortfolioState:
    cash: float
    positions: dict[str, PortfolioPosition] = field(default_factory=dict)
    
    @property
    def total_value(self) -> float:
        return self.cash + sum(
            p.quantity * p.avg_entry_price * (1 + p.unrealized_pnl / (p.quantity * p.avg_entry_price))
            for p in self.positions.values()
        )

    def get_weights(self, current_prices: dict[str, float]) -> dict[str, float]:
        total = self.cash + sum(
            p.quantity * current_prices.get(p.symbol, p.avg_entry_price)
            for p in self.positions.values()
        )
        weights = {}
        for symbol, pos in self.positions.items():
            price = current_prices.get(symbol, pos.avg_entry_price)
            weights[symbol] = (pos.quantity * price) / total
        weights['cash'] = self.cash / total
        return weights

Capital Allocation стратегии

class AllocationEngine:
    def equal_weight(self, signals: list[str], capital: float) -> dict[str, float]:
        """Равное распределение капитала"""
        if not signals:
            return {}
        weight = capital / len(signals)
        return {symbol: weight for symbol in signals}

    def risk_parity(
        self,
        signals: list[str],
        capital: float,
        volatilities: dict[str, float],
    ) -> dict[str, float]:
        """Распределение обратно пропорционально волатильности"""
        if not signals:
            return {}
        
        inv_vols = {s: 1.0 / volatilities.get(s, 0.01) for s in signals}
        total_inv_vol = sum(inv_vols.values())
        
        return {
            symbol: capital * inv_vol / total_inv_vol
            for symbol, inv_vol in inv_vols.items()
        }

    def momentum_weighted(
        self,
        signals: list[str],
        capital: float,
        returns_12m: dict[str, float],
    ) -> dict[str, float]:
        """Больший вес — активам с лучшей 12-месячной доходностью"""
        positive_returns = {s: r for s, r in returns_12m.items() if s in signals and r > 0}
        if not positive_returns:
            return self.equal_weight(signals, capital)
        
        total_return = sum(positive_returns.values())
        return {
            symbol: capital * ret / total_return
            for symbol, ret in positive_returns.items()
        }

Ребалансировка

class RebalancingEngine:
    def __init__(self, commission_pct: float = 0.001, min_trade_usd: float = 10.0):
        self.commission_pct = commission_pct
        self.min_trade_usd = min_trade_usd

    def calculate_rebalancing_trades(
        self,
        current_state: PortfolioState,
        target_weights: dict[str, float],
        current_prices: dict[str, float],
    ) -> list[dict]:
        total_value = sum(
            pos.quantity * current_prices[sym]
            for sym, pos in current_state.positions.items()
        ) + current_state.cash

        trades = []
        for symbol, target_weight in target_weights.items():
            target_value = total_value * target_weight
            current_value = (
                current_state.positions[symbol].quantity * current_prices[symbol]
                if symbol in current_state.positions else 0.0
            )

            delta = target_value - current_value
            
            if abs(delta) < self.min_trade_usd:
                continue  # Слишком маленькая сделка, пропускаем

            trades.append({
                'symbol': symbol,
                'side': 'BUY' if delta > 0 else 'SELL',
                'usd_amount': abs(delta),
                'quantity': abs(delta) / current_prices[symbol],
                'price': current_prices[symbol],
                'commission': abs(delta) * self.commission_pct,
            })

        return trades

Портфельные метрики

def calculate_portfolio_metrics(equity_curve: pd.Series, trades_df: pd.DataFrame) -> dict:
    returns = equity_curve.pct_change().dropna()
    
    # Correlation-adjusted risk
    # Если у нас есть позиции в BTC и ETH одновременно,
    # реальный риск выше чем сумма индивидуальных рисков

    annualized_return = returns.mean() * 252
    annualized_vol = returns.std() * np.sqrt(252)
    sharpe = annualized_return / annualized_vol if annualized_vol > 0 else 0

    # Portfolio-specific: turnover
    total_traded_value = trades_df['usd_amount'].sum()
    avg_portfolio_value = equity_curve.mean()
    annual_turnover = (total_traded_value / avg_portfolio_value) * (252 / len(equity_curve))

    # Transaction cost drag
    total_commissions = trades_df['commission'].sum()
    commission_drag = total_commissions / equity_curve.iloc[0]

    return {
        'annualized_return': annualized_return,
        'annualized_volatility': annualized_vol,
        'sharpe_ratio': sharpe,
        'max_drawdown': calculate_max_drawdown(equity_curve),
        'annual_turnover': annual_turnover,
        'total_commission': total_commissions,
        'commission_drag_pct': commission_drag * 100,
        'avg_positions': trades_df.groupby('timestamp')['symbol'].count().mean(),
    }

Survivorship bias в портфельных тестах

Если тестируете стратегию на портфеле из текущего состава индекса (например, топ-20 крипто по капитализации), вы тестируете на выживших: активы, которые были в топ-20 год назад и вылетели оттуда, не включены. Это систематически завышает результаты.

Правильный подход: использовать point-in-time universe — какие активы были в universe на каждую конкретную дату, не какие в нём сейчас.