2
0
forked from kodorvan/stcs

13 Commits

16 changed files with 361 additions and 179 deletions

3
.gitignore vendored
View File

@@ -210,3 +210,6 @@ cython_debug/
marimo/_static/ marimo/_static/
marimo/_lsp/ marimo/_lsp/
__marimo__/ __marimo__/
stcs_venv
venv

View File

@@ -1,6 +1,11 @@
Crypto Trading Telegram Bot # Crypto Trading Telegram Bot by [KODORVAN](https://git.svoboda.works/kodorvan)
Этот бот — автоматизированный торговый помощник для работы с криптовалютной биржей Bybit на основе стратегии мартингейла. Он позволяет торговать бессрочными контрактами с управлением рисками, тейк-профитами, стоп-лоссами и кредитным плечом. Автоматизированный торговый помощник для работы с криптовалютной биржей Bybit на основе стратегии мартингейла.<br>
Он позволяет торговать бессрочными контрактами с управлением рисками, тейк-профитами, стоп-лоссами и кредитным плечом.
**Разработано командой [КОДОРВАНЬ](https://git.svoboda.works/kodorvan)**<br>
_Мы окажем полное содействие и поддержку, пишите в телеграм: https://t.me/kodorvan_
## Основные возможности ## Основные возможности
@@ -59,7 +64,12 @@ nvim .env
alembic upgrade head alembic upgrade head
``` ```
5. Запустите бота: 6. Убедитесь в том, что установлен redis и открыт порт 6789
```bash
sudo ufw allow 6789
```
7. Запустите бота:
```bash ```bash
python run.py python run.py

View File

@@ -84,7 +84,7 @@ path_separator = os
# database URL. This is consumed by the user-maintained env.py script only. # database URL. This is consumed by the user-maintained env.py script only.
# other means of configuring database URLs may be customized within the env.py # other means of configuring database URLs may be customized within the env.py
# file. # file.
sqlalchemy.url = sqlite+aiosqlite:///./database/dbs/stcs.db sqlalchemy.url = sqlite+aiosqlite:///./database/stcs.db
[post_write_hooks] [post_write_hooks]

View File

@@ -0,0 +1,32 @@
"""initial
Revision ID: f6e7eb3f25c0
Revises:
Create Date: 2025-11-12 22:53:02.189445
"""
from typing import Sequence, Union
from alembic import op
import sqlalchemy as sa
# revision identifiers, used by Alembic.
revision: str = 'f6e7eb3f25c0'
down_revision: Union[str, Sequence[str], None] = None
branch_labels: Union[str, Sequence[str], None] = None
depends_on: Union[str, Sequence[str], None] = None
def upgrade() -> None:
"""Upgrade schema."""
# ### commands auto generated by Alembic - please adjust! ###
pass
# ### end Alembic commands ###
def downgrade() -> None:
"""Downgrade schema."""
# ### commands auto generated by Alembic - please adjust! ###
pass
# ### end Alembic commands ###

View File

@@ -38,7 +38,7 @@ async def get_active_positions(tg_id: int) -> list | None:
return None return None
async def get_active_positions_by_symbol(tg_id: int, symbol: str) -> dict | None: async def get_active_positions_by_symbol(tg_id: int, symbol: str):
""" """
Get active positions for a user by symbol Get active positions for a user by symbol
""" """
@@ -62,8 +62,12 @@ async def get_active_positions_by_symbol(tg_id: int, symbol: str) -> dict | None
) )
return None return None
except Exception as e: except Exception as e:
logger.error("Error getting active positions for user %s: %s", tg_id, e) errors = str(e)
return None if errors.startswith("Permission denied, please check your API key permissions"):
return "Invalid API key permissions"
else:
logger.error("Error getting active positions for user %s: %s", tg_id, e)
return None
async def get_active_orders(tg_id: int) -> list | None: async def get_active_orders(tg_id: int) -> list | None:

View File

