Skip to main content

Simple simulator for investors

Project description

🔄 cvxsimulator

PyPI version Apache 2.0 License Downloads Coverage Status Renovate enabled

Open in GitHub Codespaces

A simple yet powerful simulator for investment strategies and portfolio backtesting.

Given a universe of $m$ assets we are given prices for each of them at time $t_1, t_2, \ldots t_n$, e.g. we operate using an $n \times m$ matrix where each column corresponds to a particular asset.

In a backtest we iterate in time (e.g. row by row) through the matrix and allocate positions to all or some of the assets. This tool helps to simplify the accounting. It keeps track of the available cash, the profits achieved, etc.

Analytics

📥 Installation

Install cvxsimulator via pip:

pip install cvxsimulator

📊 Creating Portfolios

The simulator is completely agnostic to the trading policy/strategy. Our approach follows a rather common pattern:

We demonstrate these steps with simple example policies. They are never good strategies, but are always valid ones.

Create the builder object

The user defines a builder object by loading prices and initializing the amount of cash used in an experiment:

import pandas as pd
from cvx.simulator import Builder

# For doctest, we'll create a small DataFrame instead of reading from a file
dates = pd.date_range('2020-01-01', periods=5)
prices = pd.DataFrame({
    'A': [100, 102, 104, 103, 105],
    'B': [50, 51, 52, 51, 53],
    'C': [200, 202, 198, 205, 210],
    'D': [75, 76, 77, 78, 79]
}, index = dates)
b = Builder(prices=prices, initial_aum=1e6)

Prices have to be valid, there may be NaNs only at the beginning and the end of each column in the frame. There can be no NaNs hiding in the middle of any time series.

It is also possible to specify a model for trading costs. The builder helps to fill up the frame of positions. Only once done we construct the actual portfolio.

Loop through time

We have overloaded the __iter__ and __setitem__ methods to create a custom loop. Let's start with a first strategy. Each day we choose two names from the universe at random. Buy one (say 0.1 of your portfolio wealth) and short one the same amount.

import numpy as np

np.random.seed(42)  # Set seed for reproducibility

for t, state in b:
    # pick two assets deterministically for doctest
    pair = ['A', 'B']  # Use first two assets instead of random choice
    # compute the pair
    units = pd.Series(index=state.assets, data=0.0)
    units[pair] = [state.nav, -state.nav] / state.prices[pair].values
    # update the position
    b.position = 0.1 * units
    # Do not apply trading costs
    b.aum = state.aum
    # Check the final positions
    b.units.iloc[-1][['A', 'B']]

Here t is the growing list of timestamps, e.g. in the first iteration t is $t1$, in the second iteration it will be $t1, t2$ etc.

A lot of magic is hidden in the state variable. The state gives access to the currently available cash, the current prices and the current valuation of all holdings.

Here's a slightly more realistic loop. Given a set of $4$ assets we want to implement the popular $1/n$ strategy.

b2 = Builder(prices=prices, initial_aum=1e6)

for t, state in b2:
    # each day we invest a quarter of the capital in the assets
    b2.position = 0.25 * state.nav / state.prices
    b2.aum = state.aum

# Check the final positions
b2.units.iloc[-1]

Note that we update the position at the last element in the t list using a series of actual units rather than weights or cashpositions. The builder class also exposes setters for such alternative conventions.

b3 = Builder(prices=prices, initial_aum=1e6)

for t, state in b3:
    # each day we invest a quarter of the capital in the assets
    b3.weights = np.ones(4) * 0.25
    b3.aum = state.aum

    # Check the final positions
b3.units.iloc[-1]

Build the portfolio

Once finished it is possible to build the portfolio object:

b3 = Builder(prices=prices, initial_aum=1e6)

for t, state in b3:
    b3.weights = np.ones(4) * 0.25
    b3.aum = state.aum

# Build the portfolio from one of our builders
portfolio = b3.build()
# Verify the portfolio was created successfully
type(portfolio).__name__

📈 Analytics

The portfolio object supports further analysis and exposes a number of properties, e.g.:

b3 = Builder(prices=prices, initial_aum=1e6)

for t, state in b3:
    b3.weights = np.ones(4) * 0.25
    b3.aum = state.aum

portfolio = b3.build()

It is possible to generate a snapshot of the portfolio:

b3 = Builder(prices=prices, initial_aum=1e6)
for t, state in b3:
    b3.weights = np.ones(4) * 0.25
    b3.aum = state.aum
portfolio = b3.build()

# Generate a snapshot (returns a plotly figure)
fig = portfolio.snapshot()

🛠️ Development

UV Package Manager

Start with:

make install

This will install uv and create the virtual environment defined in pyproject.toml and locked in uv.lock.

Marimo Notebooks

We install marimo on the fly within the aforementioned virtual environment. Execute:

make marimo

This will install and start marimo for interactive notebook development.

📚 Documentation

  • Full documentation is available at cvxgrp.org/simulator/book
  • API reference can be found in the documentation
  • Example notebooks are included in the repository under the book directory

🤝 Contributing

Contributions are welcome! Here's how you can contribute:

  1. Fork the repository
  2. Create a feature branch: git checkout -b feature-name
  3. Commit your changes: git commit -m 'Add some feature'
  4. Push to the branch: git push origin feature-name
  5. Open a pull request

Please make sure to update tests as appropriate and follow the code style of the project.

📄 License

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

Copyright 2023 Stanford University Convex Optimization Group

Project details


Release history Release notifications | RSS feed

Download files

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

Source Distribution

cvxsimulator-1.5.1.tar.gz (7.7 MB view details)

Uploaded Source

Built Distribution

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

cvxsimulator-1.5.1-py3-none-any.whl (20.0 kB view details)

Uploaded Python 3

File details

Details for the file cvxsimulator-1.5.1.tar.gz.

File metadata

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

File hashes

Hashes for cvxsimulator-1.5.1.tar.gz
Algorithm Hash digest
SHA256 484f91c11eacaa8763f06c379fb016710eed97bf28e9ecf8f7f04085f730f340
MD5 5e18cd79517a71c6676e392ad116aeda
BLAKE2b-256 b534ceb60bcaab7eb5608b8cef01b94e38cfaacc36708ecb8384a0ed354a0f64

See more details on using hashes here.

Provenance

The following attestation bundles were made for cvxsimulator-1.5.1.tar.gz:

Publisher: release.yml on cvxgrp/simulator

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

File details

Details for the file cvxsimulator-1.5.1-py3-none-any.whl.

File metadata

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

File hashes

Hashes for cvxsimulator-1.5.1-py3-none-any.whl
Algorithm Hash digest
SHA256 7fd62ebb88a4dda6c8e1d417bb8150771934aeee5dbc0460bf50574c1cd81354
MD5 c684a6afe0e3935c2e683fc6baaa4eba
BLAKE2b-256 48d8c5e9fed4f951e1ce9c3b584601b785be045537bc868f20f05bed818831d8

See more details on using hashes here.

Provenance

The following attestation bundles were made for cvxsimulator-1.5.1-py3-none-any.whl:

Publisher: release.yml on cvxgrp/simulator

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