Skip to content

A flexible framework for active LP strategy simulations that use the full Uniswap v3 swap history

Notifications You must be signed in to change notification settings

miquelmalaga/active-strategy-framework

 
 

Repository files navigation

Uniswap v3 Active Strategy Framework and Simulator

Banner

Introduction

This repository contains several python scripts that are used by Gamma Strategies to simulate the performance of Uniswap v3 liquidity provision strategies' performance and evaluate risks. The main scripts of the package are:

  1. ActiveStrategyFramework.py base code of the framework which executues a Strategy, conducting either back-testing simulations (simulate_strategy function and passing in historical swap data), or conducting a live implementation of the strategy.
  2. ResetStrategy.py first implementation of a Strategy which uses the empirical distribution of returns in order to predict future prices and set ranges for the LP positions.
  3. AutoRegressiveStrategy.py second implementation of the Strategy, using an AR(1)-GARCH(1,1) model.
  4. GetPoolData.py which downloads the data necessary for the simulations from TheGraph, Bitquery and Flipside Crypto.
  5. UNI_v3_funcs.py which is a slightly modified version of JNP777's Python implementation of Uniswap v3's liquidity math.

In order to provide an illustration of potential usage, we have included two Jupyter Notebooks that show how to use the framework:

We have constructed a flexible framework for active LP strategy simulations that use the full Uniswap v3 swap history in order to improve accurracy of fee income. Thefore simulations are available in the time period since Unsiwap v3 was released (May 5th 2021 is when swap data starts to show up consistently).

Simulating your own strategy

In order to simulate your own strategy you should clone this repository to your computer, and implement your algorithm in a new Strategy script, where you define a class which must include the following functions (see the ResetStrategy.py script for an example):

  1. set_liquidity_ranges computes where the LP ranges are set in an LP strategy and stores the virtual liquidity placed for each position.
  2. check_strategy to implement your algorithm's rebalancing logic.
  3. dict_components to extract the relevant data from each strategy observation in order to evaluate performance and plot charts.

Once you have your Strategy class defined, you can use the ActiveStrategyFramework.py structure to conduct backtesting simulations or run the code live. See the Jupyter notebooks for how to conduct the implementation.

The template is currently adapted to the strategies used by Visor Finance's Hypervisor, which set a base liquidity provision position, and a limit one with the tokens that are left over as may occur due to concentrated liquidity math and single sided deposits, but this could be generalized as well.

Data & simulating a different pool

The simulator is currently set up to analyze the USDC/WETH 0.3% pool. We use two data sources that you will need to adjust to update data / analyze a different pool:

  1. TheGraph: We obtain the full history of Uniswap v3 swaps from whatever pool we need, in order to accurately simulate the performance of the simulated strategy.
  2. Bitquery: We obtain historical Uniswap prices, quoted in whatever denomination we need. To update the data or query another pool you will need to sign up, obtain an API key and save it in afile in a file called config.py in this directory, as a string: BITQUERY_API_TOKEN = XXXXXXXX.
  3. Flipside Crypto: We obtain the virtual liquidity of the pool at every block, which is used to approximate the fee income earned in the pool, as described in their documentation.

To update the data get an API key and set DOWNLOAD_DATA = True in the notebook.

The current implementation is very flexible and allows to analyze a different pool with a few changes:

  1. Generate a new Flipside Crypto query like the one in the example_flipside_query.txt file, with the pool_address for the pair that you are interested. Note that due to a 100,000 row limit, we generate two queries for the USDC/WETH 0.3%, which explains the BLOCK_ID condition, to split the data into reasonable chunks. A less active pool might not need this split.
  2. Modify the get_liquidity_flipside function from GetPoolData.py to point to your new queries.

Potential Sources of inaccurracy

There are several potential sources for imprecision, as for example gas fees are not taken into account, and can have a significant impact on performance in particular for small positions in high fee regimes. There could be rounding issues from the Python implementation of the Solidity code, and differences from the pool price due to Bitquery's price feed not being identical to that of the pool (as expected).

About

A flexible framework for active LP strategy simulations that use the full Uniswap v3 swap history

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Jupyter Notebook 92.0%
  • Python 8.0%