"""Default protocols support for EVM blockchains.
See :py:mod:`tradeexecutor.strategy.generic.pair_configurator`.
"""
from typing import Set
from web3 import Web3
from eth_defi.aave_v3.deployment import fetch_deployment as fetch_aave_deployment
from eth_defi.uniswap_v3.deployment import fetch_deployment as fetch_uniswap_v3_deployment
from eth_defi.one_delta.deployment import fetch_deployment as fetch_1delta_deployment
from eth_defi.aave_v3.constants import AAVE_V3_DEPLOYMENTS
from eth_defi.one_delta.constants import ONE_DELTA_DEPLOYMENTS
from tradeexecutor.ethereum.routing_data import base_uniswap_v3_address_map
from tradingstrategy.chain import ChainId
from tradingstrategy.exchange import ExchangeUniverse, ExchangeType
from tradeexecutor.state.identifier import TradingPairIdentifier
from tradeexecutor.strategy.default_routing_options import TradeRouting
from tradeexecutor.strategy.generic.pair_configurator import PairConfigurator, ProtocolRoutingId, ProtocolRoutingConfig
from tradeexecutor.strategy.generic.default_protocols import default_match_router, default_supported_routers
from tradeexecutor.strategy.reserve_currency import ReserveCurrency
from tradeexecutor.strategy.trading_strategy_universe import TradingStrategyUniverse
[docs]def get_exchange_type(
exchange_universe: ExchangeUniverse,
pair: TradingPairIdentifier,
) -> ExchangeType:
assert pair.exchange_address is not None, f"Pair missing exchange_address: {pair}"
exchange = exchange_universe.get_by_chain_and_factory(ChainId(pair.chain_id), pair.exchange_address)
assert exchange is not None, f"Exchange address {pair.exchange_address} for pair {pair}: data not loaded"
return exchange.exchange_type
[docs]def create_uniswap_v2_adapter(
web3: Web3,
strategy_universe: TradingStrategyUniverse,
routing_id: ProtocolRoutingId,
) -> ProtocolRoutingConfig:
# TODO: Avoid circular imports for now
from tradeexecutor.ethereum.uniswap_v2.uniswap_v2_live_pricing import UniswapV2LivePricing
from tradeexecutor.ethereum.uniswap_v2.uniswap_v2_valuation import UniswapV2PoolRevaluator
from tradeexecutor.ethereum.routing_data import create_uniswap_v2_compatible_routing
assert routing_id.router_name == "uniswap-v2"
assert len(strategy_universe.data_universe.chains) == 1
assert len(strategy_universe.reserve_assets) == 1
reserve = strategy_universe.get_reserve_asset()
assert reserve.token_symbol == "USDC"
exchange_universe = strategy_universe.data_universe.exchange_universe
chain_id = strategy_universe.get_single_chain()
exchange = exchange_universe.get_by_chain_and_slug(chain_id, routing_id.exchange_slug)
if exchange.exchange_slug == "quickswap":
routing_model = create_uniswap_v2_compatible_routing(
TradeRouting.quickswap_usdc,
ReserveCurrency.usdc,
chain_id,
)
elif exchange.exchange_slug == "sushi":
routing_model = create_uniswap_v2_compatible_routing(
TradeRouting.sushi_usdc,
ReserveCurrency.usdc,
chain_id,
)
else:
routing_model = create_uniswap_v2_compatible_routing(
TradeRouting.uniswap_v2_usdc,
ReserveCurrency.usdc,
chain_id,
)
pricing_model = UniswapV2LivePricing(
web3,
strategy_universe.data_universe.pairs,
routing_model,
)
valuation_model = UniswapV2PoolRevaluator(pricing_model)
return ProtocolRoutingConfig(
routing_id=routing_id,
routing_model=routing_model,
pricing_model=pricing_model,
valuation_model=valuation_model,
)
[docs]def create_uniswap_v3_adapter(
web3: Web3,
strategy_universe: TradingStrategyUniverse,
routing_id: ProtocolRoutingId,
) -> ProtocolRoutingConfig:
"""Always the same."""
# TODO: Avoid circular imports for now
from tradeexecutor.ethereum.uniswap_v3.uniswap_v3_live_pricing import UniswapV3LivePricing
from tradeexecutor.ethereum.uniswap_v3.uniswap_v3_routing import UniswapV3Routing
from tradeexecutor.ethereum.uniswap_v3.uniswap_v3_valuation import UniswapV3PoolRevaluator
from tradeexecutor.ethereum.routing_data import uniswap_v3_address_map
assert routing_id.router_name == "uniswap-v3"
assert len(strategy_universe.data_universe.chains) == 1
assert len(strategy_universe.reserve_assets) == 1
chain_id = strategy_universe.get_single_chain()
reserve_asset = strategy_universe.get_reserve_asset()
allowed_intermediary_pairs = UNISWAP_V3_INTERMEDIATE.get(chain_id, {})
if chain_id == ChainId.base:
# Special case for Base chain
address_map = base_uniswap_v3_address_map
else:
address_map = uniswap_v3_address_map
# TODO: Add intermediate tokens
routing_model = UniswapV3Routing(
address_map=address_map,
chain_id=chain_id,
reserve_token_address=reserve_asset.address,
allowed_intermediary_pairs=allowed_intermediary_pairs,
)
pricing_model = UniswapV3LivePricing(
web3,
strategy_universe.data_universe.pairs,
routing_model,
)
valuation_model = UniswapV3PoolRevaluator(pricing_model)
return ProtocolRoutingConfig(
routing_id=routing_id,
routing_model=routing_model,
pricing_model=pricing_model,
valuation_model=valuation_model,
)
[docs]def create_1delta_adapter(
web3: Web3,
strategy_universe: TradingStrategyUniverse,
routing_id: ProtocolRoutingId,
) -> ProtocolRoutingConfig:
# TODO: Avoid circular imports for now
from tradeexecutor.ethereum.one_delta.one_delta_live_pricing import OneDeltaLivePricing
from tradeexecutor.ethereum.one_delta.one_delta_routing import OneDeltaRouting
from tradeexecutor.ethereum.one_delta.one_delta_valuation import OneDeltaPoolRevaluator
assert routing_id.router_name == "1delta"
assert routing_id.lending_protocol_slug == "aave"
assert routing_id.exchange_slug == "uniswap-v3"
assert len(strategy_universe.data_universe.chains) == 1
assert len(strategy_universe.reserve_assets) == 1
chain_id = strategy_universe.get_single_chain()
chain_slug = chain_id.get_slug()
reserve_asset = strategy_universe.get_reserve_asset()
assert chain_slug in AAVE_V3_DEPLOYMENTS, f"Chain {chain_slug} not supported for Aave v3"
assert chain_slug in ONE_DELTA_DEPLOYMENTS, f"Chain {chain_slug} not supported for 1delta"
uniswap_v3_deployment = fetch_uniswap_v3_deployment(
web3,
"0x1F98431c8aD98523631AE4a59f267346ea31F984",
"0xE592427A0AEce92De3Edee1F18E0157C05861564",
"0xC36442b4a4522E871399CD717aBDD847Ab11FE88",
"0xb27308f9F90D607463bb33eA1BeBb41C27CE5AB6",
)
aave_v3_deployment = fetch_aave_deployment(
web3,
pool_address=AAVE_V3_DEPLOYMENTS[chain_slug]["pool"],
data_provider_address=AAVE_V3_DEPLOYMENTS[chain_slug]["data_provider"],
oracle_address=AAVE_V3_DEPLOYMENTS[chain_slug]["oracle"],
)
one_delta_deployment = fetch_1delta_deployment(
web3,
flash_aggregator_address=ONE_DELTA_DEPLOYMENTS[chain_slug]["broker_proxy"],
broker_proxy_address=ONE_DELTA_DEPLOYMENTS[chain_slug]["broker_proxy"],
quoter_address=ONE_DELTA_DEPLOYMENTS[chain_slug]["quoter"],
)
address_map = {
"one_delta_broker_proxy": one_delta_deployment.broker_proxy.address,
"one_delta_quoter": one_delta_deployment.quoter.address,
"aave_v3_pool": aave_v3_deployment.pool.address,
"aave_v3_data_provider": aave_v3_deployment.data_provider.address,
"aave_v3_oracle": aave_v3_deployment.oracle.address,
"factory": uniswap_v3_deployment.factory.address,
"router": uniswap_v3_deployment.swap_router.address,
"position_manager": uniswap_v3_deployment.position_manager.address,
"quoter": uniswap_v3_deployment.quoter.address
}
# TODO: Add intermediate tokens
routing_model = OneDeltaRouting(
address_map=address_map,
chain_id=chain_id,
reserve_token_address=reserve_asset.address,
allowed_intermediary_pairs={},
)
pricing_model = OneDeltaLivePricing(
web3,
strategy_universe.data_universe.pairs,
routing_model,
)
valuation_model = OneDeltaPoolRevaluator(pricing_model)
return ProtocolRoutingConfig(
routing_id=routing_id,
routing_model=routing_model,
pricing_model=pricing_model,
valuation_model=valuation_model,
)
[docs]def create_aave_v3_adapter(
web3: Web3,
strategy_universe: TradingStrategyUniverse,
routing_id: ProtocolRoutingId,
) -> ProtocolRoutingConfig:
# TODO: Avoid circular imports for now
from tradeexecutor.ethereum.uniswap_v3.uniswap_v3_live_pricing import UniswapV3LivePricing
from tradeexecutor.ethereum.eth_valuation import EthereumPoolRevaluator
from tradeexecutor.ethereum.aave_v3.aave_v3_routing import AaveV3Routing
assert routing_id.router_name == "aave-v3"
assert routing_id.lending_protocol_slug == "aave_v3"
assert len(strategy_universe.data_universe.chains) == 1
assert len(strategy_universe.reserve_assets) == 1
chain_id = strategy_universe.get_single_chain()
chain_slug = chain_id.get_slug()
reserve_asset = strategy_universe.get_reserve_asset()
assert chain_slug in AAVE_V3_DEPLOYMENTS, f"Chain {chain_slug} not supported for Aave v3"
aave_v3_deployment = fetch_aave_deployment(
web3,
pool_address=AAVE_V3_DEPLOYMENTS[chain_slug]["pool"],
data_provider_address=AAVE_V3_DEPLOYMENTS[chain_slug]["data_provider"],
oracle_address=AAVE_V3_DEPLOYMENTS[chain_slug]["oracle"],
)
address_map = {
"aave_v3_pool": aave_v3_deployment.pool.address,
"aave_v3_data_provider": aave_v3_deployment.data_provider.address,
"aave_v3_oracle": aave_v3_deployment.oracle.address,
}
routing_model = AaveV3Routing(
address_map=address_map,
chain_id=chain_id,
reserve_token_address=reserve_asset.address,
allowed_intermediary_pairs={},
)
pricing_model = UniswapV3LivePricing(
web3,
strategy_universe.data_universe.pairs,
routing_model,
)
valuation_model = EthereumPoolRevaluator(pricing_model)
return ProtocolRoutingConfig(
routing_id=routing_id,
routing_model=routing_model,
pricing_model=pricing_model,
valuation_model=valuation_model,
)
[docs]class EthereumPairConfigurator(PairConfigurator):
"""Set up routes for EVM trading pairs.
Supported protocols
- 1delta
- Uniswap v2 likes
- Uniswap v3 likes
- Aave v3
"""
[docs] def __init__(
self,
web3: Web3,
strategy_universe: TradingStrategyUniverse | None,
):
"""Initialise pair configuration.
:param web3:
Web3 connection to the active blockchain.
:param strategy_universe:
The initial strategy universe.
TODO: Currently only reserve currency, exchange and pair data is used.
Candle data is discarded.
"""
assert isinstance(web3, Web3)
assert isinstance(strategy_universe, TradingStrategyUniverse)
self.web3 = web3
super().__init__(strategy_universe)
[docs] def get_supported_routers(self) -> Set[ProtocolRoutingId]:
return default_supported_routers(self.strategy_universe)
[docs] def create_config(self, routing_id: ProtocolRoutingId) -> ProtocolRoutingConfig:
if routing_id.router_name == "1delta":
return create_1delta_adapter(self.web3, self.strategy_universe, routing_id)
elif routing_id.router_name == "uniswap-v2":
return create_uniswap_v2_adapter(self.web3, self.strategy_universe, routing_id)
elif routing_id.router_name == "uniswap-v3":
return create_uniswap_v3_adapter(self.web3, self.strategy_universe, routing_id)
elif routing_id.router_name == "aave-v3":
return create_aave_v3_adapter(self.web3, self.strategy_universe, routing_id)
else:
raise NotImplementedError(f"Cannot route exchange {routing_id}")
[docs] def match_router(self, pair: TradingPairIdentifier) -> ProtocolRoutingId:
return default_match_router(self.strategy_universe, pair)
UNISWAP_V3_INTERMEDIATE = {
ChainId.ethereum: {
# Route WETH through WETH-USDC 5 BPS
# https://tradingstrategy.ai/trading-view/ethereum/uniswap-v3/eth-usdc-fee-5
"0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2": "0x88e6a0c2ddd26feeb64f039a2c41296fcb3f5640",
},
ChainId.base: {
# Route WETH through WETH-USDC 5 BPS
# https://coinmarketcap.com/dexscan/base/0xd0b53d9277642d899df5c87a3966a349a798f224/
"0x4200000000000000000000000000000000000006": "0xd0b53d9277642d899df5c87a3966a349a798f224",
}
}