@@ -109,6 +109,7 @@ async def start_trading_cycle(
"The number of contracts exceeds maximum limit allowed", "The number of contracts exceeds maximum limit allowed",
"The number of contracts exceeds minimum limit allowed", "The number of contracts exceeds minimum limit allowed",
"Order placement failed as your position may exceed the max", "Order placement failed as your position may exceed the max",
"Permission denied, please check your API key permissions"
} }
else None else None
) )
@@ -232,9 +233,6 @@ async def trading_cycle(
) )
current_step += 1 current_step += 1
if max_bets_in_series < current_step:
return "Max bets in series"
await set_margin_mode(tg_id=tg_id, margin_mode=margin_type) await set_margin_mode(tg_id=tg_id, margin_mode=margin_type)
await set_leverage( await set_leverage(
tg_id=tg_id, tg_id=tg_id,
@@ -371,6 +369,7 @@ async def open_positions(
"The number of contracts exceeds maximum limit allowed": "The number of contracts exceeds maximum limit allowed", "The number of contracts exceeds maximum limit allowed": "The number of contracts exceeds maximum limit allowed",
"The number of contracts exceeds minimum limit allowed": "The number of contracts exceeds minimum limit allowed", "The number of contracts exceeds minimum limit allowed": "The number of contracts exceeds minimum limit allowed",
"Order placement failed as your position may exceed the max": "Order placement failed as your position may exceed the max", "Order placement failed as your position may exceed the max": "Order placement failed as your position may exceed the max",
"Permission denied, please check your API key permissions": "Permission denied, please check your API key permissions"
} }
for key, msg in known_errors.items(): for key, msg in known_errors.items():
if key in error_text: if key in error_text:

View File

@@ -37,7 +37,7 @@ async def user_profile_bybit(tg_id: int, message: Message, state: FSMContext) ->
) )
else: else:
await message.answer( await message.answer(
text="Ошибка при подключении к платформе. Проверьте ключи и повторите попытку.", text="Ошибка при подключении к платформе. Проверьте корректность и разрешения API ключа и добавьте повторно.",
reply_markup=kbi.connect_the_platform, reply_markup=kbi.connect_the_platform,
) )
logger.error("Error processing user profile for user %s", tg_id) logger.error("Error processing user profile for user %s", tg_id)

View File

