Skip to main content

Probabilistic Sharpe ratio and related statistics.

Project description

jsharpe

PyPI version License: MIT Coverage Downloads CodeFactor

A Python library for rigorous Sharpe ratio analysis and statistical testing.

Overview

jsharpe provides comprehensive tools for evaluating trading strategies through the lens of statistical significance. Based on the research of Marcos Lopez de Prado, this library goes beyond simple Sharpe ratio calculations to answer the critical question: Is this strategy's performance statistically significant, or could it be due to chance?

Key Features

  • Probabilistic Sharpe Ratio (PSR) - Transform Sharpe ratios into probabilities that account for estimation uncertainty
  • Non-Gaussian Returns - Correct for skewness and excess kurtosis in return distributions
  • Autocorrelation Adjustment - Handle serial correlation in returns
  • Multiple Testing Corrections - Control False Discovery Rate (FDR) and Family-Wise Error Rate (FWER) when testing multiple strategies
  • Minimum Track Record Length - Determine how long you need to observe a strategy for statistical significance
  • Portfolio Optimization - Minimum variance portfolio weights for correlated assets

Installation

Install jsharpe from PyPI:

pip install jsharpe

Quick Start

Basic Probabilistic Sharpe Ratio

from jsharpe import probabilistic_sharpe_ratio

# Observed Sharpe ratio: 0.456 (e.g., 3.6% return / 7.9% volatility)
sr = 0.036 / 0.079

# Compute PSR with 24 monthly observations
# Testing against SR0=0 (no skill)
psr = probabilistic_sharpe_ratio(SR=sr, SR0=0, T=24)
print(f"PSR: {psr:.3f}")  # Output: PSR: 0.987

The PSR of 0.987 means there's a 98.7% probability that the true Sharpe ratio exceeds zero.

Accounting for Non-Gaussian Returns

Real returns often exhibit negative skewness and excess kurtosis (fat tails):

from jsharpe import probabilistic_sharpe_ratio

sr = 0.036 / 0.079

# Include skewness and kurtosis estimates
psr = probabilistic_sharpe_ratio(
    SR=sr, 
    SR0=0, 
    T=24, 
    gamma3=-2.448,  # Negative skewness
    gamma4=10.164   # Excess kurtosis
)
print(f"PSR (adjusted): {psr:.3f}")  # Output: PSR (adjusted): 0.987

Minimum Track Record Length

How long must you observe a strategy to claim it's significantly better than a benchmark?

from jsharpe import minimum_track_record_length

# Strategy with SR=0.5, testing against SR0=0 at 95% confidence
months_needed = minimum_track_record_length(SR=0.5, SR0=0, alpha=0.05)
print(f"Months needed: {months_needed:.1f}")

Testing Multiple Strategies

When testing many strategies, control the False Discovery Rate:

from jsharpe import control_for_FDR

# Test 10 strategies, controlling FDR at 25%
alpha, beta, SR_critical, q_hat = control_for_FDR(
    q=0.25,           # Target FDR
    SR0=0,            # Null hypothesis
    SR1=0.5,          # Alternative hypothesis
    p_H1=0.05,        # Prior prob of true signal
    T=24              # Observations per strategy
)

print(f"Critical SR threshold: {SR_critical:.3f}")
print(f"Only accept strategies with SR > {SR_critical:.3f}")

Variance of Sharpe Ratio Estimates

from jsharpe import sharpe_ratio_variance
import math

# Variance under Gaussian assumptions
var_gaussian = sharpe_ratio_variance(SR=0.5, T=24)
print(f"Std error (Gaussian): {math.sqrt(var_gaussian):.3f}")

# Variance with fat tails (higher kurtosis)
var_fat_tails = sharpe_ratio_variance(SR=0.5, T=24, gamma4=6.0)
print(f"Std error (fat tails): {math.sqrt(var_fat_tails):.3f}")
PSR: 0.987
PSR (adjusted): 0.987
Months needed: 10.8
Critical SR threshold: 0.479
Only accept strategies with SR > 0.479
Std error (Gaussian): 0.217
Std error (fat tails): 0.234

