Skip to main content

Volatility trading research

Project description

Volatility Trading on Equity Options

CI Pages Python Ruff Pyright pre-commit License: MIT

This project develops and evaluates daily options-volatility strategies on index and single-stock underlyings. Research spans the full pipeline: data engineering and quality checks, implied-volatility surface modelling, volatility forecasting, and strategy backtesting. Backtests use realistic execution assumptions (bid/ask, slippage, commissions, position sizing, and risk limits) and are documented with reproducible notebooks and published reports.

Notebook reports (GitHub Pages): https://anthonymakarewicz.github.io/volatility-trading/

Quickstart

  1. Clone the repository:
git clone https://github.com/anthonymakarewicz/volatility-trading.git
cd volatility_trading
  1. Install uv and create a virtual environment (Python 3.12+):
brew install uv  # or: pipx install uv
uv venv --python 3.12 .venv
source .venv/bin/activate
  1. Install dependencies:

Primary contributor setup (editable package + dev tooling):

uv pip install -e ".[dev]"

Secondary options:

  • Runtime-only install (users running package code without dev tools):
uv pip install .
  • Editable runtime-only install (local source edits, no dev tools):
uv pip install -e .

pip remains supported as a fallback if you do not want to use uv:

python -m venv .venv
source .venv/bin/activate
python -m pip install -U pip
pip install -e ".[dev]"
  1. Optional: set credentials for ORATS data access:
cp .env.example .env

Then set ORATS_API_KEY, ORATS_FTP_USER, and ORATS_FTP_PASS in .env if you plan to run the ORATS download/extract pipeline. You can skip this if you are using already-prepared data or a different options data source.

ORATS ETL Pipeline (End-to-End)

Pipeline steps:

  • API download
  • API extract
  • FTP download
  • FTP extract
  • Build options chain
  • Build daily features
  • QC options chain
  • QC daily features

Use --dry-run to validate config, paths, and credentials before running writes/network.

orats-api-download --config config/orats/api_download.yml --dry-run

For the full command sequence, see Data pipeline.

Current Data Support Status

  • Options ETL (options chain + daily features) is currently supported through the ORATS pipeline.
  • External feed sync (fred-sync, yfinance-sync) currently covers rates/market time series, not a full generic options ETL path.
  • The options backtesting runtime expects the current project options-chain schema (for example quotes/Greeks fields used by entry, sizing, and lifecycle).
  • You can run backtests with non-ORATS data if it is pre-normalized to the expected schema.

Stability

  • Current release line is 0.5.x (alpha / pre-1.0).
  • Public APIs, data contracts, and configuration surfaces may evolve between minor versions.
  • For reproducible research, pin exact package versions and review CHANGELOG.md before upgrading.
  • Public vs internal boundaries are defined in API Scope.

Data Contract / Supported Inputs

Source Input expected by backtester Support status Adapter path
ORATS ETL output Canonical long-format options chain First-class CanonicalOptionsChainAdapter (or mode canonical)
OptionsDX ETL output Cleaned long-format panel (reshape='long') Supported OptionsDxOptionsChainAdapter
Custom/vendor dataset Long-format panel mapped to canonical fields Supported with mapping ColumnMapOptionsChainAdapter

Notes:

Quick VRP Backtest Example

Assume you already prepared:

  • options: long-format pandas options panel indexed by trade_date
from volatility_trading.backtesting import (
    AccountConfig,
    Backtester,
    BacktestRunConfig,
    BrokerConfig,
    MarginConfig,
    OptionsBacktestDataBundle,
    OptionsMarketData,
    print_performance_report,
    to_daily_mtm,
)
from volatility_trading.options import RegTMarginModel
from volatility_trading.signals import ShortOnlySignal
from volatility_trading.strategies import VRPHarvestingSpec, make_vrp_strategy

data = OptionsBacktestDataBundle(
    options_market=OptionsMarketData(chain=options),
)
strategy = make_vrp_strategy(
    VRPHarvestingSpec(
        signal=ShortOnlySignal(),
        rebalance_period=10,
        risk_budget_pct=0.03,
        margin_budget_pct=0.4,
    )
)
cfg = BacktestRunConfig(
    account=AccountConfig(initial_capital=50_000),
    broker=BrokerConfig(
        margin=MarginConfig(model=RegTMarginModel(broad_index=False))
    ),
)

bt = Backtester(data=data, strategy=strategy, config=cfg)
trades, mtm = bt.run()
daily_mtm = to_daily_mtm(mtm, cfg.account.initial_capital)
print_performance_report(
    trades=trades,
    mtm_daily=daily_mtm,
    risk_free_rate=0.02,
)

Quick Skew Backtest Example

Assume you also prepared:

  • features: daily ORATS features indexed by trade_date, including iv_dlt25_30d and iv_dlt75_30d
from volatility_trading.strategies import (
    SkewMispricingSpec,
    make_skew_mispricing_strategy,
)

data = OptionsBacktestDataBundle(
    options_market=OptionsMarketData(chain=options),
    features=features,
)

strategy = make_skew_mispricing_strategy(
    SkewMispricingSpec(
        risk_budget_pct=0.03,
        margin_budget_pct=0.4,
        max_holding_period=30,
    )
)

bt = Backtester(data=data, strategy=strategy, config=cfg)
trades, mtm = bt.run()

