Source code for tradeexecutor.strategy.dust
"""Dust amounts and epsilon rounding.
Because of very small and very accurate token units,
a lot of trades may end up having rounding artifacts.
We need to deal with these rounding artifacts by checking for "dust".
"""
from _decimal import Decimal
from decimal import Decimal
from tradeexecutor.state.identifier import TradingPairIdentifier, AssetIdentifier
from tradeexecutor.state.types import Percent
#: The absolute number of tokens we consider the value to be zero
#:
#:
#: Because of funny %s of values divided near zero,
#: we cannot use relative comparison near zero values.
#:
#:
DEFAULT_DUST_EPSILON = Decimal(10 ** -10)
#: The default % we allow the balance to drift before we consider it a mismatch.
#:
#: Set to 5 BPS
#:
DEFAULT_RELATIVE_EPSILON = 5 * 10 ** -4
#: When to close 1delta positions
ONE_DELTA_CLOSE_EPSILON = 1 * 10**-4
[docs]def get_dust_epsilon_for_pair(pair: TradingPairIdentifier) -> Decimal:
"""Get the dust threshold for a trading pair.
See also :py:func:`get_close_epsilon_for_pair`.
:param pair:
Trading pair identifier.
:return:
Maximum amount of units we consider "zero".
"""
return get_dust_epsilon_for_asset(pair.base)
[docs]def get_close_epsilon_for_pair(pair: TradingPairIdentifier) -> Decimal:
"""Get the close threshold for a trading pair.
- Currently same as dust epsilon
See also :py:func:`get_dust_epsilon_for_pair`.
:param pair:
Trading pair identifier.
:return:
Maximum amount of units we consider "zero".
"""
return get_dust_epsilon_for_asset(pair.base)
[docs]def get_dust_epsilon_for_asset(asset: AssetIdentifier) -> Decimal:
"""Get the dust threshold for a trading pair.
:param pair:
Trading pair identifier.
:return:
Maximum amount of units we consider "zero".
"""
# Hardcoded rules for now.
# Some practical problems we have run across in backtesting.
# We have wrapped and non-wrapped token symbols as we are backtesting both on DEX and CEX data
if asset.token_symbol in ("WBTC", "BTC"):
return Decimal(10 ** -7)
elif asset.token_symbol in ("ETH", "WETH"):
return Decimal(10 ** -7)
elif asset.token_symbol in ("USDC", "USDC.e"):
return Decimal(0.1)
elif asset.token_symbol in ("aPolUSDC", "aEthUSDC"):
return Decimal(0.1)
elif "variableDebt" in asset.token_symbol:
# 1delta closing epsilon higher than default
return ONE_DELTA_CLOSE_EPSILON
else:
return DEFAULT_DUST_EPSILON
[docs]def get_relative_epsilon_for_asset(asset: AssetIdentifier) -> Percent:
"""Get the relative threshold for a trading pair.
:param pair:
Trading pair identifier.
:return:
Maximum amount of units we consider "zero".
"""
if asset.token_symbol in ("aPolUSDC", "USDC"):
# Temp allow 0.5% tolerance
return 0.005
# 5 BPS
return DEFAULT_RELATIVE_EPSILON
[docs]def get_relative_epsilon_for_pair(pair: TradingPairIdentifier) -> Percent:
return get_relative_epsilon_for_asset(pair.base)