Source code for tradeexecutor.state.sync
""""Store information about caught up chain state.
- Treasury understanding is needed in order to reflect on-chain balance changes to the strategy execution
- Most treasury changes are deposits and redemptions
- Interest rate events also change on-chain treasury balances
- See :py:mod:`tradeexecutor.strategy.sync_model` how to on-chain treasuty
"""
import datetime
from dataclasses import dataclass, field
from typing import Optional, List, Iterable, Dict
from dataclasses_json import dataclass_json
from tradeexecutor.state.interest_distribution import InterestDistributionOperation
from tradeexecutor.state.types import USDollarAmount, BlockNumber
from tradingstrategy.chain import ChainId
from tradeexecutor.state.balance_update import BalanceUpdate, BalanceUpdateCause, BalanceUpdatePositionType
[docs]@dataclass_json
@dataclass
class Deployment:
"""Information for the strategy deployment.
- Capture information about the vault deployment in the strategy's persistent state
- This information can be later used to look up information (e.g deposit transactions)
- This information can be later used to look up verify data
"""
#: Which chain we are deployed
chain_id: Optional[ChainId] = None
#: Vault smart contract address
#:
#: For hot wallet execution, the address of the hot wallet
address: Optional[str] = None
#: When the vault was deployed
#:
#: Not available for hot wallet based strategies
block_number: Optional[int] = None
#: When the vault was deployed
#:
#: Not available for hot wallet based strategies
tx_hash: Optional[str] = None
#: UTC block timestamp of the vault deployment tx
#:
#: Not available for hot wallet based strategies
block_mined_at: Optional[datetime.datetime] = None
#: Vault name
#:
#: Enzyme vault name - same as vault toke name
vault_token_name: Optional[str] = None
#: Vault token symbol
#:
#: Enzyme vault name - same as vault toke name
vault_token_symbol: Optional[str] = None
#: When the initialisation was complete
#:
initialised_at: Optional[datetime.datetime] = None
def __repr__(self):
return f"<Deployment chain:{self.chain_id.name} address:{self.address} name:{self.vault_token_name} token:{self.vault_token_symbol}>"
[docs]@dataclass_json
@dataclass
class BalanceEventRef:
"""Register the balance event in the treasury model.
Balance updates can happen for
- Treasury
- Open trading positions
We maintain a list of balance update references across all positions using :py:class:`BalanceEventRef`.
This allows us quickly to calculate net inflow/outflow.
"""
#: Balance event id we are referring to
balance_event_id: int
#: When this update was made.
#:
#: Strategy cycle timestamp when the deposit/redemption was included
#: in the strategy treasury.
#:
#: It might be outside the cycle frequency if treasuries were processed
#: in a cron job outside the cycle for slow moving strategies.
#:
strategy_cycle_included_at: Optional[datetime.datetime]
#: Cause of the event
cause: BalanceUpdateCause
#: Reserve currency or underlying position
position_type: BalanceUpdatePositionType
#: Which trading positions were affected
position_id: Optional[int]
#: How much this deposit/redemption was worth
#:
#: Used for deposit/redemption inflow/outflow calculation.
#: This is the asset value from our internal price keeping at the time of the event.
#:
usd_value: Optional[USDollarAmount]
[docs] @staticmethod
def from_balance_update_event(evt: BalanceUpdate) -> "BalanceEventRef":
"""Create a reference to a balance update event stored elsewhere in the state."""
ref = BalanceEventRef(
balance_event_id=evt.balance_update_id,
strategy_cycle_included_at=evt.strategy_cycle_included_at,
cause=evt.cause,
position_type=evt.position_type,
position_id=evt.position_id,
usd_value=evt.usd_value,
)
return ref
[docs]@dataclass_json
@dataclass
class Treasury:
"""State of syncind deposits and redemptions from the chain.
"""
#: Wall clock time. timestamp for which we run the last sync
#:
#: Wall clock time, at the beginning on the sync cycle.
last_updated_at: Optional[datetime.datetime] = None
#: The strategy cycle timestamp for which we run the last sync
#:
#: Wall clock time, at the beginning on the sync cycle.
last_cycle_at: Optional[datetime.datetime] = None
#: What is the last processed block for deposit
#:
#: 0 = not scanned yet
last_block_scanned: Optional[int] = None
#: List of refences to all balance update events.
#:
#: The actual balance update content is stored on the position itself.
balance_update_refs: List[BalanceEventRef] = field(default_factory=list)
#: How much pending redemptions we have?
#:
#: For Lagoon based vaults, we need to sell assets to satisfy redemptions on the next cycle.
#:
pending_redemptions: Optional[USDollarAmount] = None
def __repr__(self):
return f"<Treasury updated:{self.last_updated_at} cycle:{self.last_cycle_at} block scanned:{self.last_block_scanned or 0:,} refs:{len(self.balance_update_refs)}>"
[docs]@dataclass_json
@dataclass
class Accounting:
"""State of accounting corrections.
"""
#: Wall clock time. timestamp for which we run the last sync
#:
#: Wall clock time, at the beginning on the sync cycle.
last_updated_at: Optional[datetime.datetime] = None
#: What is the last processed block for deposit
#:
#: 0 = not scanned yet
last_block_scanned: Optional[int] = None
#: List of refernces to all balance update events.
#:
#: The actual balance update content is stored on the position itself.
balance_update_refs: List[BalanceEventRef] = field(default_factory=list)
[docs]@dataclass_json
@dataclass(slots=True)
class InterestSync:
"""Track the interest sync for on-chain rebase tokens.
"""
#: When did we perform the last interest sync all all assets
#:
last_sync_at: datetime.datetime | None = None
#: Block number when we synced the portfolio
#:
#: Backtesting does not use block numbers and has this always set to ``None``.
#:
last_sync_block: BlockNumber | None = None
#: Last operation we did.
#:
#: Data is not needed. Stored only for diagnostics purposes.
#: Always overwritten in the next sync.
#:
last_distribution: InterestDistributionOperation | None = None
[docs]@dataclass_json
@dataclass
class Sync:
"""On-chain sync state.
- Store persistent information about the vault on transactions we have synced,
so that the strategy knows its available capital
- Updated before the strategy execution step
"""
deployment: Deployment = field(default_factory=Deployment)
treasury: Treasury = field(default_factory=Treasury)
accounting: Accounting = field(default_factory=Accounting)
interest: InterestSync = field(default_factory=InterestSync)
[docs] def is_initialised(self) -> bool:
"""Have we scanned the initial deployment event for the sync model."""
return self.deployment.block_number is not None