@@ -1,10 +1,9 @@
import logging.config import logging.config
import math import math
# import json import json
import app.telegram.keyboards.inline as kbi import app.telegram.keyboards.inline as kbi
import database.request as rq import database.request as rq
from app.bybit.get_functions.get_instruments_info import get_instruments_info from app.bybit.get_functions.get_instruments_info import get_instruments_info
from app.bybit.get_functions.get_positions import get_active_positions_by_symbol
from app.bybit.logger_bybit.logger_bybit import LOGGING_CONFIG from app.bybit.logger_bybit.logger_bybit import LOGGING_CONFIG
from app.bybit.open_positions import trading_cycle, trading_cycle_profit from app.bybit.open_positions import trading_cycle, trading_cycle_profit
from app.bybit.set_functions.set_tp_sl import set_tp_sl_for_position from app.bybit.set_functions.set_tp_sl import set_tp_sl_for_position
@@ -123,7 +122,6 @@ class TelegramMessageHandler:
current_step = user_deals_data.current_step current_step = user_deals_data.current_step
order_quantity = user_deals_data.order_quantity order_quantity = user_deals_data.order_quantity
pnl_series = user_deals_data.pnl_series pnl_series = user_deals_data.pnl_series
margin_type = user_deals_data.margin_type
take_profit_percent = user_deals_data.take_profit_percent take_profit_percent = user_deals_data.take_profit_percent
stop_loss_percent = user_deals_data.stop_loss_percent stop_loss_percent = user_deals_data.stop_loss_percent
leverage = safe_float(user_deals_data.leverage) leverage = safe_float(user_deals_data.leverage)
@@ -167,24 +165,14 @@ class TelegramMessageHandler:
if commission_place == "Commission_for_tp": if commission_place == "Commission_for_tp":
total_commission = safe_float(total_fee) / qty_formatted total_commission = safe_float(total_fee) / qty_formatted
if margin_type == "ISOLATED_MARGIN": if side == "Buy":
if side == "Buy": take_profit_price = safe_float(exec_price) * (
take_profit_price = safe_float(exec_price) * ( 1 + take_profit_percent / 100) + total_commission
1 + take_profit_percent / 100) + total_commission stop_loss_price = safe_float(exec_price) * (1 - stop_loss_percent / 100)
stop_loss_price = None
else:
take_profit_price = safe_float(exec_price) * (
1 - take_profit_percent / 100) - total_commission
stop_loss_price = None
else: else:
if side == "Buy": take_profit_price = safe_float(exec_price) * (
take_profit_price = safe_float(exec_price) * ( 1 - take_profit_percent / 100) - total_commission
1 + take_profit_percent / 100) + total_commission stop_loss_price = safe_float(exec_price) * (1 + stop_loss_percent / 100)
stop_loss_price = safe_float(exec_price) * (1 - stop_loss_percent / 100)
else:
take_profit_price = safe_float(exec_price) * (
1 - take_profit_percent / 100) - total_commission
stop_loss_price = safe_float(exec_price) * (1 + stop_loss_percent / 100)
ress = await set_tp_sl_for_position(tg_id=tg_id, ress = await set_tp_sl_for_position(tg_id=tg_id,
symbol=symbol, symbol=symbol,
@@ -193,27 +181,11 @@ class TelegramMessageHandler:
position_idx=0) position_idx=0)
if ress or ress == "not modified": if ress or ress == "not modified":
take_profit_truncated = await truncate_float(take_profit_price, 6) take_profit_truncated = await truncate_float(take_profit_price, 6)
stop_loss_truncated = await truncate_float(stop_loss_price, 6)
text += (f"Движение: {side_rus}\n" text += (f"Движение: {side_rus}\n"
f"Тейк-профит: {take_profit_truncated}\n" f"Тейк-профит: {take_profit_truncated}\n"
f"Стоп-лосс: {stop_loss_truncated}\n"
) )
if stop_loss_price is not None:
stop_loss_truncated = await truncate_float(stop_loss_price, 6)
else:
stop_loss_truncated = None
if stop_loss_truncated is not None:
text += f"Стоп-лосс: {stop_loss_truncated}\n"
else:
deals = await get_active_positions_by_symbol(
tg_id=tg_id, symbol=symbol
)
position = next((d for d in deals if d.get("symbol") == symbol), None)
if position:
liq_price = position.get("liqPrice", 0)
text += f"Цена ликвидации: {liq_price}\n"
else: else:
text += (f"Движение: {side_rus}\n" text += (f"Движение: {side_rus}\n"
"Не удалось установить ТП и СЛ\n") "Не удалось установить ТП и СЛ\n")
@@ -258,7 +230,6 @@ class TelegramMessageHandler:
pass pass
else: else:
errors = { errors = {
"Max bets in series": "❗️ Максимальное количество сделок в серии достигнуто",
"Risk is too high for this trade": "❗️ Риск сделки слишком высок для продолжения", "Risk is too high for this trade": "❗️ Риск сделки слишком высок для продолжения",
"ab not enough for new order": "❗️ Недостаточно средств для продолжения торговли", "ab not enough for new order": "❗️ Недостаточно средств для продолжения торговли",
"InvalidRequestError": "❗️ Недостаточно средств для размещения нового ордера с заданным количеством и плечом.", "InvalidRequestError": "❗️ Недостаточно средств для размещения нового ордера с заданным количеством и плечом.",
@@ -285,49 +256,105 @@ class TelegramMessageHandler:
) )
elif stop_order_type == "StopLoss" or exec_type == "BustTrade": elif stop_order_type == "StopLoss" or exec_type == "BustTrade":
open_order_text = "\n❗️ Открываю новую сделку с увеличенной ставкой.\n" current_step = user_deals_data.current_step
await self.telegram_bot.send_message( max_bets_in_series = user_deals_data.max_bets_in_series
chat_id=tg_id, text=open_order_text current_step += 1
)
if side == "Buy": if max_bets_in_series < current_step:
r_side = "Sell" text_series = ("\n❗️ Максимальное количество сделок в серии достигнуто.\n"
else: "📈 Начинаю новую серию с базовой ставки\n")
r_side = "Buy" await self.telegram_bot.send_message(
chat_id=tg_id, text=text_series
res = await trading_cycle(
tg_id=tg_id, symbol=symbol, side=r_side
)
if res == "OK":
pass
else:
errors = {
"Max bets in series": "❗️ Максимальное количество сделок в серии достигнуто",
"Risk is too high for this trade": "❗️ Риск сделки слишком высок для продолжения",
"ab not enough for new order": "❗️ Недостаточно средств для продолжения торговли",
"InvalidRequestError": "❗️ Недостаточно средств для размещения нового ордера с заданным количеством и плечом.",
"The number of contracts exceeds maximum limit allowed": "❗️ Превышен максимальный лимит ставки",
"Order placement failed as your position may exceed the max": "❗️ Превышен максимальный лимит ставки с текущим плечом",
}
error_text = errors.get(
res, "❗️ Не удалось открыть новую сделку"
)
await rq.set_auto_trading(
tg_id=tg_id, symbol=symbol, auto_trading=False
) )
if side == "Buy":
r_side = "Sell"
else:
r_side = "Buy"
await rq.set_last_side_by_symbol(
tg_id=tg_id, symbol=symbol, last_side=r_side)
await rq.set_total_fee_user_auto_trading( await rq.set_total_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, total_fee=0 tg_id=tg_id, symbol=symbol, total_fee=0
) )
await rq.set_fee_user_auto_trading( await rq.set_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, fee=0 tg_id=tg_id, symbol=symbol, fee=0
) )
await self.telegram_bot.send_message( await rq.set_pnl_series_by_symbol(tg_id=tg_id, symbol=symbol, pnl_series=0)
chat_id=tg_id,
text=error_text, res = await trading_cycle_profit(
reply_markup=kbi.profile_bybit, tg_id=tg_id, symbol=symbol, side=r_side
) )
if res == "OK":
pass
else:
errors = {
"Risk is too high for this trade": "❗️ Риск сделки слишком высок для продолжения",
"ab not enough for new order": "❗️ Недостаточно средств для продолжения торговли",
"InvalidRequestError": "❗️ Недостаточно средств для размещения нового ордера с заданным количеством и плечом.",
"The number of contracts exceeds maximum limit allowed": "❗️ Превышен максимальный лимит ставки",
"Order placement failed as your position may exceed the max": "❗️ Превышен максимальный лимит ставки",
}
error_text = errors.get(
res, "❗️ Не удалось открыть новую сделку"
)
await rq.set_auto_trading(
tg_id=tg_id, symbol=symbol, auto_trading=False
)
await rq.set_total_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, total_fee=0
)
await rq.set_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, fee=0
)
await self.telegram_bot.send_message(
chat_id=tg_id,
text=error_text,
reply_markup=kbi.profile_bybit,
)
else:
open_order_text = "\n❗️ Открываю новую сделку с увеличенной ставкой.\n"
await self.telegram_bot.send_message(
chat_id=tg_id, text=open_order_text
)
if side == "Buy":
r_side = "Sell"
else:
r_side = "Buy"
res = await trading_cycle(
tg_id=tg_id, symbol=symbol, side=r_side
)
if res == "OK":
pass
else:
errors = {
"Risk is too high for this trade": "❗️ Риск сделки слишком высок для продолжения",
"ab not enough for new order": "❗️ Недостаточно средств для продолжения торговли",
"InvalidRequestError": "❗️ Недостаточно средств для размещения нового ордера с заданным количеством и плечом.",
"The number of contracts exceeds maximum limit allowed": "❗️ Превышен максимальный лимит ставки",
"Order placement failed as your position may exceed the max": "❗️ Превышен максимальный лимит ставки с текущим плечом",
}
error_text = errors.get(
res, "❗️ Не удалось открыть новую сделку"
)
await rq.set_auto_trading(
tg_id=tg_id, symbol=symbol, auto_trading=False
)
await rq.set_total_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, total_fee=0
)
await rq.set_fee_user_auto_trading(
tg_id=tg_id, symbol=symbol, fee=0
)
await self.telegram_bot.send_message(
chat_id=tg_id,
text=error_text,
reply_markup=kbi.profile_bybit,
)
elif create_type == "CreateByClosing": elif create_type == "CreateByClosing":
await self.telegram_bot.send_message( await self.telegram_bot.send_message(
chat_id=tg_id, chat_id=tg_id,
@@ -346,5 +373,7 @@ class TelegramMessageHandler:
) )
logger.info("Stop trading for symbol: %s, create_type: %s, stop_order_type: %s: %s", logger.info("Stop trading for symbol: %s, create_type: %s, stop_order_type: %s: %s",
symbol, create_type, stop_order_type, tg_id) symbol, create_type, stop_order_type, tg_id)
else:
logger.info("Execution update: %s", json.dumps(message))
except Exception as e: except Exception as e:
logger.error("Error in telegram_message_handler: %s", e, exc_info=True) logger.error("Error in telegram_message_handler: %s", e, exc_info=True)