For a full scriptable workflow (data loading + backtest run), see VRP end-to-end example. For the skew counterpart using daily features, see Skew end-to-end example. For config-driven CLI runs, see Backtest Runner Workflows, which points to the shipped config/backtesting/vrp_harvesting.yml and config/backtesting/skew_mispricing.yml templates. For focused backtesting configuration examples (execution, margin, adapters, hedging), see examples/README.md. For hedging model semantics and WW/fixed-band configuration details, see hedging.md. For option execution model behavior and option-cost attribution fields, see option_execution.md. For the research-style workflow and reporting exploration, see VRP notebook.

Preferred imports for common backtesting usage come from volatility_trading.backtesting. The narrower volatility_trading.backtesting.options_engine namespace remains available for advanced engine-specific helpers.

Advanced Execution And Hedging Setup

Backtester intentionally keeps a stable high-level API and does not expose execution-model arguments directly.

If you want explicit transaction-cost models for both options and hedge trades, configure them on BacktestRunConfig.execution. If the strategy uses dynamic delta hedging, also provide hedge_market:

# Reuse the imports from the quick VRP example above, then add:
from volatility_trading.backtesting import (
    BidAskFeeOptionExecutionModel,
    DeltaHedgePolicy,
    ExecutionConfig,
    FixedDeltaBandModel,
    FixedBpsHedgeExecutionModel,
    HedgeMarketData,
    HedgeTriggerPolicy,
)

hedge_mid = options.groupby(level=0)["spot_price"].first().astype(float)
data = OptionsBacktestDataBundle(
    options_market=OptionsMarketData(chain=options),
    hedge_market=HedgeMarketData(mid=hedge_mid),
)
strategy = make_vrp_strategy(
    VRPHarvestingSpec(
        signal=ShortOnlySignal(),
        rebalance_period=10,
        delta_hedge=DeltaHedgePolicy(
            enabled=True,
            target_net_delta=0.0,
            trigger=HedgeTriggerPolicy(
                band_model=FixedDeltaBandModel(half_width_abs=25.0),
                rebalance_every_n_days=5,
                combine_mode="or",
            ),
            min_rebalance_qty=1.0,
        ),
    )
)
cfg = BacktestRunConfig(
    execution=ExecutionConfig(
        option_execution_model=BidAskFeeOptionExecutionModel(
            commission_per_leg=0.0,
        ),
        hedge_execution_model=FixedBpsHedgeExecutionModel(
            fee_bps=1.0,
        ),
    ),
)
bt = Backtester(data=data, strategy=strategy, config=cfg)
trades, mtm = bt.run()

Tests

Run unit tests (default):

pytest -q

Run integration tests:

pytest -q -m integration

See Testing guide for layout and conventions.

Continuous Integration (CI)

GitHub Actions runs:

  • Ruff lint + format checks
  • Pyright type checks
  • Unit tests by default
  • Integration tests on PRs and pushes to main (and manual runs)

See CI workflow.

Developer Workflow

Common commands are available via Makefile:

make lint
make format
make check
make typecheck
make test
make test-unit
make test-integration
make package-check
make sync-nb
make sync-nb-all
make ci

For full setup and tooling details, see the Documentation index. Notebook HTML reports are built in GitHub Actions and published to GitHub Pages.

Docs

See Documentation index for the full docs map. Most-used pages:

Research Highlights

We publish research notebooks and strategy diagnostics as HTML reports on GitHub Pages.

  • RV forecasting: HAR-RV-VIX reaches about 30% OOS R² vs a naive baseline.
  • IV surface modelling: parametric vs non-parametric surface comparison workflows.
  • Skew trading: delta-hedged RR strategy with realistic costs and risk controls.

Explore details here:

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

volatility_trading-0.5.0.tar.gz (233.0 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

volatility_trading-0.5.0-py3-none-any.whl (353.1 kB view details)

Uploaded Python 3

File details

Details for the file volatility_trading-0.5.0.tar.gz.

File metadata

  • Download URL: volatility_trading-0.5.0.tar.gz
  • Upload date:
  • Size: 233.0 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for volatility_trading-0.5.0.tar.gz
Algorithm Hash digest
SHA256 cb70247aaa795e152085b0522e10b69949c17f3ab0fed6049cb06ca7577c6a93
MD5 4ee28adddba92b9b7dec2e1dc0f23c99
BLAKE2b-256 1d300430d3f5ad0e8f1001cb81db816cdb374dd06a7e51f7e8cd8f46cb5775c3

See more details on using hashes here.

Provenance

The following attestation bundles were made for volatility_trading-0.5.0.tar.gz:

Publisher: publish-pypi.yml on anthonymakarewicz/volatility-trading

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file volatility_trading-0.5.0-py3-none-any.whl.

File metadata

File hashes

Hashes for volatility_trading-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 af86d9d5836775efbb0d5aee502a2c2901e224adbbbce6a1c3d32cc7f82737c2
MD5 128558134defa23f7fa7449924cc2965
BLAKE2b-256 7061be6dcaaedc5a218f2cfabce731458f0adc6ecf25a3163b1a5bca3752e2e6

See more details on using hashes here.

Provenance

The following attestation bundles were made for volatility_trading-0.5.0-py3-none-any.whl:

Publisher: publish-pypi.yml on anthonymakarewicz/volatility-trading

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page