Core Functions

  • probabilistic_sharpe_ratio() - Compute PSR with various adjustments
  • sharpe_ratio_variance() - Variance of SR estimator under non-Gaussian returns
  • minimum_track_record_length() - Min observations for significance
  • critical_sharpe_ratio() - Threshold for hypothesis testing
  • sharpe_ratio_power() - Statistical power of SR test
  • control_for_FDR() - False Discovery Rate control for multiple testing
  • adjusted_p_values_bonferroni() - Bonferroni correction
  • adjusted_p_values_holm() - Holm's step-down procedure
  • adjusted_p_values_sidak() - Šidák correction
  • minimum_variance_weights_for_correlated_assets() - Portfolio optimization

Documentation

References

This library implements methods from:

  • Bailey, D. H., & López de Prado, M. (2012). "The Sharpe Ratio Efficient Frontier." Journal of Risk, 15(2), 3-44.
  • Bailey, D. H., & López de Prado, M. (2014). "The Deflated Sharpe Ratio: Correcting for Selection Bias, Backtest Overfitting and Non-Normality." Journal of Portfolio Management, 40(5), 94-107.

For Developers

Setup Development Environment

# Clone the repository
git clone https://github.com/tschm/jsharpe.git
cd jsharpe

# Install dependencies and setup environment
make install

This installs uv, creates a virtual environment, and installs all dependencies.

Development Workflow

# Run tests
make tests

# Format code
make fmt

# Start interactive notebooks
make marimo

Project Structure

jsharpe/
├── src/jsharpe/       # Main package source code
│   ├── __init__.py    # Public API exports
│   └── sharpe.py      # Core implementations
├── tests/             # Test suite
│   └── test_sharpe.py # Unit tests
├── book/              # Documentation and interactive notebooks
│   └── marimo/        # Marimo notebooks for exploration
└── pyproject.toml     # Project metadata and dependencies

Contributing

We welcome contributions! Please:

  1. Fork the repository
  2. Create a feature branch (git checkout -b feature/amazing-feature)
  3. Make your changes and add tests
  4. Run make tests and make fmt
  5. Commit your changes (git commit -m 'Add some amazing feature')
  6. Push to the branch (git push origin feature/amazing-feature)
  7. Open a Pull Request

See CONTRIBUTING.md for more details.

Running Tests

# Run all tests with coverage
make tests

# Run specific test file
pytest tests/test_sharpe.py -v

License

MIT License - see LICENSE file for details.

Citation

If you use jsharpe in your research, please cite:

@software{jsharpe,
  author = {Thomas Schmelzer},
  title = {jsharpe: Probabilistic Sharpe Ratio and Statistical Testing},
  year = {2024},
  url = {https://github.com/tschm/jsharpe}
}

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

jsharpe-0.6.1.tar.gz (7.2 MB view details)

Uploaded Source

Built Distribution

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

jsharpe-0.6.1-py3-none-any.whl (17.5 kB view details)

Uploaded Python 3

File details

Details for the file jsharpe-0.6.1.tar.gz.

File metadata

  • Download URL: jsharpe-0.6.1.tar.gz
  • Upload date:
  • Size: 7.2 MB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for jsharpe-0.6.1.tar.gz
Algorithm Hash digest
SHA256 b0acc4441ffe2d76a3e5465e7d086ae7a57292ccab95fea4f2ac49e3dd7ff848
MD5 cb84b2de3091904329c87e3e3c336deb
BLAKE2b-256 b13a2385c33c7f578a80548473fa7a0179f8784bd5450d425953ffc654e8f2e0

See more details on using hashes here.

Provenance

The following attestation bundles were made for jsharpe-0.6.1.tar.gz:

Publisher: rhiza_release.yml on tschm/jsharpe

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

File details

Details for the file jsharpe-0.6.1-py3-none-any.whl.

File metadata

  • Download URL: jsharpe-0.6.1-py3-none-any.whl
  • Upload date:
  • Size: 17.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for jsharpe-0.6.1-py3-none-any.whl
Algorithm Hash digest
SHA256 8f2f67ffba2ed35e86782eae2271c889ff487928c234cbe4d04c52e9409657df
MD5 9fb0457a9048692e336839411ccc75a8
BLAKE2b-256 a942736e3b5a0a7d364a07d03e186b503f908fc038eb2db45c4da4618872a53d

See more details on using hashes here.

Provenance

The following attestation bundles were made for jsharpe-0.6.1-py3-none-any.whl:

Publisher: rhiza_release.yml on tschm/jsharpe

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