Skip to main content

Critical line algorithm for the efficient frontier

Project description

📈 cvxcla - Critical Line Algorithm for Portfolio Optimization

PyPI version License: MIT Downloads Coverage


Quick Links: 📖 Documentation🐛 Report Bug💡 Request Feature💬 Discussions


📋 Overview

cvxcla is a Python package that implements the Critical Line Algorithm (CLA) for portfolio optimization. The CLA efficiently computes the entire efficient frontier for portfolio optimization problems with linear constraints and bounds on the weights.

The Critical Line Algorithm was introduced by Harry Markowitz in The Optimization of Quadratic Functions Subject to Linear Constraints and further described in his book Portfolio Selection.

The algorithm is based on the observation that the efficient frontier is a piecewise linear function when expected return is plotted against expected variance. The CLA computes the turning points (corners) of the efficient frontier, allowing for efficient representation of the entire frontier.

I gave the plenary talk at EQD's Singapore conference.

🧮 Why the Algorithm Works

The Markowitz problem is a quadratic program parametrized by a return target λ:

min  wᵀΣw - λ · μᵀw
s.t. Aw = b,  Gw ≤ h,  lb ≤ w ≤ ub

where Aw = b are linear equalities (the budget, sector/factor neutrality, …) and Gw ≤ h are linear inequalities (group or sector exposure caps; a floor is the negated row). As λ sweeps from ∞ (maximize return) down to 0 (minimize variance), the solution traces the entire efficient frontier. The key insight is that between consecutive events, the optimal weights are a linear function of λ:

w(λ) = α + λ · β

This holds because the KKT optimality conditions are linear in λ whenever the active set — which assets sit at their bounds — is fixed. The algorithm exploits this in three steps:

  1. Start at λ = ∞, where the portfolio concentrates on the highest-return asset within bounds (init_algo).

  2. Solve the KKT system for the current active set to find α and β by block elimination, then decrease λ until one of two events occurs (main loop in cla.py):

    • a free asset hits its bound (leaves the free set), or
    • a blocked asset's KKT multiplier changes sign (enters the free set).

    General inequality rows Gw ≤ h add the row analogue of these two events: an inactive row becomes binding when its slack reaches zero, and an active row is released when its multiplier reaches zero. An active row is carried through the same block-eliminated solve as an extra equality row, so the structure is preserved.

  3. Update the active set (exactly one asset or constraint row changes status) and repeat until λ ≤ 0.

Because only one coordinate changes per step and each step requires only a single linear solve, the algorithm traces the full frontier cheaply and exactly — no approximation needed.

✨ Features

  • Efficient computation of the entire efficient frontier
  • Fluent builder API — CLA.problem(mean, cov).long_only().budget().trace() — as a readable alternative to the explicit constructor
  • Box bounds on every weight, plus general linear equality constraints Aw = b (budget, dollar-neutral, sector/factor neutrality) and general linear inequality constraints Gw ≤ h (group or sector exposure caps)
  • Factor covariance backend: exact frontiers for diagonal-plus-low-rank covariances (factor models, RMT-cleaned matrices) in O(nk) memory via the Woodbury identity
  • Visualization of the efficient frontier using Plotly
  • Computation of the maximum Sharpe ratio portfolio
  • Fully tested and documented codebase

🚀 Installation

Using pip

pip install cvxcla

To include plotting support (Plotly and Kaleido):

pip install cvxcla[plot]

Development Setup

To set up a development environment:

  1. Clone the repository:

    git clone https://github.com/cvxgrp/cvxcla.git
    cd cvxcla
    
  2. Create a virtual environment and install dependencies:

    make install
    

This will:

  • Install the uv package manager
  • Create a Python 3.12 virtual environment
  • Install all dependencies from pyproject.toml

🔧 Usage

Here's a simple example of how to use cvxcla to compute the efficient frontier:

import numpy as np
# Set a seed for reproducibility
np.random.seed(42)
from cvxcla import CLA

# Define your portfolio problem
n = 10  # Number of assets
mean = np.random.randn(n)  # Expected returns
cov = np.random.randn(n, n)
covariance = cov @ cov.T  # Covariance matrix
lower_bounds = np.zeros(n)  # No short selling
upper_bounds = np.ones(n)  # No leverage

# Create a CLA instance
cla = CLA(
    mean = mean,
    covariance = covariance,
    lower_bounds = lower_bounds,
    upper_bounds = upper_bounds,
    a = np.ones((1, n)),  # Fully invested constraint
    b = np.ones(1)
)

# Access the efficient frontier
frontier = cla.frontier

# Get the maximum Sharpe ratio portfolio
max_sharpe_ratio, max_sharpe_weights = frontier.max_sharpe
print(f"Maximum Sharpe ratio: {max_sharpe_ratio:.6f}")
# Print first few weights to avoid long output
print(f"First 3 weights: {max_sharpe_weights[:3]}")
Maximum Sharpe ratio: 2.946979
First 3 weights: [0.         0.         0.08509841]