View File

@@ -1,4 +1,5 @@
import asyncio import asyncio
from collections import deque
import logging.config import logging.config
from pybit.unified_trading import WebSocket from pybit.unified_trading import WebSocket
@@ -11,104 +12,178 @@ logging.config.dictConfig(LOGGING_CONFIG)
logger = logging.getLogger("web_socket") logger = logging.getLogger("web_socket")
class CustomWebSocket(WebSocket):
"""Custom WebSocket wrapper with enhanced error handling."""
def _on_error(self, error):
logger.error(f"WebSocket error: {error}")
return super()._on_error(error)
def _on_close(self):
logger.warning("WebSocket connection closed")
super()._on_close()
class WebSocketBot: class WebSocketBot:
""" """
Class to handle WebSocket connections and messages. Manages multiple Bybit private WebSocket connections for Telegram users.
Uses queue-based message processing to handle thread-safe async calls.
""" """
def __init__(self, telegram_bot): def __init__(self, telegram_bot):
"""Initialize the TradingBot class.""" """
Initialize WebSocketBot.
Args:
telegram_bot: Telegram bot instance for message handling
"""
self.telegram_bot = telegram_bot self.telegram_bot = telegram_bot
self.ws_private = None
self.user_messages = {}
self.user_sockets = {} self.user_sockets = {}
self.user_messages = {}
self.user_keys = {} self.user_keys = {}
self.loop = None self.loop = None
self.message_handler = TelegramMessageHandler(telegram_bot) self.message_handler = TelegramMessageHandler(telegram_bot)
self.order_queues = {} # {tg_id: deque}
self.execution_queues = {} # {tg_id: deque}
self.processing_tasks = {} # {tg_id: task}
async def run_user_check_loop(self): async def run_user_check_loop(self):
"""Run a loop to check for users and connect them to the WebSocket.""" """Main loop that continuously checks users and maintains connections."""
self.loop = asyncio.get_running_loop() self.loop = asyncio.get_running_loop()
logger.info("Starting WebSocket user check loop")
while True: while True:
users = await WebSocketBot.get_users_from_db() try:
for user in users: users = await WebSocketBot.get_users_from_db()
tg_id = user.tg_id for user in users:
api_key, api_secret = await rq.get_user_api(tg_id=tg_id) tg_id = user.tg_id
api_key, api_secret = await rq.get_user_api(tg_id=tg_id)
if not api_key or not api_secret: if not api_key or not api_secret:
continue continue
keys_stored = self.user_keys.get(tg_id) keys_stored = self.user_keys.get(tg_id)
if tg_id in self.user_sockets and keys_stored == (api_key, api_secret): socket_exists = tg_id in self.user_sockets
continue
if tg_id in self.user_sockets: if socket_exists and keys_stored == (api_key, api_secret):
self.user_sockets.clear() continue
self.user_messages.clear()
self.user_keys.clear()
logger.info(
"Closed old websocket for user %s due to key change", tg_id
)
success = await self.try_connect_user(api_key, api_secret, tg_id) if socket_exists:
if success: await self.close_user_socket(tg_id)
self.user_keys[tg_id] = (api_key, api_secret)
self.user_messages.setdefault( success = await self.try_connect_user(api_key, api_secret, tg_id)
tg_id, {"position": None, "order": None, "execution": None} if success:
) self.user_keys[tg_id] = (api_key, api_secret)
logger.info("User %s connected to WebSocket", tg_id) self.user_messages.setdefault(
else: tg_id, {"position": None, "order": None, "execution": None}
await asyncio.sleep(5) )
await self.try_connect_user(api_key, api_secret, tg_id) logger.info("User %s successfully connected", tg_id)
except Exception as e:
logger.error("Error in user check loop: %s", e)
await asyncio.sleep(10) await asyncio.sleep(10)
async def clear_user_sockets(self):
"""Clear the user_sockets and user_messages dictionaries."""
self.user_sockets.clear()
self.user_messages.clear()
self.user_keys.clear()
logger.info("Cleared user_sockets")
async def try_connect_user(self, api_key, api_secret, tg_id): async def try_connect_user(self, api_key, api_secret, tg_id):
"""Try to connect a user to the WebSocket.""" """
Create and setup WebSocket streams with thread-safe queues.
"""
try: try:
self.ws_private = WebSocket( ws = CustomWebSocket(
demo=True, demo=True,
testnet=False, testnet=False,
channel_type="private", channel_type="private",
api_key=api_key, api_key=api_key,
api_secret=api_secret, api_secret=api_secret
) )
self.user_sockets[tg_id] = self.ws_private self.user_sockets[tg_id] = ws
# Connect to the WebSocket private channel
# Handle order updates self.order_queues[tg_id] = deque()
self.ws_private.order_stream( self.execution_queues[tg_id] = deque()
lambda msg: self.loop.call_soon_threadsafe(
asyncio.create_task, self.handle_order_update(msg, tg_id) self.processing_tasks[tg_id] = asyncio.create_task(
) self._process_order_queue(tg_id)
) )
# Handle execution updates self.processing_tasks[tg_id + 1] = asyncio.create_task(
self.ws_private.execution_stream( self._process_execution_queue(tg_id)
lambda msg: self.loop.call_soon_threadsafe(
asyncio.create_task, self.handle_execution_update(msg, tg_id)
)
) )
def order_callback(msg):
self.order_queues[tg_id].append(msg)
def execution_callback(msg):
self.execution_queues[tg_id].append(msg)
ws.order_stream(order_callback)
ws.execution_stream(execution_callback)
logger.info("WebSocket streams configured for user %s", tg_id)
return True return True
except Exception as e: except Exception as e:
logger.error("Error connecting user %s: %s", tg_id, e) logger.error("Error connecting user %s: %s", tg_id, e)
self.user_sockets.pop(tg_id, None)
return False return False
async def _process_order_queue(self, tg_id):
"""Continuously process order queue for user."""
while tg_id in self.user_sockets:
try:
if self.order_queues[tg_id]:
msg = self.order_queues[tg_id].popleft()
await self.handle_order_update(msg, tg_id)
except Exception as e:
logger.error("Error processing order queue %s: %s", tg_id, e)
await asyncio.sleep(0.01)
async def _process_execution_queue(self, tg_id):
"""Continuously process execution queue for user."""
while tg_id in self.user_sockets:
try:
if self.execution_queues[tg_id]:
msg = self.execution_queues[tg_id].popleft()
await self.handle_execution_update(msg, tg_id)
except Exception as e:
logger.error("Error processing execution queue %s: %s", tg_id, e)
await asyncio.sleep(0.01)
async def close_user_socket(self, tg_id):
"""Gracefully close user connection."""
if tg_id in self.user_sockets:
self.user_sockets.pop(tg_id, None)
for key in (tg_id, tg_id + 1):
task = self.processing_tasks.pop(key, None)
if task and not task.done():
task.cancel()
self.order_queues.pop(tg_id, None)
self.execution_queues.pop(tg_id, None)
self.user_messages.pop(tg_id, None)
self.user_keys.pop(tg_id, None)
logger.info("Cleaned up user %s", tg_id)
async def handle_order_update(self, message, tg_id): async def handle_order_update(self, message, tg_id):
"""Handle order updates.""" """Process order updates."""
await self.message_handler.format_order_update(message, tg_id) try:
await self.message_handler.format_order_update(message, tg_id)
except Exception as e:
logger.error("Error handling order update for %s: %s", tg_id, e)
async def handle_execution_update(self, message, tg_id): async def handle_execution_update(self, message, tg_id):
"""Handle execution updates without duplicate processing.""" """Process execution updates."""
await self.message_handler.format_execution_update(message, tg_id) try:
await self.message_handler.format_execution_update(message, tg_id)
except Exception as e:
logger.error("Error handling execution update for %s: %s", tg_id, e)
@staticmethod @staticmethod
async def get_users_from_db(): async def get_users_from_db():
"""Get all users from the database.""" """Fetch all users from database."""
return await rq.get_users() try:
return await rq.get_users()
except Exception as e:
logger.error("Error getting users from DB: %s", e)
return []

