Stage 07.4.3.2 — price polling, event bus and UI throttling
This commit is contained in:
@@ -9,6 +9,8 @@ from typing import Callable
|
||||
from aiogram import Bot
|
||||
from aiogram.exceptions import TelegramBadRequest, TelegramRetryAfter
|
||||
|
||||
from src.core.event_bus import EventBus
|
||||
from src.integrations.exchange.market_data_runner import MarketDataRunner
|
||||
from src.trading.auto.service import AutoTradeService
|
||||
|
||||
|
||||
@@ -20,9 +22,16 @@ class AutoTradeRunner:
|
||||
_render_text: Callable[[], str] | None = None
|
||||
_render_markup: Callable[[], object] | None = None
|
||||
_current_screen: str | None = None
|
||||
_interval_seconds = 15
|
||||
|
||||
# анализ стратегии — часто
|
||||
_analysis_interval_seconds = 5
|
||||
|
||||
# Telegram UI — редко
|
||||
_ui_interval_seconds = 60
|
||||
|
||||
_last_text: str | None = None
|
||||
_last_ui_refresh_at: float = 0.0
|
||||
_last_event_version: int = 0
|
||||
_retry_after_until: float = 0.0
|
||||
|
||||
@classmethod
|
||||
@@ -89,6 +98,13 @@ class AutoTradeRunner:
|
||||
|
||||
@classmethod
|
||||
def start(cls) -> None:
|
||||
service = AutoTradeService()
|
||||
|
||||
MarketDataRunner.start(
|
||||
symbol_provider=lambda: service.get_state().symbol,
|
||||
interval_seconds=1,
|
||||
)
|
||||
|
||||
if cls._task is not None and not cls._task.done():
|
||||
return
|
||||
|
||||
@@ -96,6 +112,8 @@ class AutoTradeRunner:
|
||||
|
||||
@classmethod
|
||||
def stop(cls) -> None:
|
||||
MarketDataRunner.stop()
|
||||
|
||||
if cls._task is None:
|
||||
return
|
||||
|
||||
@@ -111,16 +129,29 @@ class AutoTradeRunner:
|
||||
|
||||
if state.status == "OFF":
|
||||
cls._task = None
|
||||
MarketDataRunner.stop()
|
||||
break
|
||||
|
||||
service.run_cycle()
|
||||
|
||||
await cls._refresh_screen()
|
||||
await asyncio.sleep(cls._interval_seconds)
|
||||
current_event_version = EventBus.version()
|
||||
has_important_event = current_event_version != cls._last_event_version
|
||||
|
||||
if has_important_event:
|
||||
cls._last_event_version = current_event_version
|
||||
|
||||
await cls._refresh_screen(force=has_important_event)
|
||||
|
||||
await asyncio.sleep(cls._analysis_interval_seconds)
|
||||
|
||||
@classmethod
|
||||
async def _refresh_screen(cls) -> None:
|
||||
if time.monotonic() < cls._retry_after_until:
|
||||
async def _refresh_screen(cls, *, force: bool = False) -> None:
|
||||
now = time.monotonic()
|
||||
|
||||
if now < cls._retry_after_until:
|
||||
return
|
||||
|
||||
if not force and now - cls._last_ui_refresh_at < cls._ui_interval_seconds:
|
||||
return
|
||||
|
||||
if not all(
|
||||
@@ -147,6 +178,7 @@ class AutoTradeRunner:
|
||||
reply_markup=cls._render_markup(),
|
||||
)
|
||||
cls._last_text = text
|
||||
cls._last_ui_refresh_at = now
|
||||
|
||||
except TelegramRetryAfter as exc:
|
||||
cls._retry_after_until = time.monotonic() + exc.retry_after + 5
|
||||
@@ -156,6 +188,7 @@ class AutoTradeRunner:
|
||||
|
||||
if "message is not modified" in error_text:
|
||||
cls._last_text = text
|
||||
cls._last_ui_refresh_at = now
|
||||
return
|
||||
|
||||
if "message to edit not found" in error_text:
|
||||
|
||||
@@ -10,6 +10,7 @@ from src.trading.auto.state import AutoTradeState
|
||||
from src.trading.journal.service import JournalService
|
||||
from src.trading.strategies.base import BaseStrategy, StrategyContext
|
||||
from src.trading.strategies.registry import StrategyRegistry
|
||||
from src.core.event_bus import EventBus
|
||||
|
||||
|
||||
class AutoTradeService:
|
||||
@@ -69,18 +70,30 @@ class AutoTradeService:
|
||||
# запустить активную торговлю
|
||||
def start(self) -> tuple[AutoTradeState, str]:
|
||||
state = self.get_state()
|
||||
previous_status = state.status
|
||||
|
||||
if state.status == "RUNNING":
|
||||
self.start_loop()
|
||||
return state, "Автоторговля уже активна."
|
||||
|
||||
if state.status == "OBSERVING":
|
||||
state.status = "RUNNING"
|
||||
self.start_loop()
|
||||
EventBus.emit(
|
||||
"auto_status_changed",
|
||||
{
|
||||
"previous_status": previous_status,
|
||||
"status": state.status,
|
||||
},
|
||||
)
|
||||
return state, "Автоторговля активирована."
|
||||
|
||||
state.status = "RUNNING"
|
||||
self.start_loop()
|
||||
EventBus.emit(
|
||||
"auto_status_changed",
|
||||
{
|
||||
"previous_status": previous_status,
|
||||
"status": state.status,
|
||||
},
|
||||
)
|
||||
return state, "Автоторговля запущена."
|
||||
|
||||
# включить режим наблюдения
|
||||
@@ -89,11 +102,17 @@ class AutoTradeService:
|
||||
previous_status = state.status
|
||||
|
||||
if previous_status == "OBSERVING":
|
||||
self.start_loop()
|
||||
return state, "Режим наблюдения уже включён."
|
||||
|
||||
state.status = "OBSERVING"
|
||||
self.start_loop()
|
||||
|
||||
EventBus.emit(
|
||||
"auto_status_changed",
|
||||
{
|
||||
"previous_status": previous_status,
|
||||
"status": state.status,
|
||||
},
|
||||
)
|
||||
|
||||
if previous_status == "OFF":
|
||||
return state, "Включён режим наблюдения."
|
||||
@@ -103,6 +122,7 @@ class AutoTradeService:
|
||||
# полностью выключить автоторговлю
|
||||
def stop(self) -> tuple[AutoTradeState, str]:
|
||||
state = self.get_state()
|
||||
previous_status = state.status
|
||||
|
||||
if state.status == "OFF":
|
||||
self.stop_loop()
|
||||
@@ -110,6 +130,15 @@ class AutoTradeService:
|
||||
|
||||
state.status = "OFF"
|
||||
self.stop_loop()
|
||||
|
||||
EventBus.emit(
|
||||
"auto_status_changed",
|
||||
{
|
||||
"previous_status": previous_status,
|
||||
"status": state.status,
|
||||
},
|
||||
)
|
||||
|
||||
return state, "Автоторговля выключена."
|
||||
|
||||
# установить инструмент
|
||||
@@ -132,7 +161,7 @@ class AutoTradeService:
|
||||
state.risk_percent = risk_percent
|
||||
return state
|
||||
|
||||
# установить плечо
|
||||
# установить плечо
|
||||
def set_leverage(self, leverage: float) -> AutoTradeState:
|
||||
state = self.get_state()
|
||||
state.leverage = leverage
|
||||
@@ -285,6 +314,9 @@ class AutoTradeService:
|
||||
reason: str,
|
||||
confidence: float,
|
||||
) -> None:
|
||||
previous_signal = state.last_signal
|
||||
previous_decision_status = state.decision_status
|
||||
|
||||
state.last_signal = signal
|
||||
state.last_signal_repeat_count = self._same_signal_count
|
||||
state.last_signal_confidence = confidence
|
||||
@@ -296,6 +328,29 @@ class AutoTradeService:
|
||||
confidence=confidence,
|
||||
)
|
||||
|
||||
if previous_signal != state.last_signal:
|
||||
EventBus.emit(
|
||||
"auto_signal_changed",
|
||||
{
|
||||
"previous_signal": previous_signal,
|
||||
"signal": state.last_signal,
|
||||
"repeat_count": state.last_signal_repeat_count,
|
||||
"confidence": state.last_signal_confidence,
|
||||
},
|
||||
)
|
||||
|
||||
if previous_decision_status != state.decision_status:
|
||||
EventBus.emit(
|
||||
"auto_decision_changed",
|
||||
{
|
||||
"previous_decision_status": previous_decision_status,
|
||||
"decision_status": state.decision_status,
|
||||
"signal": state.last_signal,
|
||||
"repeat_count": state.last_signal_repeat_count,
|
||||
"confidence": state.last_signal_confidence,
|
||||
},
|
||||
)
|
||||
|
||||
# записать одиночный сигнал в журнал
|
||||
def _log_signal_event(
|
||||
self,
|
||||
|
||||
Reference in New Issue
Block a user