The same problem reads more declaratively through the fluent builder, reached via CLA.problem(...). Each step maps onto one constructor argument, and the terminal .trace() returns the identical CLA:

# Same fully-invested, long-only problem as above, assembled fluently.
built = CLA.problem(mean, covariance).long_only().budget().trace()

# Pure sugar over the constructor: it returns the identical frontier.
assert built.frontier.max_sharpe[0] == cla.frontier.max_sharpe[0]

.bounds, .equality, and .inequality add general bounds and constraints, as in the examples below.

For visualization, you can plot the efficient frontier:

# Plot the efficient frontier
fig = frontier.plot(volatility=True)
fig.show()

Group and sector constraints

Pass g and h to add general inequality rows Gw ≤ h on top of the box bounds and the budget. For example, to cap the combined weight of the first three assets (a sector) at 40%:

import numpy as np
from cvxcla import CLA

n = 10
rng = np.random.default_rng(42)
mean = rng.standard_normal(n)
factor = rng.standard_normal((n, n))
covariance = factor @ factor.T  # symmetric positive definite

# sum of the first three weights ≤ 0.40
g = np.zeros((1, n))
g[0, :3] = 1.0
h = np.array([0.40])

cla = CLA(
    mean=mean,
    covariance=covariance,
    lower_bounds=np.zeros(n),
    upper_bounds=np.full(n, 0.35),
    a=np.ones((1, n)),  # fully invested
    b=np.ones(1),
    g=g,                # inequality matrix  G  (p x n)
    h=h,                # inequality vector  h  (p,)
)

# every turning point now satisfies the sector cap
assert all(tp.weights[:3].sum() <= 0.40 + cla.tol for tp in cla.turning_points)

g/h default to None, so omitting them recovers the equality-only problem unchanged. Stack multiple rows in g for several caps at once, and express a floor by negating a row (-Gw ≤ -h).

Factor models at scale

For diagonal-plus-low-rank covariances (factor risk models, eigenvalue-clipped covariances) pass a FactorCovariance instead of a dense matrix; the algorithm then never forms an n-by-n matrix:

import numpy as np
from cvxcla import CLA, FactorCovariance

rng = np.random.default_rng(42)
n, k = 10_000, 50
covariance = FactorCovariance(
    d=rng.uniform(0.1, 0.5, n),      # idiosyncratic variances
    u=rng.standard_normal((n, k)),   # factor loadings
    delta=rng.uniform(0.5, 2.0, k),  # factor variances, (k,) or (k, k)
)

See the factor backend documentation for the protocol, the math, and benchmarks against the dense path.

🧪 Testing

Run the test suite with:

make test

or directly via pytest:

uv run pytest

🧹 Code Quality

Format and lint the code with:

make fmt

📖 Documentation

👥 Contributing

Contributions are welcome! Please feel free to submit a Pull Request.

  1. Fork the repository
  2. Create your feature branch (git checkout -b feature/amazing-feature)
  3. Run the tests to make sure everything works (make test)
  4. Format your code (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

📄 License

This project is licensed under the MIT License - see the LICENSE file for details.

🔍 Related Projects

  • PyCLA by Philipp Schiele - A previous implementation of the Critical Line Algorithm in Python.

  • CLA by Martin Dengler - The original implementation by David Bailey and Marcos Lopez de Prado.

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

cvxcla-1.8.0.tar.gz (5.3 MB view details)

Uploaded Source

Built Distribution

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

cvxcla-1.8.0-py3-none-any.whl (44.1 kB view details)

Uploaded Python 3

File details

Details for the file cvxcla-1.8.0.tar.gz.

File metadata

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

File hashes

Hashes for cvxcla-1.8.0.tar.gz
Algorithm Hash digest
SHA256 83496538523b83baedc33c4956736be17b6831558b85ccd4c187e7081c2b2ce2
MD5 181745c3783cb00cf04a790f2ada523a
BLAKE2b-256 923ec2730e75693779398c46d42fd66f08487803f764897075156426931ae975

See more details on using hashes here.

Provenance

The following attestation bundles were made for cvxcla-1.8.0.tar.gz:

Publisher: release.yml on cvxgrp/cvxcla

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

File details

Details for the file cvxcla-1.8.0-py3-none-any.whl.

File metadata

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

File hashes

Hashes for cvxcla-1.8.0-py3-none-any.whl
Algorithm Hash digest
SHA256 0c689da644bd7202670d8c6b95b1d088d58216389a9614287b7614273cf5dd2c
MD5 3c3025c9d7e5a8d8c410f582c7d664ae
BLAKE2b-256 315104e28dec2775e0359a0dd1bbd244bf80a54660f650b652d8440e919caf9e

See more details on using hashes here.

Provenance

The following attestation bundles were made for cvxcla-1.8.0-py3-none-any.whl:

Publisher: release.yml on cvxgrp/cvxcla

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