TradingStrategyUniverse#
API documentation for tradeexecutor.strategy.trading_strategy_universe.TradingStrategyUniverse Python class in Trading Strategy framework.
- class TradingStrategyUniverse[source]#
Bases:
StrategyExecutionUniverse
A trading strategy universe using our own data feeds.
Captures both the market data feeds and factors needed to make trading decisions, like the strategy reserve currency and backtesting tweaks
- __init__(reserve_assets, data_universe=None, backtest_stop_loss_time_bucket=None, backtest_stop_loss_candles=None, required_history_period=None, options=None, price_data_delay_tolerance=None, pair_cache=<factory>)#
- Parameters:
reserve_assets (Collection[AssetIdentifier]) –
data_universe (Optional[Universe]) –
backtest_stop_loss_time_bucket (Optional[TimeBucket]) –
backtest_stop_loss_candles (Optional[GroupedCandleUniverse]) –
options (tradeexecutor.strategy.universe_model.UniverseOptions | None) –
price_data_delay_tolerance (datetime.timedelta | None) –
pair_cache (dict) –
- Return type:
None
Methods
__init__
(reserve_assets[, data_universe, ...])can_open_credit_supply
(timestamp, pair[, ...])Can we do a credit supply trade for a trading pair.
can_open_short
(timestamp, pair[, ...])Can we do a short trade for a trading pair.
can_open_spot
(timestamp, pair[, ...])Can we do a spot trade for a trading pair.
clone
()Create a copy of this universe.
create_from_dataset
(dataset[, ...])Create a universe from loaded dataset.
create_limited_pair_universe
(dataset, ...[, ...])Filters down the dataset for couple trading pair.
Create a trading universe based on list of (exchange, pair tuples)
create_multipair_universe
(dataset, ...[, ...])Create a trading universe where pairs match a filter conditions.
create_single_pair_universe
([chain_id, ...])Filters down the dataset for a single trading pair.
Get semi-human-readable filename id for this universe.
Get the credit supply trading pair.
get_latest_supply_apr
([asset, timestamp, ...])Get the latest supply APR for the lending asset.
get_pair_by_address
(address)Get a trading pair data by a smart contract address.
Get pair by its human-readable description.
get_pair_by_id
(internal_id)Get a trading pair data by its internal id (pair_id).
get_pair_count
()get_reserve_asset
()Get the default reserve asset.
get_shorting_pair
(pair)Get the shorting pair from trading pair
Get the single trading pair in this universe.
Get the single trading pair in this universe.
get_trading_broken_reason
(pair, ...[, ...])Can we trade a pair.
get_trading_pair
(pair)Get a pair by id or by its data description.
Does this trading universe has any lending data loaded
Is any lending data available.
has_lending_market_available
(timestamp, asset)Did an asset have a lending market available at certain historic point of time.
Is any liquidty data available.
Do we have data available to determine trade stop losses.
is_empty
()This is an empty universe
Can new trading pairs to be added to this universe over time.
Is this trading universe made for a single pair trading.
Iterate over all available credit supply pairs.
Iterate over all available trading pairs.
read_pickle_dangerous
(path)Write a pickled ouput of this universe.
validate
()Check that the created universe looks good.
Make sure all data is properly
write_pickle
(path)Write a pickled ouput of this universe
Attributes
The list of reserve assets used in this strategy.
Trading universe datasets.
Are we using special take profit/stop loss trigger data.
Special Pandas data feed for candles used only during the backtesting.
How much historic data is needed by the strategy to make its decision.
What options were applied to this universe
How much lag we allow in the price feed.
Translated asset and trading pair identifier cache.
End timestamp of the data.
Start timestamp of the data.
Backwards compatibility method.
- data_universe: Optional[Universe]#
Trading universe datasets.
This encapsulates more generic Universe class from tradingstrategy package.
- backtest_stop_loss_time_bucket: Optional[TimeBucket]#
Are we using special take profit/stop loss trigger data.
If we are, what is the time granularity of this data.
- backtest_stop_loss_candles: Optional[GroupedCandleUniverse]#
Special Pandas data feed for candles used only during the backtesting.
This allows us to simulate take profit and stop loss trigger orders that would be based on real-time market data in a live execution environment.
- required_history_period: Optional[timedelta]#
How much historic data is needed by the strategy to make its decision.
This is the explicit time frame in days or hours for the historical data before today for the strategy to make a decision.
This will limit the amount of data downloaded from the oracle and greatly optimise the strategy decision execution.
E.g. for 90 days you can use datetime.timedelta(days=90)
- options: tradeexecutor.strategy.universe_model.UniverseOptions | None#
What options were applied to this universe
Could be e.g.
Loaded data range
Backtested data range
Not set in legacy code paths. Currently the life cycle of this variable is not well-defined, mostly used to pass the backtest data range around, and set before backtest run, not during the universe construction.
- price_data_delay_tolerance: datetime.timedelta | None#
How much lag we allow in the price feed.
This can be set for a very high value when working with open ended universes where the liquidity pools are disappear when liquidity providers leave.
TODO: Use carefully - this is mostly a workaround variable and will have a more robust TVL/liquidity solution in the future.
- pair_cache: dict#
Translated asset and trading pair identifier cache.
When called asset.set_tags() or setting asset.other_data() it is stored here.
See
warm_up_data()
- get_cache_key()[source]#
Get semi-human-readable filename id for this universe.
Note
Currently does not capture all the nuances of the data. Must be defined later to produce an accurate hash on Universe.
- Return type:
- property start_at: datetime#
Start timestamp of the data.
Valid for backtesting only
Based on OHLCV candles
- property end_at: datetime#
End timestamp of the data.
Valid for backtesting only
Based on OHLCV candles
- property universe#
Backwards compatibility method.
Deprecate in some point.
- is_open_ended_universe()[source]#
Can new trading pairs to be added to this universe over time.
- Returns:
True if the trading universe may contain hundreds or thousands of trading pairs.
- Return type:
- has_lending_data()[source]#
Is any lending data available.
Note
For live trading, lending candles are not available, but any lending rates are directly updated from on-chain sources.
- Return type:
- get_pair_by_id(internal_id)[source]#
Get a trading pair data by its internal id (pair_id).
- Parameters:
internal_id (int) –
- Return type:
- get_trading_pair(pair)[source]#
Get a pair by id or by its data description.
- Parameters:
pair (int | tradingstrategy.pair.DEXPair) – Trading pair internal id or DEXPair object
- Returns:
Tradind strategy pair definition.
- Raises:
PairNotFoundError – If we have not loaded data for the given pair id.
- Return type:
- can_open_spot(timestamp, pair, liquidity_threshold=None)[source]#
Can we do a spot trade for a trading pair.
To be used with backtesting. We will check a spot market exists at a certain historic point of time.
- Parameters:
timestamp (Timestamp) – When
pair (TradingPairIdentifier) – The wanted trading pair
liquidity_threshold – Not implemented yet.
- Returns:
True if we can open a spot position.
- Return type:
- has_lending_market_available(timestamp, asset, liquidity_threshold=None, market_metric=LendingCandleType.variable_borrow_apr, data_lag_tolerance=Timedelta('7 days 00:00:00'))[source]#
Did an asset have a lending market available at certain historic point of time.
To be used with backtesting. We will check a lending market exists at a certain historic point of time.
- Parameters:
timestamp (pandas._libs.tslibs.timestamps.Timestamp | datetime.datetime) – When
pair – The wanted trading pair
liquidity_threshold – Not implemented yet.
asset (AssetIdentifier) –
market_metric (LendingCandleType) –
- Returns:
True if we can open a spot position.
- Return type:
- warm_up_data()[source]#
Make sure all data is properly
Trading pair data must be preprocessed before we can call set_tags() or set other_data
See
pair_cache
for details
- can_open_short(timestamp, pair, liquidity_threshold=None)[source]#
Can we do a short trade for a trading pair.
To be used with backtesting. We will check a lending market exists at a certain historic point of time for both base and quote asset.
- Parameters:
timestamp (pandas._libs.tslibs.timestamps.Timestamp | datetime.datetime) – When
pair (TradingPairIdentifier) – The wanted trading pair
liquidity_threshold – Not implemented yet.
- Returns:
True if we can open a short position.
- Return type:
- can_open_credit_supply(timestamp, pair, liquidity_threshold=None)[source]#
Can we do a credit supply trade for a trading pair.
To be used with backtesting. We will check a lending market exists at a certain historic point of time for both base and quote asset.
- Parameters:
timestamp (pandas._libs.tslibs.timestamps.Timestamp | datetime.datetime) – When
pair (TradingPairIdentifier) – The wanted trading pair
liquidity_threshold – Not implemented yet.
- Returns:
True if we can open a credit supply position.
- Return type:
- clone()[source]#
Create a copy of this universe.
Any dataframes are now copied, but set by reference, so do not mutate in place.
- Return type:
- static read_pickle_dangerous(path)[source]#
Write a pickled ouput of this universe.
Warning
Only use for trusted input. Python pickle issue.
- Parameters:
path (Path) –
- Return type:
- is_empty()[source]#
This is an empty universe
without trading pairs
…or without candles
- Return type:
- is_single_pair_universe()[source]#
Is this trading universe made for a single pair trading.
Note that even a single pair universe may contain two trading pairs, if we need intermediate routing pairs. E.g. AAVE -> BNB -> BUSD/
- Return type:
- has_stop_loss_data()[source]#
Do we have data available to determine trade stop losses.
Note that this applies for backtesting only - when doing production trade execution, stop loss data is not part of the universe but real time pricing comes directly from the exchange using real-time side channels.
- Return type:
- validate()[source]#
Check that the created universe looks good.
- Raises:
TradingUniverseIssue – In the case of detected issues
- get_pair_by_human_description(desc)[source]#
Get pair by its human-readable description.
See
tradingstrategy.pair.PandasPairUniverse.get_pair_by_human_description()
The trading pair must be loaded in the exchange universe.
- iterate_pairs()[source]#
Iterate over all available trading pairs.
Different from
tradingstrategy.pair.PandasPairUniverse.iterate_pairs()
as this yields TradingPairIdentifier instances
- Return type:
- iterate_credit_for_reserve()[source]#
Iterate over all available credit supply pairs.
We can deposit our reserve and get aToken for interest
- Return type:
- create_single_pair_universe(chain_id=None, exchange_slug=None, base_token=None, quote_token=None, pair=None)[source]#
Filters down the dataset for a single trading pair.
This is ideal for strategies that only want to trade a single pair.
- Parameters:
pair (Optional[Union[Tuple[ChainId, str | None, str, str, float], Tuple[ChainId, str | None, str, str]]]) – Give the pair we create the universe for
exchange_slug (Optional[str]) –
Legacy.
This or pair.
chain_id (Optional[ChainId]) –
Legacy.
This or pair.
Legacy.
This or pair.
Legacy.
This or pair.
dataset (Dataset) –
- Return type:
- static create_limited_pair_universe(dataset, chain_id, exchange_slug, pairs, reserve_asset_pair_ticker=None, forward_fill=False)[source]#
Filters down the dataset for couple trading pair.
This is ideal for strategies that only want to trade few pairs, or a single pair using three-way trading on a single exchange.
The university reserve currency is set to the quote token of the first pair.
- Parameters:
dataset (Dataset) – Datasets downloaded from the server
pairs (Set[Tuple[str, str]]) – List of trading pairs as ticket tuples. E.g. [ (“WBNB, “BUSD”), (“Cake”, “WBNB”) ]
reserve_asset_pair_ticker (Optional[Tuple[str, str]]) – Choose the quote token of this trading pair as a reserve asset. This must be given if there are several pairs (Python set order is unstable).
forward_fill –
Forward-fill the data.
When working with sparse data (gaps in candles), many strategies need these gaps to be filled. Setting this parameter True will automatically forward-fill any data we are loading from the dataset.
See forward fill for more information.
chain_id (ChainId) –
exchange_slug (str) –
- Return type:
- get_pair_by_address(address)[source]#
Get a trading pair data by a smart contract address.
- Parameters:
address (str) –
- Return type:
- get_single_pair()[source]#
Get the single trading pair in this universe.
- Raises:
Exception – If we have more than one trading pair
- Return type:
- static create_multipair_universe(dataset, chain_ids, exchange_slugs, quote_tokens, reserve_token, factory_router_map, liquidity_resample_frequency=None)[source]#
Create a trading universe where pairs match a filter conditions.
These universe may contain thousands of trading pairs. This is for strategies that trade across multiple pairs, like momentum strategies.
- Parameters:
dataset (Dataset) – Datasets downloaded from the oracle
quote_tokens (Iterable[str]) – Allowed quote tokens as smart contract addresses
reserve_token (str) – The token addresses that are used as reserve assets.
factory_router_map (Dict[str, tuple]) – Ensure we have a router address for every exchange we are going to use. TODO: In the future this information is not needed.
liquidity_resample_frequency (Optional[str]) –
Create a resampled liquidity universe instead of accurate one.
If given, this will set Universe.resampled_liquidity attribute.
Using
ResampledLiquidityUniverse
will greatly speed up backtests that estimate trading pair liquidity, by trading off sample accuracy for code execution speed.Must be a Pandas frequency string value, like 1d.
Note that resamping itself takes a lot of time upfront, so you want to use this only if the backtest takes lont time.
- Return type:
- static create_from_dataset(dataset, reserve_asset=None, forward_fill=False, forward_fill_until=None)[source]#
Create a universe from loaded dataset.
Example:
def create_trading_universe( timestamp: datetime.datetime, client: Client, execution_context: ExecutionContext, universe_options: UniverseOptions, ) -> TradingStrategyUniverse: # ... a lot of code goes here # First phase: # Load all data files and JSON API streams # from the server needed to construct the trading universe dataset = load_partial_data( client=client, time_bucket=Parameters.candle_time_bucket, pairs=pairs_df, execution_context=execution_context, universe_options=universe_options, liquidity=True, liquidity_time_bucket=TimeBucket.d1, liquidity_query_type=OHLCVCandleType.tvl_v2, ) # Second phase # Construct trading strategy universe from the dataset. # Wrangles/massages data out for the issues strategy_universe = TradingStrategyUniverse.create_from_dataset( dataset, # USDC on Base reserve_asset="0x833589fCD6eDb6E08f4c7C32D4f71b54bdA02913", # Because we use sparse candles, we need to fill gaps between candles forward_fill=True, # Because some pairs may have died during real-time trading, we still # need to forward fill until today, not the last day they had a trade forward_fill_until=timestamp, ) return strategy_universe
For more code examples see
load_trading_and_lending_data()
.- Parameters:
reserve_asset (str) –
Which reserve asset to use.
As the token address or symbol.
If not given try to guess from the dataset.
Assume all trading pairs have the same quote token and that is our reserve asset
If dataset has trading pairs with different quote tokens, aborts
Examples:
0x2791bca1f2de4661ed88a30c99a7a9449aa84174 (USDC.e bridged address on Polygon)
0x3c499c542cEF5E3811e1192ce70d8cC03d5c3359 (USDC native)
forward_fill –
Forward-fill the data.
When working with sparse data (gaps in candles), many strategies need these gaps to be filled. Setting this parameter True will automatically forward-fill any data we are loading from the dataset.
See forward fill for more information.
dataset (Dataset) –
forward_fill_until (datetime.datetime | None) –
- static create_multichain_universe_by_pair_descriptions(dataset, pairs, reserve_token_symbol)[source]#
Create a trading universe based on list of (exchange, pair tuples)
This is designed for backtesting multipe pairs across different chains. The created universe do not have any routing options and thus cannot make any trades.
- Parameters:
dataset (Dataset) – Datasets downloaded from the oracle
pairs (Collection[Union[Tuple[ChainId, str | None, str, str, float], Tuple[ChainId, str | None, str, str]]]) –
List of trading pairs to filter down.
The pair set is desigend to be small, couple of dozens of pairs max.
See
HumanReadableTradingPairDescription
.reserve_token_symbol (str) –
The token symbol of the reverse asset.
Because we do not support routing, we just pick the first matching token.
- Return type:
- get_credit_supply_pair()[source]#
Get the credit supply trading pair.
This trading pair identifies the trades where we move our strategy reserve currency to Aave lending pool to gain interest.
Assume we move our single reserve currency to a single lending reserve we know for it.
- Returns:
Credit supply trading pair with ticket (aToken for reserve asset, reserve asset)
- Return type:
- get_shorting_pair(pair)[source]#
Get the shorting pair from trading pair
This trading pair identifies the trades where we borrow asset from Aave against collateral.
- Parameters:
pair (TradingPairIdentifier) – Trading pair of the asset to be shorted
- Returns:
Short pair with ticker (vToken for borrowed asseet, aToken for reserve asset)
- Return type:
- get_trading_broken_reason(pair, min_candles_required, min_price=1e-08, max_price=1000000)[source]#
Can we trade a pair.
Check if we can trade a particular trading pair.
Work around spotty low cap coins
Check that we have minimum amout of bars of data
Check that price data does not look weird
- Parameters:
pair (TradingPairIdentifier) – Trading pair
min_candles_required (int) – How many bars of adta we need
min_price – Avoid low cap tokens with float64 breaking prices
max_pric – Avoid low cap tokens with float64 breaking prices
- Returns:
A string why the trading pair is broken.
None if good.
- Return type:
str | None
- get_latest_supply_apr(asset=None, timestamp=None, tolerance=datetime.timedelta(days=1))[source]#
Get the latest supply APR for the lending asset.
- Parameters:
asset (tradeexecutor.state.identifier.AssetIdentifier | None) –
Specify the asset to get the rate for If none, use the reserve asset.
Must be non-wrapped, not aToken.
timestamp (datetime.datetime | None) – Specify timestamp to get the rate for If none, return the latest rate
tolerance (timedelta) –
Lookback tolerance with stale data.
Default to 1 days if not given.
- Returns:
Latest supply APR rate
- Return type:
- reserve_assets: Collection[AssetIdentifier]#
The list of reserve assets used in this strategy.
Currently we support only one reserve asset per strategy, though in the future there can be several.
Usually return the list of a BUSD/USDC/similar stablecoin.
TODO: Migrate to Set[] in all the code
- __init__(reserve_assets, data_universe=None, backtest_stop_loss_time_bucket=None, backtest_stop_loss_candles=None, required_history_period=None, options=None, price_data_delay_tolerance=None, pair_cache=<factory>)#
- Parameters:
reserve_assets (Collection[AssetIdentifier]) –
data_universe (Optional[Universe]) –
backtest_stop_loss_time_bucket (Optional[TimeBucket]) –
backtest_stop_loss_candles (Optional[GroupedCandleUniverse]) –
options (tradeexecutor.strategy.universe_model.UniverseOptions | None) –
price_data_delay_tolerance (datetime.timedelta | None) –
pair_cache (dict) –
- Return type:
None