AdvancedStrategiesPython
Run this module
Vectorised Backtesting Engine¶
A backtest answers one question: if I had traded this rule, what would have happened? This module is a compact, honest backtester for single-asset strategies — it converts a price series and a position series into an equity curve and the performance statistics every strategy report quotes.
It deliberately avoids the two mistakes that make most home-grown backtests lie: look-ahead bias and ignoring trading costs.
Functions¶
| Function | Description |
|---|---|
to_returns(prices) |
Simple period returns aligned to the price array |
run_backtest(prices, target_position, fee_bps, slippage_bps) |
Core engine → equity curve, net/gross returns, costs, turnover |
performance_summary(result, periods_per_year) |
CAGR, vol, Sharpe, Sortino, max drawdown, Calmar, hit rate, turnover |
max_drawdown(equity) |
Largest peak-to-trough decline and where it happened |
sma_crossover_signal(prices, fast, slow) |
Example long/flat signal to drive the engine |
How look-ahead is removed¶
The position you decide on bar t is only held from t to t+1:
If you instead let target[t] earn return[t], you would be trading on
information you did not have yet — the classic way to manufacture a beautiful,
fake equity curve.
How costs are charged¶
Costs are paid on the traded notional, i.e. the change in position:
traded[t] = |held[t] - held[t-1]|
cost[t] = traded[t] * (fee_bps + slippage_bps) / 10000
net[t] = held[t] * return[t] - cost[t]
So a full flip (+1 → -1) costs twice as much as opening a position
(0 → +1), exactly as it should.
Example¶
import numpy as np
from backtest_engine import run_backtest, performance_summary, sma_crossover_signal
prices = ... # your price series
signal = sma_crossover_signal(prices, fast=20, slow=50)
result = run_backtest(prices, signal, fee_bps=1.0, slippage_bps=0.5)
stats = performance_summary(result, periods_per_year=252)
print(stats["sharpe"], stats["max_drawdown"], stats["cagr"])
To plug in your own strategy, just pass any target_position array — values in
{-1, 0, 1} for long/flat/short, or continuous weights for sizing.
Performance metrics, briefly¶
- CAGR — geometric annual growth rate of the equity curve.
- Sharpe — annualised mean return ÷ annualised volatility (excess over a zero risk-free rate here).
- Sortino — like Sharpe but only penalises downside deviation.
- Max drawdown — worst peak-to-trough loss; the number that ends careers.
- Calmar — CAGR ÷ |max drawdown|; return per unit of pain.
- Hit rate — fraction of bars with a positive net return.
Practical notes¶
- A backtest is a hypothesis, not a guarantee. Costs, slippage and survivorship bias all erode live results versus paper.
- Always compare against buy & hold — the
__main__demo does this. If your clever strategy cannot beat holding the asset after costs, it is not clever. - Vectorised backtests assume you can transact at the marked price. For
realistic fills on large size, see
Order Execution SimulatorandFinance - Transaction Cost Analysis. - Beware overfitting: tuning
fast/slowuntil the curve looks great is curve-fitting. Validate out-of-sample (seeQuantitative Methods - Bootstrap).
Continue in Strategies¶
-
This utility does NOT use any external APIs. All trades and portfolio data are managed locally for learning and experimentation.
-
Implementation of the Avellaneda-Stoikov (2008) continuous-time market making model. A dealer posts bid/ask quotes to maximize expected PnL while penalizing inventory accumulation.
-
Mean reversion is the statistical tendency for an asset's price to return to its historical average after deviating from it. While Momentum strategies bet on continuation, Mean Reversion strategies bet on reversal — buying when something is "too cheap" and selling when it is "too expensive" relative to recent history.
-
Momentum trading is a strategy that capitalizes on the continuance of existing trends in the market. The core philosophy is "buy high, sell higher." If an asset's price is rising strongly, momentum traders assume it will continue to rise.
-
This module demonstrates a statistical arbitrage strategy known as Pairs Trading. It identifies two assets that move together and trades the convergence of their spread. When the correlation weakens temporarily, executing trades on both assets allows for capturing profits as they revert to their historical relationship. This quantitative technique relies strictly on mathematical relationships rather than fundamental valuation.
-
Strategies - Statistical Arbitrage
This module demonstrates a basic Statistical Arbitrage strategy, specifically pairs trading.