View File

@@ -299,6 +299,12 @@ async def settings_for_margin_type(
deals = await get_active_positions_by_symbol( deals = await get_active_positions_by_symbol(
tg_id=callback_query.from_user.id, symbol=symbol tg_id=callback_query.from_user.id, symbol=symbol
) )
if deals == "Invalid API key permissions":
await callback_query.answer(
text="API ключ не имеет достаточных прав для смены маржи",
)
return
position = next((d for d in deals if d.get("symbol") == symbol), None) position = next((d for d in deals if d.get("symbol") == symbol), None)
if position: if position:
@@ -676,10 +682,19 @@ async def set_leverage_handler(message: Message, state: FSMContext) -> None:
await state.clear() await state.clear()
except Exception as e: except Exception as e:
await message.answer( errors_text = str(e)
text="Произошла ошибка при установке кредитного плеча. Пожалуйста, попробуйте позже.", known_errors = {
reply_markup=kbi.back_to_additional_settings, "Permission denied, please check your API key permissions": "API ключ не имеет достаточных прав для установки кредитного плеча"
)
}
for key, msg in known_errors.items():
if key in errors_text:
await message.answer(msg, reply_markup=kbi.back_to_additional_settings)
else:
await message.answer(
text="Произошла ошибка при установке кредитного плеча. Пожалуйста, попробуйте позже.",
reply_markup=kbi.back_to_additional_settings,
)
logger.error( logger.error(
"Error processing command leverage for user %s: %s", message.from_user.id, e "Error processing command leverage for user %s: %s", message.from_user.id, e
) )

View File

@@ -38,6 +38,12 @@ async def start_trading(callback_query: CallbackQuery, state: FSMContext) -> Non
deals = await get_active_positions_by_symbol( deals = await get_active_positions_by_symbol(
tg_id=callback_query.from_user.id, symbol=symbol tg_id=callback_query.from_user.id, symbol=symbol
) )
if deals == "Invalid API key permissions":
await callback_query.answer(
text="API ключ не имеет достаточных прав для запуска торговли",
)
return
position = next((d for d in deals if d.get("symbol") == symbol), None) position = next((d for d in deals if d.get("symbol") == symbol), None)
if position: if position:
@@ -109,7 +115,9 @@ async def start_trading(callback_query: CallbackQuery, state: FSMContext) -> Non
"The number of contracts exceeds minimum limit allowed": "️️Лимит ставки меньше минимально допустимого", "The number of contracts exceeds minimum limit allowed": "️️Лимит ставки меньше минимально допустимого",
"Order placement failed as your position may exceed the max": "Order placement failed as your position may exceed the max":
"Не удалось разместить ордер, так как ваша позиция может превышать максимальный лимит." "Не удалось разместить ордер, так как ваша позиция может превышать максимальный лимит."
"Пожалуйста, уменьшите кредитное плечо, чтобы увеличить максимальное значение" "Пожалуйста, уменьшите кредитное плечо, чтобы увеличить максимальное значение",
"Permission denied, please check your API key permissions": "API ключ не имеет достаточных прав для запуска торговли"
} }
if res == "OK": if res == "OK":
@@ -131,7 +139,16 @@ async def start_trading(callback_query: CallbackQuery, state: FSMContext) -> Non
await add_start_task_merged(user_id=callback_query.from_user.id, task=task) await add_start_task_merged(user_id=callback_query.from_user.id, task=task)
except Exception as e: except Exception as e:
await callback_query.answer(text="Произошла ошибка при запуске торговли") error_text = str(e)
known_errors = {
"Permission denied, please check your API key permissions": "API ключ не имеет достаточных прав для запуска торговли"
}
for key, msg in known_errors.items():
if key in error_text:
await callback_query.answer(msg)
else:
await callback_query.answer(text="Произошла ошибка при запуске торговли")
logger.error( logger.error(
"Error processing command start_trading for user %s: %s", "Error processing command start_trading for user %s: %s",
callback_query.from_user.id, callback_query.from_user.id,

View File

@@ -10,10 +10,9 @@ logging.config.dictConfig(LOGGING_CONFIG)
logger = logging.getLogger("database") logger = logging.getLogger("database")
BASE_DIR = Path(__file__).parent.resolve() BASE_DIR = Path(__file__).parent.resolve()
DATA_DIR = BASE_DIR / "dbs" BASE_DIR.mkdir(parents=True, exist_ok=True)
DATA_DIR.mkdir(parents=True, exist_ok=True)
DATABASE_URL = f"sqlite+aiosqlite:///{DATA_DIR / 'stcs.db'}" DATABASE_URL = f"sqlite+aiosqlite:///{BASE_DIR / 'stcs.db'}"
async_engine = create_async_engine( async_engine = create_async_engine(
DATABASE_URL, DATABASE_URL,
@@ -39,7 +38,7 @@ async_session = async_sessionmaker(
async def init_db(): async def init_db():
try: try:
async with async_engine.begin() as conn: async with async_engine.begin() as conn:
await conn.run_sync(Base.metadata.create_all) await conn.run_sync(lambda sync_conn: Base.metadata.create_all(bind=sync_conn, checkfirst=True))
logger.info("Database initialized.") logger.info("Database initialized.")
except Exception as e: except Exception as e:
logger.error("Database initialization failed: %s", e) logger.error("Database initialization failed: %s", e)

View File

@@ -154,8 +154,8 @@ class UserDeals(Base):
order_quantity = Column(Float, nullable=True) order_quantity = Column(Float, nullable=True)
martingale_factor = Column(Float, nullable=True) martingale_factor = Column(Float, nullable=True)
max_bets_in_series = Column(Integer, nullable=True) max_bets_in_series = Column(Integer, nullable=True)
take_profit_percent = Column(Integer, nullable=True) take_profit_percent = Column(Float, nullable=True)
stop_loss_percent = Column(Integer, nullable=True) stop_loss_percent = Column(Float, nullable=True)
trigger_price = Column(Float, nullable=True) trigger_price = Column(Float, nullable=True)
current_series = Column(Integer, nullable=True) current_series = Column(Integer, nullable=True)
commission_fee = Column(String, nullable=True) commission_fee = Column(String, nullable=True)

View File

@@ -86,7 +86,7 @@ async def set_user_api(tg_id: int, api_key: str, api_secret: str) -> bool:
else: else:
# Creating new record # Creating new record
user_api = UserApi( user_api = UserApi(
user=user, api_key=api_key, api_secret=api_secret user_id=user.id, api_key=api_key, api_secret=api_secret
) )
session.add(user_api) session.add(user_api)
@@ -141,7 +141,7 @@ async def set_user_symbol(tg_id: int, symbol: str) -> bool:
# Creating new record # Creating new record
user_symbol = UserSymbol( user_symbol = UserSymbol(
symbol=symbol, symbol=symbol,
user=user, user_id=user.id,
) )
session.add(user_symbol) session.add(user_symbol)
@@ -197,7 +197,7 @@ async def create_user_additional_settings(tg_id: int) -> None:
# Create the user additional settings # Create the user additional settings
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
user=user, user_id=user.id,
trade_mode="Long", # Default value trade_mode="Long", # Default value
switch_side="По направлению", switch_side="По направлению",
side="Buy", side="Buy",
@@ -267,7 +267,7 @@ async def set_trade_mode(tg_id: int, trade_mode: str) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
trade_mode=trade_mode, trade_mode=trade_mode,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -306,7 +306,7 @@ async def set_margin_type(tg_id: int, margin_type: str) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
margin_type=margin_type, margin_type=margin_type,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -345,7 +345,7 @@ async def set_switch_side(tg_id: int, switch_side: str) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
switch_side=switch_side, switch_side=switch_side,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -384,7 +384,7 @@ async def set_side(tg_id: int, side: str) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
side=side, side=side,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -423,7 +423,7 @@ async def set_leverage(tg_id: int, leverage: str) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
leverage=leverage, leverage=leverage,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -462,7 +462,7 @@ async def set_order_quantity(tg_id: int, order_quantity: float) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
order_quantity=order_quantity, order_quantity=order_quantity,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -503,7 +503,7 @@ async def set_martingale_factor(tg_id: int, martingale_factor: float) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
martingale_factor=martingale_factor, martingale_factor=martingale_factor,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -546,7 +546,7 @@ async def set_max_bets_in_series(tg_id: int, max_bets_in_series: int) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
max_bets_in_series=max_bets_in_series, max_bets_in_series=max_bets_in_series,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -587,7 +587,7 @@ async def set_trigger_price(tg_id: int, trigger_price: float) -> bool:
# Creating new record # Creating new record
user_additional_settings = UserAdditionalSettings( user_additional_settings = UserAdditionalSettings(
trigger_price=trigger_price, trigger_price=trigger_price,
user=user, user_id=user.id,
) )
session.add(user_additional_settings) session.add(user_additional_settings)
@@ -627,7 +627,7 @@ async def create_user_risk_management(tg_id: int) -> None:
# Create the user risk management # Create the user risk management
user_risk_management = UserRiskManagement( user_risk_management = UserRiskManagement(
user=user, user_id=user.id,
take_profit_percent=1.0, take_profit_percent=1.0,
stop_loss_percent=1.0, stop_loss_percent=1.0,
commission_fee="Yes_commission_fee", commission_fee="Yes_commission_fee",
@@ -692,7 +692,7 @@ async def set_take_profit_percent(tg_id: int, take_profit_percent: float) -> boo
# Creating new record # Creating new record
user_risk_management = UserRiskManagement( user_risk_management = UserRiskManagement(
take_profit_percent=take_profit_percent, take_profit_percent=take_profit_percent,
user=user, user_id=user.id,
) )
session.add(user_risk_management) session.add(user_risk_management)
@@ -733,7 +733,7 @@ async def set_stop_loss_percent(tg_id: int, stop_loss_percent: float) -> bool:
# Creating new record # Creating new record
user_risk_management = UserRiskManagement( user_risk_management = UserRiskManagement(
stop_loss_percent=stop_loss_percent, stop_loss_percent=stop_loss_percent,
user=user, user_id=user.id,
) )
session.add(user_risk_management) session.add(user_risk_management)
@@ -774,7 +774,7 @@ async def set_commission_fee(tg_id: int, commission_fee: str) -> bool:
# Creating new record # Creating new record
user_risk_management = UserRiskManagement( user_risk_management = UserRiskManagement(
commission_fee=commission_fee, commission_fee=commission_fee,
user=user, user_id=user.id,
) )
session.add(user_risk_management) session.add(user_risk_management)
@@ -815,7 +815,7 @@ async def set_commission_place(tg_id: int, commission_place: str) -> bool:
# Creating new record # Creating new record
user_risk_management = UserRiskManagement( user_risk_management = UserRiskManagement(
commission_place=commission_place, commission_place=commission_place,
user=user, user_id=user.id,
) )
session.add(user_risk_management) session.add(user_risk_management)
@@ -855,7 +855,7 @@ async def create_user_conditional_settings(tg_id: int) -> None:
# Create the user conditional settings # Create the user conditional settings
user_conditional_settings = UserConditionalSettings( user_conditional_settings = UserConditionalSettings(
user=user, user_id=user.id,
timer_start=0, timer_start=0,
timer_end=0, timer_end=0,
) )
@@ -920,7 +920,7 @@ async def set_start_timer(tg_id: int, timer_start: int) -> bool:
# Creating new record # Creating new record
user_conditional_settings = UserConditionalSettings( user_conditional_settings = UserConditionalSettings(
timer_start=timer_start, timer_start=timer_start,
user=user, user_id=user.id,
) )
session.add(user_conditional_settings) session.add(user_conditional_settings)
@@ -959,7 +959,7 @@ async def set_stop_timer(tg_id: int, timer_end: int) -> bool:
# Creating new record # Creating new record
user_conditional_settings = UserConditionalSettings( user_conditional_settings = UserConditionalSettings(
timer_end=timer_end, timer_end=timer_end,
user=user, user_id=user.id,
) )
session.add(user_conditional_settings) session.add(user_conditional_settings)
@@ -1051,7 +1051,7 @@ async def set_user_deal(
else: else:
# Creating new record # Creating new record
new_deal = UserDeals( new_deal = UserDeals(
user=user, user_id=user.id,
symbol=symbol, symbol=symbol,
current_step=current_step, current_step=current_step,
current_series=current_series, current_series=current_series,

View File

@@ -8,7 +8,7 @@ After=syslog.target network-online.target
ExecStart=sudo -u www-data /usr/bin/python3 /var/www/stcs/BybitBot_API.py ExecStart=sudo -u www-data /usr/bin/python3 /var/www/stcs/BybitBot_API.py
PIDFile=/var/run/python/stcs.pid PIDFile=/var/run/python/stcs.pid
RemainAfterExit=no RemainAfterExit=no
RuntimeMaxSec=3600s RuntimeMaxSec=604800s
Restart=always Restart=always
RestartSec=5s RestartSec=5s

1
run.py
View File

@@ -31,7 +31,6 @@ async def main():
dp = Dispatcher(storage=storage) dp = Dispatcher(storage=storage)
dp.include_router(router) dp.include_router(router)
web_socket = WebSocketBot(telegram_bot=bot) web_socket = WebSocketBot(telegram_bot=bot)
await web_socket.clear_user_sockets()
ws_task = asyncio.create_task(web_socket.run_user_check_loop()) ws_task = asyncio.create_task(web_socket.run_user_check_loop())
tg_task = asyncio.create_task(dp.start_polling(bot)) tg_task = asyncio.create_task(dp.start_polling(bot))