Source code for tradeexecutor.ethereum.eth_valuation

"""Value model based on Uniswap v3 market price.

Value positions based on their "dump" price on Uniswap,
assuming we get the worst possible single trade execution.
"""
import datetime
import logging

from tradeexecutor.ethereum.eth_pricing_model import EthereumPricingModel
from tradeexecutor.state.position import TradingPosition
from tradeexecutor.state.valuation import ValuationUpdate
from tradeexecutor.strategy.valuation import ValuationModel


logger = logging.getLogger(__name__)



[docs]class EthereumPoolRevaluator(ValuationModel): """Re-value spot assets based on their on-chain price. Does directly JSON-RPC call to get the latest price in the Uniswap pools. Only uses direct route - mostly useful for testing, may not give a realistic price in real world with multiple order routing options. .. warning :: This valuation metohd always uses the latest price. It cannot be used for backtesting. """
[docs] def __init__(self, pricing_model: EthereumPricingModel): self.pricing_model = pricing_model
[docs] def __call__( self, ts: datetime.datetime, position: TradingPosition, ) -> ValuationUpdate: """ :param ts: When to revalue. Used in backesting. Live strategies may ignore. :param position: Open position :return: (revaluation date, price) tuple. Note that revaluation date may differ from the wantead timestamp if there is no data available. """ assert isinstance(ts, datetime.datetime) pair = position.pair if position.is_long(): quantity = position.get_quantity() # Cannot do pricing for zero quantity if quantity == 0: # We can actually get zero quantity because if we buy some scam token # that sets out wallet balance to zero. # In these case, do not crash here. # However this is an exceptional case, so let's be verbose about it. logger.warning(f"Trying to value position with zero quantity: {position}, {ts}, {self.__class__.__name__}") assert quantity >= 0, f"Trying to value position with non-positive quantity: {position}, {ts}, {self.__class__.__name__}" old_price = position.last_token_price old_value = position.get_value() if quantity != 0: price_structure = self.pricing_model.get_sell_price(ts, pair, quantity) new_price = price_structure.price new_value = position.revalue_base_asset(ts, float(new_price)) else: # Frozen positions may get quantity as zero? new_price = 0 new_value = 0 evt = ValuationUpdate( created_at=ts, position_id=position.position_id, valued_at=ts, old_value=old_value, new_value=new_value, old_price=old_price, new_price=new_price, quantity=quantity, ) elif position.is_credit_supply(): # The position should be valued in `sync_interests` # so we just return the latest synced data here. loan = position.loan new_price = loan.collateral.last_usd_price valued_at = loan.collateral.last_pricing_at new_value = loan.get_net_asset_value() block_number = loan.collateral_interest.last_updated_block_number evt = ValuationUpdate( created_at=ts, position_id=position.position_id, valued_at=valued_at, new_price=new_price, new_value=new_value, block_number=block_number, ) else: raise ValueError(f"Unknown position kind: {pair.kind}") position.valuation_updates.append(evt) return evt