25 Commits

Author SHA1 Message Date
fec367cc1d Merge pull request 'dev' (#8) from Alex/stcs:dev into stable
Reviewed-on: #8
2025-09-19 17:17:25 +07:00
algizn97
4bbff680aa Update 2025-09-19 14:57:08 +05:00
algizn97
49d4bb26bf Added trigger price 2025-09-19 14:45:34 +05:00
algizn97
29bb6bd0a8 The buttons for selecting the type of entry and setting the limit and trigger prices have been updated. 2025-09-19 14:43:42 +05:00
algizn97
2fb8cb4acb Added the ability to open a deal at a trigger price 2025-09-19 14:42:46 +05:00
algizn97
887b46c1d4 Added a condition for the price trigger 2025-09-19 14:42:12 +05:00
algizn97
b074d1d8a1 Added the ability to set a price trigger and limit 2025-09-19 14:41:18 +05:00
aebcc9dff2 Merge pull request 'dev' (#7) from Alex/stcs:dev into stable
Reviewed-on: #7
2025-09-18 22:49:55 +07:00
algizn97
e2f9478971 Fixed 2025-09-18 16:17:39 +05:00
algizn97
4f0668970f Fixed 2025-09-18 08:14:23 +05:00
algizn97
4c9901c14a Fixed 2025-09-17 20:51:30 +05:00
algizn97
17dba19078 Updated 2025-09-13 12:30:40 +05:00
58a4c6af06 systemd + gitignore 2025-09-13 00:34:49 +07:00
b37b7193b2 Обновить README.md 2025-09-13 00:16:03 +07:00
05e8005ec9 Merge pull request 'develops' (#6) from Alex/stcs:develops into stable
Reviewed-on: #6
2025-09-12 23:52:03 +07:00
algizn97
0de3b17d1d Updated 2025-09-12 21:37:38 +05:00
algizn97
b77c0f7dcc Merge remote-tracking branch 'origin/develop' into develop 2025-09-12 21:31:42 +05:00
algizn97
3ccfb64be8 Fixed 2025-09-12 21:31:13 +05:00
13d69e2f73 Merge branch 'stable' into develop 2025-09-12 23:23:46 +07:00
algizn97
751cde86f9 Fixed 2025-09-12 12:38:02 +05:00
algizn97
1b95992297 Merge remote-tracking branch 'origin/develop' into develop 2025-09-12 12:11:14 +05:00
algizn97
d8bb3fda82 Fixed 2025-09-12 12:10:20 +05:00
algizn97
4704d4a486 Fixed 2025-09-12 12:07:21 +05:00
algizn97
c7b3ae7876 Fixed 2025-09-12 12:04:40 +05:00
algizn97
6fb876ade2 Fixed 2025-09-12 11:47:06 +05:00
18 changed files with 429 additions and 305 deletions

4
.gitignore vendored
View File

@@ -11,3 +11,7 @@ venv/
/.idea /.idea
/myenv /myenv
myenv myenv
*.sqlite3
*.log

View File

@@ -1,7 +1,7 @@
import asyncio import asyncio
import logging.config import logging.config
from aiogram import Bot, Dispatcher from aiogram import Bot, Dispatcher
from aiogram.fsm.storage.redis import RedisStorage
from app.services.Bybit.functions.bybit_ws import get_or_create_event_loop, set_event_loop from app.services.Bybit.functions.bybit_ws import get_or_create_event_loop, set_event_loop
from app.telegram.database.models import async_main from app.telegram.database.models import async_main
from app.telegram.handlers.handlers import router from app.telegram.handlers.handlers import router
@@ -17,9 +17,8 @@ from config import TOKEN_TG_BOT_1
logging.config.dictConfig(LOGGING_CONFIG) logging.config.dictConfig(LOGGING_CONFIG)
logger = logging.getLogger("main") logger = logging.getLogger("main")
storage = RedisStorage.from_url("redis://localhost:6379/0")
bot = Bot(token=TOKEN_TG_BOT_1) bot = Bot(token=TOKEN_TG_BOT_1)
dp = Dispatcher(storage=storage) dp = Dispatcher()
async def main() -> None: async def main() -> None:

View File

@@ -27,7 +27,7 @@ Crypto Trading Telegram Bot
- Хранение пользовательских настроек и статистики в базе данных. - Хранение пользовательских настроек и статистики в базе данных.
## Установка и запуск ## Установка
1. Клонируйте репозиторий: 1. Клонируйте репозиторий:
@@ -41,6 +41,10 @@ git clone https://git.svoboda.works/kodorvan/stcs
```bash ```bash
pip install -r requirements.txt pip install -r requirements.txt
``` ```
или для отдельного пользователя
```bash
sudo -u www-data /usr/bin/pip install -r requirements.txt
```
3. Зарегистрируйте чат-робота и сгенерируйте ключ авторизации<br> 3. Зарегистрируйте чат-робота и сгенерируйте ключ авторизации<br>
[@BotFather](https://t.me/BotFather) [@BotFather](https://t.me/BotFather)
@@ -57,6 +61,32 @@ nvim .env
python BybitBot_API.py python BybitBot_API.py
``` ```
## Настройка автономной работы
1. Создаём файл конфигурации SystemD
```bash
sudo cp examples/systemd/stcs.service /etc/systemd/system/
```
2. Настраиваем его
```bash
nvim /etc/systemd/system/stcs.service
```
3. Добавляем в автозапуск
```bash
sudo systemctl enable stcs
```
4. Запускаем
```bash
sudo service stcs start
```
5. Проверяем
```bash
sudo service stcs status
```
## Настройки пользователя ## Настройки пользователя
- Кредитное плечо (например, 15x) - Кредитное плечо (например, 15x)

View File

@@ -1,7 +1,6 @@
import asyncio import asyncio
import logging.config import logging.config
import time import time
import app.services.Bybit.functions.balance as balance_g import app.services.Bybit.functions.balance as balance_g
import app.services.Bybit.functions.price_symbol as price_symbol import app.services.Bybit.functions.price_symbol as price_symbol
import app.telegram.database.requests as rq import app.telegram.database.requests as rq
@@ -47,11 +46,10 @@ def format_trade_details_position(data, commission_fee):
Форматирует информацию о сделке в виде строки. Форматирует информацию о сделке в виде строки.
""" """
msg = data.get("data", [{}])[0] msg = data.get("data", [{}])[0]
closed_size = safe_float(msg.get("closedSize", 0)) closed_size = safe_float(msg.get("closedSize", 0))
symbol = msg.get("symbol", "N/A") symbol = msg.get("symbol", "N/A")
entry_price = safe_float(msg.get("execPrice", 0)) entry_price = safe_float(msg.get("execPrice", 0))
qty = safe_float(msg.get("execQty", 0)) qty = safe_float(msg.get("orderQty", 0))
order_type = msg.get("orderType", "N/A") order_type = msg.get("orderType", "N/A")
side = msg.get("side", "") side = msg.get("side", "")
commission = safe_float(msg.get("execFee", 0)) commission = safe_float(msg.get("execFee", 0))
@@ -69,7 +67,7 @@ def format_trade_details_position(data, commission_fee):
movement = side movement = side
if closed_size > 0: if closed_size > 0:
return ( text = (
f"Сделка закрыта:\n" f"Сделка закрыта:\n"
f"Торговая пара: {symbol}\n" f"Торговая пара: {symbol}\n"
f"Цена исполнения: {entry_price:.6f}\n" f"Цена исполнения: {entry_price:.6f}\n"
@@ -80,8 +78,9 @@ def format_trade_details_position(data, commission_fee):
f"Комиссия за сделку: {commission:.6f}\n" f"Комиссия за сделку: {commission:.6f}\n"
f"Реализованная прибыль: {pnl:.6f} USDT" f"Реализованная прибыль: {pnl:.6f} USDT"
) )
return text
if order_type == "Market": if order_type == "Market":
return ( text = (
f"Сделка открыта:\n" f"Сделка открыта:\n"
f"Торговая пара: {symbol}\n" f"Торговая пара: {symbol}\n"
f"Цена исполнения: {entry_price:.6f}\n" f"Цена исполнения: {entry_price:.6f}\n"
@@ -90,6 +89,7 @@ def format_trade_details_position(data, commission_fee):
f"Движение: {movement}\n" f"Движение: {movement}\n"
f"Комиссия за сделку: {commission:.6f}" f"Комиссия за сделку: {commission:.6f}"
) )
return text
return None return None
@@ -102,12 +102,11 @@ def format_order_details_position(data):
qty = safe_float(msg.get("qty", 0)) qty = safe_float(msg.get("qty", 0))
cum_exec_qty = safe_float(msg.get("cumExecQty", 0)) cum_exec_qty = safe_float(msg.get("cumExecQty", 0))
cum_exec_fee = safe_float(msg.get("cumExecFee", 0)) cum_exec_fee = safe_float(msg.get("cumExecFee", 0))
take_profit = safe_float(msg.get("takeProfit", 0))
stop_loss = safe_float(msg.get("stopLoss", 0))
order_status = msg.get("orderStatus", "N/A") order_status = msg.get("orderStatus", "N/A")
symbol = msg.get("symbol", "N/A") symbol = msg.get("symbol", "N/A")
order_type = msg.get("orderType", "N/A") order_type = msg.get("orderType", "N/A")
side = msg.get("side", "") side = msg.get("side", "")
trigger_price = msg.get("triggerPrice", "N/A")
movement = "" movement = ""
if side.lower() == "buy": if side.lower() == "buy":
@@ -126,8 +125,6 @@ def format_order_details_position(data):
f"Исполнено позиций: {cum_exec_qty}\n" f"Исполнено позиций: {cum_exec_qty}\n"
f"Тип ордера: {order_type}\n" f"Тип ордера: {order_type}\n"
f"Движение: {movement}\n" f"Движение: {movement}\n"
f"Тейк-профит: {take_profit:.6f}\n"
f"Стоп-лосс: {stop_loss:.6f}\n"
f"Комиссия за сделку: {cum_exec_fee:.6f}\n" f"Комиссия за сделку: {cum_exec_fee:.6f}\n"
) )
return text return text
@@ -140,8 +137,6 @@ def format_order_details_position(data):
f"Количество: {qty}\n" f"Количество: {qty}\n"
f"Тип ордера: {order_type}\n" f"Тип ордера: {order_type}\n"
f"Движение: {movement}\n" f"Движение: {movement}\n"
f"Тейк-профит: {take_profit:.6f}\n"
f"Стоп-лосс: {stop_loss:.6f}\n"
) )
return text return text
@@ -153,8 +148,28 @@ def format_order_details_position(data):
f"Количество: {qty}\n" f"Количество: {qty}\n"
f"Тип ордера: {order_type}\n" f"Тип ордера: {order_type}\n"
f"Движение: {movement}\n" f"Движение: {movement}\n"
f"Тейк-профит: {take_profit:.6f}\n" )
f"Стоп-лосс: {stop_loss:.6f}\n" return text
elif order_status.lower() == "untriggered":
text = (
f"Условный ордер создан:\n"
f"Торговая пара: {symbol}\n"
f"Цена: {price:.6f}\n"
f"Триггер цена: {trigger_price}\n"
f"Количество: {qty}\n"
f"Тип ордера: {order_type}\n"
f"Движение: {movement}\n"
)
return text
elif order_status.lower() == "deactivated":
text = (
f"Условный ордер отменен:\n"
f"Торговая пара: {symbol}\n"
f"Цена: {price:.6f}\n"
f"Триггер цена: {trigger_price}\n"
f"Количество: {qty}\n"
f"Тип ордера: {order_type}\n"
f"Движение: {movement}\n"
) )
return text return text
return None return None
@@ -172,7 +187,7 @@ def parse_pnl_from_msg(msg) -> float:
return 0.0 return 0.0
async def calculate_total_budget(starting_quantity, martingale_factor, max_steps, commission_fee_percent, leverage, current_price): async def calculate_total_budget(starting_quantity, martingale_factor, max_steps, commission_fee_percent):
""" """
Вычисляет общий бюджет серии ставок с учётом цены пары, комиссии и кредитного плеча. Вычисляет общий бюджет серии ставок с учётом цены пары, комиссии и кредитного плеча.
@@ -189,22 +204,16 @@ async def calculate_total_budget(starting_quantity, martingale_factor, max_steps
""" """
total = 0 total = 0
for step in range(max_steps): for step in range(max_steps):
quantity = starting_quantity * (martingale_factor ** step) # размер ставки на текущем шаге в USDT base_quantity = starting_quantity * (martingale_factor ** step)
if commission_fee_percent == 0:
# Комиссия уже включена в сумму ставки, поэтому реальный размер позиции меньше
quantity = base_quantity / (1 + commission_fee_percent)
else:
# Комиссию добавляем сверху
quantity = base_quantity * (1 + commission_fee_percent)
# Переводим ставку из USDT в количество актива по текущей цене total += quantity
quantity_in_asset = quantity / current_price return total
# Учитываем комиссию за вход и выход (умножаем на 2)
quantity_with_fee = quantity * (1 + 2 * commission_fee_percent / 100)
# Учитываем кредитное плечо - реальные собственные вложения меньше
effective_quantity = quantity_with_fee / leverage
total += effective_quantity
# Возвращаем бюджет в USDT
total_usdt = total * current_price
return total_usdt
async def handle_execution_message(message, msg): async def handle_execution_message(message, msg):
@@ -212,7 +221,6 @@ async def handle_execution_message(message, msg):
Обработчик сообщений об исполнении сделки. Обработчик сообщений об исполнении сделки.
Логирует событие и проверяет условия для мартингейла и TP. Логирует событие и проверяет условия для мартингейла и TP.
""" """
tg_id = message.from_user.id tg_id = message.from_user.id
data = msg.get("data", [{}])[0] data = msg.get("data", [{}])[0]
data_main_risk_stgs = await rq.get_user_risk_management_settings(tg_id) data_main_risk_stgs = await rq.get_user_risk_management_settings(tg_id)
@@ -239,6 +247,9 @@ async def handle_execution_message(message, msg):
if trade_info: if trade_info:
await message.answer(f"{trade_info}", reply_markup=inline_markup.back_to_main) await message.answer(f"{trade_info}", reply_markup=inline_markup.back_to_main)
if data is not None:
await rq.update_trigger_price(tg_id, 0.0)
if closed_size == 0: if closed_size == 0:
side = data.get("side", "") side = data.get("side", "")
@@ -248,13 +259,13 @@ async def handle_execution_message(message, msg):
await rq.set_last_series_info(tg_id, last_side="Sell") await rq.set_last_series_info(tg_id, last_side="Sell")
if trigger == "Автоматический" and closed_size > 0: if trigger == "Автоматический" and closed_size > 0:
if pnl < 0:
if trading_mode == 'Switch': if trading_mode == 'Switch':
side = data_main_stgs.get("last_side") side = data_main_stgs.get("last_side")
else: else:
side = "Buy" if trading_mode == "Long" else "Sell" side = "Buy" if trading_mode == "Long" else "Sell"
if pnl < 0:
current_martingale = await rq.get_martingale_step(tg_id) current_martingale = await rq.get_martingale_step(tg_id)
current_martingale_step = int(current_martingale) current_martingale_step = int(current_martingale)
current_martingale += 1 current_martingale += 1
@@ -262,6 +273,7 @@ async def handle_execution_message(message, msg):
float(martingale_factor) ** current_martingale_step float(martingale_factor) ** current_martingale_step
) )
await rq.update_martingale_step(tg_id, current_martingale) await rq.update_martingale_step(tg_id, current_martingale)
await rq.update_starting_quantity(tg_id=tg_id, num=next_quantity)
await message.answer( await message.answer(
f"❗️ Сделка закрылась в минус, открываю новую сделку с увеличенной ставкой.\n" f"❗️ Сделка закрылась в минус, открываю новую сделку с увеличенной ставкой.\n"
) )
@@ -275,9 +287,12 @@ async def handle_execution_message(message, msg):
) )
elif pnl > 0: elif pnl > 0:
await rq.update_martingale_step(tg_id, 0) await rq.update_martingale_step(tg_id, 1)
num = data_main_stgs.get("base_quantity")
await rq.update_starting_quantity(tg_id=tg_id, num=num)
await message.answer( await message.answer(
"❗️ Прибыль достигнута, шаг мартингейла сброшен." "❗️ Прибыль достигнута, шаг мартингейла сброшен. "
"Возврат к начальной ставке."
) )
@@ -287,7 +302,10 @@ async def handle_order_message(message, msg: dict) -> None:
Логирует событие и проверяет условия для мартингейла и TP. Логирует событие и проверяет условия для мартингейла и TP.
""" """
# logger.info(f"Исполнен ордер:\n{json.dumps(msg, indent=4, ensure_ascii=False)}") # logger.info(f"Исполнен ордер:\n{json.dumps(msg, indent=4, ensure_ascii=False)}")
data = msg.get("data", [{}])[0]
tg_id = message.from_user.id
if data is not None:
await rq.update_trigger_price(tg_id, 0.0)
trade_info = format_order_details_position(msg) trade_info = format_order_details_position(msg)
if trade_info: if trade_info:
@@ -333,7 +351,7 @@ async def open_position(
bybit_margin_mode = ( bybit_margin_mode = (
"ISOLATED_MARGIN" if margin_mode == "Isolated" else "REGULAR_MARGIN" "ISOLATED_MARGIN" if margin_mode == "Isolated" else "REGULAR_MARGIN"
) )
trigger_price = await rq.get_trigger_price(tg_id)
limit_price = None limit_price = None
if order_type == "Limit": if order_type == "Limit":
limit_price = await rq.get_limit_price(tg_id) limit_price = await rq.get_limit_price(tg_id)
@@ -348,8 +366,6 @@ async def open_position(
max_risk_percent = safe_float(data_risk_stgs.get("max_risk_deal")) max_risk_percent = safe_float(data_risk_stgs.get("max_risk_deal"))
loss_profit = safe_float(data_risk_stgs.get("price_loss")) loss_profit = safe_float(data_risk_stgs.get("price_loss"))
commission_fee = data_risk_stgs.get("commission_fee") commission_fee = data_risk_stgs.get("commission_fee")
starting_quantity = safe_float(data_main_stgs.get('starting_quantity'))
martingale_factor = safe_float(data_main_stgs.get('martingale_factor'))
fee_info = client.get_fee_rates(category='linear', symbol=symbol) fee_info = client.get_fee_rates(category='linear', symbol=symbol)
instruments_resp = client.get_instruments_info(category="linear", symbol=symbol) instruments_resp = client.get_instruments_info(category="linear", symbol=symbol)
instrument = instruments_resp.get("result", {}).get("list", []) instrument = instruments_resp.get("result", {}).get("list", [])
@@ -359,33 +375,19 @@ async def open_position(
else: else:
commission_fee_percent = 0.0 commission_fee_percent = 0.0
total_budget = await calculate_total_budget( if commission_fee_percent > 0:
starting_quantity=starting_quantity, # Добавляем к тейк-профиту процент комиссии
martingale_factor=martingale_factor, tp_multiplier = 1 + (loss_profit / 100) + commission_fee_percent
max_steps=max_martingale_steps, else:
commission_fee_percent=commission_fee_percent, tp_multiplier = 1 + (loss_profit / 100)
leverage=leverage,
current_price=entry_price,
)
balance = await balance_g.get_balance(tg_id, message)
if safe_float(balance) < total_budget:
logger.error(
f"Недостаточно средств для серии из {max_martingale_steps} шагов с текущими параметрами. "
f"Требуемый бюджет: {total_budget:.2f} USDT, доступно: {balance} USDT."
)
await message.answer(
f"Недостаточно средств для серии из {max_martingale_steps} шагов с текущими параметрами. "
f"Требуемый бюджет: {total_budget:.2f} USDT, доступно: {balance} USDT.",
reply_markup=inline_markup.back_to_main,
)
return
if order_type == "Limit" and limit_price: if order_type == "Limit" and limit_price:
price_for_calc = limit_price price_for_calc = limit_price
else: else:
price_for_calc = entry_price price_for_calc = entry_price
balance = await balance_g.get_balance(tg_id, message)
potential_loss = safe_float(quantity) * price_for_calc * (loss_profit / 100) potential_loss = safe_float(quantity) * price_for_calc * (loss_profit / 100)
adjusted_loss = potential_loss / leverage adjusted_loss = potential_loss / leverage
allowed_loss = safe_float(balance) * (max_risk_percent / 100) allowed_loss = safe_float(balance) * (max_risk_percent / 100)
@@ -423,7 +425,7 @@ async def open_position(
logger.info(f"Set leverage to {leverage_to_set} for {symbol}") logger.info(f"Set leverage to {leverage_to_set} for {symbol}")
except exceptions.InvalidRequestError as e: except exceptions.InvalidRequestError as e:
if "110043" in str(e): if "110043" in str(e):
logger.info(f"Leverage already set to {leverage} for {symbol}") logger.info(f"Leverage already set to {leverage_to_set} for {symbol}")
else: else:
raise e raise e
@@ -453,18 +455,35 @@ async def open_position(
if bybit_margin_mode == "ISOLATED_MARGIN": if bybit_margin_mode == "ISOLATED_MARGIN":
# Открываем позицию # Открываем позицию
if trigger_price and float(trigger_price) > 0:
response = client.place_order(
category="linear",
symbol=symbol,
side=side,
orderType="Stop" if order_type == "Conditional" else order_type,
qty=str(quantity),
price=(str(limit_price) if order_type == "Limit" and limit_price else None),
triggerPrice=str(trigger_price),
triggerBy="LastPrice",
triggerDirection=2 if side == "Buy" else 1,
timeInForce="GTC",
orderLinkId=f"deal_{symbol}_{int(time.time())}",
)
else:
# Обычный ордер, без триггера
response = client.place_order( response = client.place_order(
category="linear", category="linear",
symbol=symbol, symbol=symbol,
side=side, side=side,
orderType=order_type, orderType=order_type,
qty=str(quantity), qty=str(quantity),
price=( price=(str(limit_price) if order_type == "Limit" and limit_price else None),
str(limit_price) if order_type == "Limit" and limit_price else None
),
timeInForce="GTC", timeInForce="GTC",
orderLinkId=f"deal_{symbol}_{int(time.time())}", orderLinkId=f"deal_{symbol}_{int(time.time())}",
) )
if response.get("retCode", -1) == 0:
return True
if response.get("retCode", -1) != 0: if response.get("retCode", -1) != 0:
logger.error(f"Ошибка открытия ордера: {response}") logger.error(f"Ошибка открытия ордера: {response}")
await message.answer( await message.answer(
@@ -480,9 +499,11 @@ async def open_position(
if liq_price > 0 and avg_price > 0: if liq_price > 0 and avg_price > 0:
if side.lower() == "buy": if side.lower() == "buy":
take_profit_price = avg_price + (avg_price - liq_price) base_tp = avg_price + (avg_price - liq_price)
take_profit_price = base_tp * (1 + commission_fee_percent)
else: else:
take_profit_price = avg_price - (liq_price - avg_price) base_tp = avg_price - (liq_price - avg_price)
take_profit_price = base_tp * (1 - commission_fee_percent)
take_profit_price = max(take_profit_price, 0) take_profit_price = max(take_profit_price, 0)
@@ -496,7 +517,7 @@ async def open_position(
logger.info("Режим TP/SL уже установлен - пропускаем") logger.info("Режим TP/SL уже установлен - пропускаем")
else: else:
raise raise
resp = client.set_trading_stop( client.set_trading_stop(
category="linear", category="linear",
symbol=symbol, symbol=symbol,
takeProfit=str(round(take_profit_price, 5)), takeProfit=str(round(take_profit_price, 5)),
@@ -531,10 +552,10 @@ async def open_position(
base_price = limit_price base_price = limit_price
if side.lower() == "buy": if side.lower() == "buy":
take_profit_price = base_price * (1 + loss_profit / 100) take_profit_price = base_price * tp_multiplier
stop_loss_price = base_price * (1 - loss_profit / 100) stop_loss_price = base_price * (1 - loss_profit / 100)
else: else:
take_profit_price = base_price * (1 - loss_profit / 100) take_profit_price = base_price * (1 - (loss_profit / 100) - (commission_fee_percent))
stop_loss_price = base_price * (1 + loss_profit / 100) stop_loss_price = base_price * (1 + loss_profit / 100)
take_profit_price = max(take_profit_price, 0) take_profit_price = max(take_profit_price, 0)
@@ -567,6 +588,9 @@ async def open_position(
slOrderType=sl_order_type, slOrderType=sl_order_type,
slLimitPrice=str(sl_limit_price) if sl_limit_price else None, slLimitPrice=str(sl_limit_price) if sl_limit_price else None,
tpslMode=tpsl_mode, tpslMode=tpsl_mode,
triggerPrice=str(trigger_price) if trigger_price and float(trigger_price) > 0 else None,
triggerBy="LastPrice" if trigger_price and float(trigger_price) > 0 else None,
triggerDirection=2 if side == "Buy" else 1 if trigger_price and float(trigger_price) > 0 else None,
timeInForce="GTC", timeInForce="GTC",
orderLinkId=f"deal_{symbol}_{int(time.time())}", orderLinkId=f"deal_{symbol}_{int(time.time())}",
) )
@@ -747,20 +771,20 @@ async def get_active_orders(tg_id, message):
""" """
client = await get_bybit_client(tg_id) client = await get_bybit_client(tg_id)
response = client.get_open_orders( response = client.get_open_orders(
category="linear", settleCoin="USDT", orderType="Limit" category="linear", settleCoin="USDT", orderType="Limit" and "Market"
) )
orders = response.get("result", {}).get("list", []) orders = response.get("result", {}).get("list", [])
limit_orders = [order for order in orders if order.get("orderType") == "Limit"] limit_orders = [order for order in orders]
if limit_orders: if limit_orders:
symbols = [order["symbol"] for order in limit_orders] symbols = [order["symbol"] for order in limit_orders]
await message.answer( await message.answer(
"📈 Ваши активные лимитные ордера:", "📈 Ваши активные ордера:",
reply_markup=inline_markup.create_trades_inline_keyboard_limits(symbols), reply_markup=inline_markup.create_trades_inline_keyboard_limits(symbols),
) )
else: else:
await message.answer( await message.answer(
"❗️ У вас нет активных лимитных ордеров.", "❗️ У вас нет активных ордеров.",
reply_markup=inline_markup.back_to_main, reply_markup=inline_markup.back_to_main,
) )
return return
@@ -775,12 +799,11 @@ async def get_active_orders_by_symbol(tg_id, symbol, message):
limit_orders = [ limit_orders = [
order order
for order in active_orders.get("result", {}).get("list", []) for order in active_orders.get("result", {}).get("list", [])
if order.get("orderType") == "Limit"
] ]
if not limit_orders: if not limit_orders:
await message.answer( await message.answer(
"Нет активных лимитных ордеров по данной торговой паре.", "Нет активных ордеров по данной торговой паре.",
reply_markup=inline_markup.back_to_main, reply_markup=inline_markup.back_to_main,
) )
return return
@@ -792,9 +815,8 @@ async def get_active_orders_by_symbol(tg_id, symbol, message):
f"Тип ордера: {order.get('orderType')}\n" f"Тип ордера: {order.get('orderType')}\n"
f"Сторона: {order.get('side')}\n" f"Сторона: {order.get('side')}\n"
f"Цена: {order.get('price')}\n" f"Цена: {order.get('price')}\n"
f"Триггер цена: {order.get('triggerPrice')}\n"
f"Количество: {order.get('qty')}\n" f"Количество: {order.get('qty')}\n"
f"Тейк-профит: {order.get('takeProfit')}\n"
f"Стоп-лосс: {order.get('stopLoss')}\n"
) )
texts.append(text) texts.append(text)

View File

@@ -71,7 +71,7 @@ def on_order_callback(message, msg):
if event_loop is not None: if event_loop is not None:
from app.services.Bybit.functions.Futures import handle_order_message from app.services.Bybit.functions.Futures import handle_order_message
asyncio.run_coroutine_threadsafe(handle_order_message(message, msg), event_loop) asyncio.run_coroutine_threadsafe(handle_order_message(message, msg), event_loop)
logger.info("Callback выполнен.") logger.info("Callback для ордера выполнен.")
else: else:
logger.error("Event loop не установлен, callback пропущен.") logger.error("Event loop не установлен, callback пропущен.")
@@ -80,7 +80,7 @@ def on_execution_callback(message, ws_msg):
if event_loop is not None: if event_loop is not None:
from app.services.Bybit.functions.Futures import handle_execution_message from app.services.Bybit.functions.Futures import handle_execution_message
asyncio.run_coroutine_threadsafe(handle_execution_message(message, ws_msg), event_loop) asyncio.run_coroutine_threadsafe(handle_execution_message(message, ws_msg), event_loop)
logger.info("Callback выполнен.") logger.info("Callback для маркета выполнен.")
else: else:
logger.error("Event loop не установлен, callback пропущен.") logger.error("Event loop не установлен, callback пропущен.")

View File

@@ -17,8 +17,9 @@ import app.telegram.Keyboards.inline_keyboards as inline_markup
import app.telegram.database.requests as rq import app.telegram.database.requests as rq
from aiogram.types import Message, CallbackQuery from aiogram.types import Message, CallbackQuery
from app.services.Bybit.functions.price_symbol import get_price from app.services.Bybit.functions.price_symbol import get_price
# import app.services.Bybit.functions.balance as balance_g
from app.states.States import (state_update_entry_type, state_update_symbol, state_limit_price, from app.states.States import (state_update_symbol,
SetTP_SL_State, CloseTradeTimerState) SetTP_SL_State, CloseTradeTimerState)
from aiogram.fsm.context import FSMContext from aiogram.fsm.context import FSMContext
@@ -73,7 +74,7 @@ async def start_bybit_trade_message(message: Message) -> None:
text = ( text = (
f"💎 Торговля на Bybit\n\n" f"💎 Торговля на Bybit\n\n"
f"⚖️ Ваш баланс (USDT): {balance}\n" f"⚖️ Ваш баланс (USDT): {float(balance):.2f}\n"
f"📊 Текущая торговая пара: {symbol}\n" f"📊 Текущая торговая пара: {symbol}\n"
f"$$$ Цена: {price}\n\n" f"$$$ Цена: {price}\n\n"
"Как начать торговлю?\n\n" "Как начать торговлю?\n\n"
@@ -119,72 +120,6 @@ async def update_symbol_for_trade(message: Message, state: FSMContext) -> None:
await state.clear() await state.clear()
@router_functions_bybit_trade.callback_query(F.data == 'clb_update_entry_type')
async def update_entry_type_message(callback: CallbackQuery, state: FSMContext) -> None:
"""
Запрашивает у пользователя тип входа в позицию (Market или Limit).
"""
await state.set_state(state_update_entry_type.entry_type)
await callback.message.answer("Выберите тип входа в позицию:", reply_markup=inline_markup.entry_order_type_markup)
await callback.answer()
@router_functions_bybit_trade.callback_query(lambda c: c.data and c.data.startswith('entry_order_type:'))
async def entry_order_type_callback(callback: CallbackQuery, state: FSMContext) -> None:
"""
Обработка выбора типа входа в позицию.
Если Limit, запрашивает цену лимитного ордера.
Если Market — обновляет настройки.
"""
order_type = callback.data.split(':')[1]
if order_type not in ['Market', 'Limit']:
await callback.answer("Ошибка выбора", show_alert=True)
return
if order_type == 'Limit':
await state.set_state(state_limit_price.price)
await callback.message.answer("Введите цену лимитного ордера:", reply_markup=inline_markup.cancel)
await callback.answer()
return
try:
await state.update_data(entry_order_type=order_type)
await rq.update_entry_order_type(callback.from_user.id, order_type)
await callback.message.answer(f"Выбран тип входа в позицию: {order_type}",
reply_markup=inline_markup.start_trading_markup)
await callback.answer()
except Exception as e:
logger.error(f"Произошла ошибка при обновлении типа входа в позицию: {e}")
await callback.message.answer("Произошла ошибка при обновлении типа входа в позицию",
reply_markup=inline_markup.back_to_main)
await state.clear()
@router_functions_bybit_trade.message(state_limit_price.price)
async def set_limit_price(message: Message, state: FSMContext) -> None:
"""
Обрабатывает ввод цены лимитного ордера, проверяет формат и сохраняет настройки.
"""
try:
price = float(message.text)
if price <= 0:
await message.answer("Цена должна быть положительным числом. Попробуйте снова.",
reply_markup=inline_markup.cancel)
return
except ValueError:
await message.answer("Некорректный формат цены. Введите число.", reply_markup=inline_markup.cancel)
return
await state.update_data(entry_order_type='Limit', limit_price=price)
await rq.update_entry_order_type(message.from_user.id, 'Limit')
await rq.update_limit_price(message.from_user.id, price)
await message.answer(f"Цена лимитного ордера установлена: {price}", reply_markup=inline_markup.start_trading_markup)
await state.clear()
@router_functions_bybit_trade.callback_query(F.data == "clb_start_chatbot_trading") @router_functions_bybit_trade.callback_query(F.data == "clb_start_chatbot_trading")
async def start_trading_process(callback: CallbackQuery) -> None: async def start_trading_process(callback: CallbackQuery) -> None:
""" """
@@ -196,11 +131,18 @@ async def start_trading_process(callback: CallbackQuery) -> None:
tg_id = callback.from_user.id tg_id = callback.from_user.id
message = callback.message message = callback.message
data_main_stgs = await rq.get_user_main_settings(tg_id) data_main_stgs = await rq.get_user_main_settings(tg_id)
# data_risk_stgs = await rq.get_user_risk_management_settings(tg_id)
# client = await get_bybit_client(tg_id)
symbol = await rq.get_symbol(tg_id) symbol = await rq.get_symbol(tg_id)
margin_mode = data_main_stgs.get('margin_type', 'Isolated') margin_mode = data_main_stgs.get('margin_type', 'Isolated')
trading_mode = data_main_stgs.get('trading_mode') trading_mode = data_main_stgs.get('trading_mode')
starting_quantity = safe_float(data_main_stgs.get('starting_quantity')) starting_quantity = safe_float(data_main_stgs.get('starting_quantity'))
switch_state = data_main_stgs.get("switch_state", "По направлению") switch_state = data_main_stgs.get("switch_state", "По направлению")
# martingale_factor = safe_float(data_main_stgs.get('martingale_factor'))
# max_martingale_steps = int(data_main_stgs.get("maximal_quantity", 0))
# commission_fee = data_risk_stgs.get("commission_fee")
# fee_info = client.get_fee_rates(category='linear', symbol=symbol)
if trading_mode == 'Switch': if trading_mode == 'Switch':
if switch_state == "По направлению": if switch_state == "По направлению":
@@ -221,7 +163,33 @@ async def start_trading_process(callback: CallbackQuery) -> None:
reply_markup=inline_markup.back_to_main) reply_markup=inline_markup.back_to_main)
return return
# if commission_fee == "Да":
# commission_fee_percent = safe_float(fee_info['result']['list'][0]['takerFeeRate'])
# else:
# commission_fee_percent = 0.0
# total_budget = await calculate_total_budget(
# starting_quantity=starting_quantity,
# martingale_factor=martingale_factor,
# max_steps=max_martingale_steps,
# commission_fee_percent=commission_fee_percent,
# )
# balance = await balance_g.get_balance(tg_id, message)
# if safe_float(balance) < total_budget:
# logger.error(
# f"Недостаточно средств для серии из {max_martingale_steps} шагов с текущими параметрами. "
# f"Требуемый бюджет: {total_budget:.2f} USDT, доступно: {balance} USDT."
# )
# await message.answer(
# f"Недостаточно средств для серии из {max_martingale_steps} шагов с текущими параметрами. "
# f"Требуемый бюджет: {total_budget:.2f} USDT, доступно: {balance} USDT.",
# reply_markup=inline_markup.back_to_main,
# )
# return
await message.answer("Начинаю торговлю с использованием текущих настроек...") await message.answer("Начинаю торговлю с использованием текущих настроек...")
await rq.update_trigger(tg_id=tg_id, trigger="Автоматический")
timer_data = await rq.get_user_timer(tg_id) timer_data = await rq.get_user_timer(tg_id)
if isinstance(timer_data, dict): if isinstance(timer_data, dict):
@@ -259,6 +227,7 @@ async def cancel_start_trading(callback: CallbackQuery):
pass pass
user_trade_tasks.pop(tg_id, None) user_trade_tasks.pop(tg_id, None)
await rq.update_user_timer(tg_id, minutes=0) await rq.update_user_timer(tg_id, minutes=0)
await rq.update_trigger(tg_id, "Ручной")
await callback.message.answer("Запуск торговли отменён.", reply_markup=inline_markup.back_to_main) await callback.message.answer("Запуск торговли отменён.", reply_markup=inline_markup.back_to_main)
await callback.message.edit_reply_markup(reply_markup=None) await callback.message.edit_reply_markup(reply_markup=None)
else: else:
@@ -272,10 +241,10 @@ async def show_my_trades(callback: CallbackQuery) -> None:
""" """
await callback.answer() await callback.answer()
try: try:
await callback.message.answer(f"Выберите тип сделки:", await callback.message.answer("Выберите тип сделки:",
reply_markup=inline_markup.my_deals_select_markup) reply_markup=inline_markup.my_deals_select_markup)
except Exception as e: except Exception as e:
logger.error(f"Произошла ошибка при выборе типа сделки: {e}") logger.error("Произошла ошибка при выборе типа сделки: %s", e)
@router_functions_bybit_trade.callback_query(F.data == "clb_open_deals") @router_functions_bybit_trade.callback_query(F.data == "clb_open_deals")
@@ -288,7 +257,7 @@ async def show_my_trades_callback(callback: CallbackQuery):
try: try:
await get_active_positions(callback.from_user.id, message=callback.message) await get_active_positions(callback.from_user.id, message=callback.message)
except Exception as e: except Exception as e:
logger.error(f"Произошла ошибка при выборе сделки: {e}") logger.error("Произошла ошибка при выборе сделки: %s", e)
await callback.message.answer("Произошла ошибка при выборе сделки", reply_markup=inline_markup.back_to_main) await callback.message.answer("Произошла ошибка при выборе сделки", reply_markup=inline_markup.back_to_main)
@@ -304,7 +273,7 @@ async def show_deal_callback(callback_query: CallbackQuery) -> None:
tg_id = callback_query.from_user.id tg_id = callback_query.from_user.id
await get_active_positions_by_symbol(tg_id, symbol, message=callback_query.message) await get_active_positions_by_symbol(tg_id, symbol, message=callback_query.message)
except Exception as e: except Exception as e:
logger.error(f"Произошла ошибка при выборе сделки: {e}") logger.error("Произошла ошибка при выборе сделки: %s", e)
await callback_query.message.answer("Произошла ошибка при выборе сделки", await callback_query.message.answer("Произошла ошибка при выборе сделки",
reply_markup=inline_markup.back_to_main) reply_markup=inline_markup.back_to_main)
@@ -319,7 +288,7 @@ async def show_my_orders_callback(callback: CallbackQuery) -> None:
try: try:
await get_active_orders(callback.from_user.id, message=callback.message) await get_active_orders(callback.from_user.id, message=callback.message)
except Exception as e: except Exception as e:
logger.error(f"Произошла ошибка при выборе ордера: {e}") logger.error("Произошла ошибка при выборе ордера: %s", e)
await callback.message.answer("Произошла ошибка при выборе ордера", reply_markup=inline_markup.back_to_main) await callback.message.answer("Произошла ошибка при выборе ордера", reply_markup=inline_markup.back_to_main)
@@ -335,7 +304,7 @@ async def show_limit_callback(callback_query: CallbackQuery) -> None:
tg_id = callback_query.from_user.id tg_id = callback_query.from_user.id
await get_active_orders_by_symbol(tg_id, symbol, message=callback_query.message) await get_active_orders_by_symbol(tg_id, symbol, message=callback_query.message)
except Exception as e: except Exception as e:
logger.error(f"Произошла ошибка при выборе сделки: {e}") logger.error("Произошла ошибка при выборе сделки: %s", e)
await callback_query.message.answer("Произошла ошибка при выборе сделки", await callback_query.message.answer("Произошла ошибка при выборе сделки",
reply_markup=inline_markup.back_to_main) reply_markup=inline_markup.back_to_main)
@@ -505,9 +474,13 @@ async def stop_immediately(callback: CallbackQuery):
Останавливает торговлю немедленно. Останавливает торговлю немедленно.
""" """
tg_id = callback.from_user.id tg_id = callback.from_user.id
symbol = await rq.get_symbol(tg_id)
await close_user_trade(tg_id, symbol)
await rq.update_trigger(tg_id, "Ручной") await rq.update_trigger(tg_id, "Ручной")
await callback.message.answer("Автоматическая торговля остановлена.", reply_markup=inline_markup.back_to_main) await rq.update_martingale_step(tg_id, 1)
await callback.message.answer("Торговля остановлена.", reply_markup=inline_markup.back_to_main)
await callback.answer() await callback.answer()
@@ -543,8 +516,13 @@ async def process_stop_delay(message: Message, state: FSMContext):
await message.answer(f"Торговля будет остановлена через {delay_minutes} минут.", await message.answer(f"Торговля будет остановлена через {delay_minutes} минут.",
reply_markup=inline_markup.back_to_main) reply_markup=inline_markup.back_to_main)
await asyncio.sleep(delay_seconds) await asyncio.sleep(delay_seconds)
symbol = await rq.get_symbol(tg_id)
await close_user_trade(tg_id, symbol)
await rq.update_trigger(tg_id, "Ручной") await rq.update_trigger(tg_id, "Ручной")
await message.answer("Автоматическая торговля остановлена.", reply_markup=inline_markup.back_to_main) await rq.update_martingale_step(tg_id, 1)
await message.answer("Торговля остановлена.", reply_markup=inline_markup.back_to_main)
await state.clear() await state.clear()
@@ -554,6 +532,9 @@ async def cancel(callback: CallbackQuery, state: FSMContext) -> None:
""" """
Отменяет текущее состояние FSM и сообщает пользователю об отмене. Отменяет текущее состояние FSM и сообщает пользователю об отмене.
""" """
try:
await state.clear() await state.clear()
await callback.message.answer("Отменено!", reply_markup=inline_markup.back_to_main) await callback.message.answer("Отменено!", reply_markup=inline_markup.back_to_main)
await callback.answer() await callback.answer()
except Exception as e:
logger.error("Ошибка при обработке отмены: %s", e)

View File

@@ -21,6 +21,9 @@ class state_limit_price(StatesGroup):
"""FSM состояние для установки лимита.""" """FSM состояние для установки лимита."""
price = State() price = State()
class state_trigger_price(StatesGroup):
"""FSM состояние для установки лимита."""
price = State()
class CloseTradeTimerState(StatesGroup): class CloseTradeTimerState(StatesGroup):
"""FSM состояние ожидания задержки перед закрытием сделки.""" """FSM состояние ожидания задержки перед закрытием сделки."""

View File

@@ -35,8 +35,6 @@ special_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
back_btn_to_main back_btn_to_main
]) ])
connect_bybit_api_markup = InlineKeyboardMarkup(inline_keyboard=[ connect_bybit_api_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text="Подключить Bybit", callback_data='clb_new_user_connect_bybit_api')] [InlineKeyboardButton(text="Подключить Bybit", callback_data='clb_new_user_connect_bybit_api')]
]) ])
@@ -45,7 +43,7 @@ trading_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text="Настройки", callback_data='clb_settings_message')], [InlineKeyboardButton(text="Настройки", callback_data='clb_settings_message')],
[InlineKeyboardButton(text="Мои сделки", callback_data='clb_my_deals')], [InlineKeyboardButton(text="Мои сделки", callback_data='clb_my_deals')],
[InlineKeyboardButton(text="Указать торговую пару", callback_data='clb_update_trading_pair')], [InlineKeyboardButton(text="Указать торговую пару", callback_data='clb_update_trading_pair')],
[InlineKeyboardButton(text="Начать торговать", callback_data='clb_update_entry_type')], [InlineKeyboardButton(text="Начать торговать", callback_data='clb_start_chatbot_trading')],
[InlineKeyboardButton(text="Остановить торговлю", callback_data='clb_stop_trading')], [InlineKeyboardButton(text="Остановить торговлю", callback_data='clb_stop_trading')],
]) ])
@@ -61,14 +59,12 @@ cancel = InlineKeyboardMarkup(inline_keyboard=[
entry_order_type_markup = InlineKeyboardMarkup( entry_order_type_markup = InlineKeyboardMarkup(
inline_keyboard=[ inline_keyboard=[
[ [
InlineKeyboardButton(text="Market (текущая цена)", callback_data="entry_order_type:Market"), InlineKeyboardButton(text="Маркет", callback_data="entry_order_type:Market"),
InlineKeyboardButton(text="Limit (фиксированная цена)", callback_data="entry_order_type:Limit"), InlineKeyboardButton(text="Лимит", callback_data="entry_order_type:Limit"),
], back_btn_to_main ], back_btn_to_main
] ]
) )
back_to_main = InlineKeyboardMarkup(inline_keyboard=[ back_to_main = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text="На главную", callback_data='back_to_main')], [InlineKeyboardButton(text="На главную", callback_data='back_to_main')],
]) ])
@@ -79,7 +75,7 @@ main_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
InlineKeyboardButton(text='Тип маржи', callback_data='clb_change_margin_type')], InlineKeyboardButton(text='Тип маржи', callback_data='clb_change_margin_type')],
[InlineKeyboardButton(text='Размер кредитного плеча', callback_data='clb_change_size_leverage'), [InlineKeyboardButton(text='Размер кредитного плеча', callback_data='clb_change_size_leverage'),
InlineKeyboardButton(text='Начальная ставка', callback_data='clb_change_starting_quantity')], InlineKeyboardButton(text='Ставка', callback_data='clb_change_starting_quantity')],
[InlineKeyboardButton(text='Коэффициент Мартингейла', callback_data='clb_change_martingale_factor'), [InlineKeyboardButton(text='Коэффициент Мартингейла', callback_data='clb_change_martingale_factor'),
InlineKeyboardButton(text='Сбросить шаги Мартингейла', callback_data='clb_change_martingale_reset')], InlineKeyboardButton(text='Сбросить шаги Мартингейла', callback_data='clb_change_martingale_reset')],
@@ -101,8 +97,10 @@ risk_management_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
]) ])
condition_settings_markup = InlineKeyboardMarkup(inline_keyboard=[ condition_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text='Режим торговли', callback_data='clb_change_mode'), [InlineKeyboardButton(text='Таймер', callback_data='clb_change_timer'),
InlineKeyboardButton(text='Таймер', callback_data='clb_change_timer')], InlineKeyboardButton(text='Тип позиции', callback_data='clb_update_entry_type')],
[InlineKeyboardButton(text='Триггер цена', callback_data='clb_change_trigger_price'),
InlineKeyboardButton(text='Лимит цена', callback_data='clb_change_limit_price')],
# #
# [InlineKeyboardButton(text='Фильтр волатильности', callback_data='clb_change_filter_volatility'), # [InlineKeyboardButton(text='Фильтр волатильности', callback_data='clb_change_filter_volatility'),
# InlineKeyboardButton(text='Внешние сигналы', callback_data='clb_change_external_cues')], # InlineKeyboardButton(text='Внешние сигналы', callback_data='clb_change_external_cues')],
@@ -116,6 +114,11 @@ condition_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
back_btn_to_main back_btn_to_main
]) ])
back_to_condition_settings = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text='Назад', callback_data='clb_change_condition_settings')],
back_btn_to_main
])
additional_settings_markup = InlineKeyboardMarkup(inline_keyboard=[ additional_settings_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text='Сохранить шаблон', callback_data='clb_change_save_pattern'), [InlineKeyboardButton(text='Сохранить шаблон', callback_data='clb_change_save_pattern'),
InlineKeyboardButton(text='Автозапуск', callback_data='clb_change_auto_start')], InlineKeyboardButton(text='Автозапуск', callback_data='clb_change_auto_start')],
@@ -162,11 +165,12 @@ buttons_on_off_markup = InlineKeyboardMarkup(inline_keyboard=[ # ИЗМЕНИТ
]) ])
my_deals_select_markup = InlineKeyboardMarkup(inline_keyboard=[ my_deals_select_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text='Открытые сделки', callback_data="clb_open_deals"), [InlineKeyboardButton(text='Позиции', callback_data="clb_open_deals"),
InlineKeyboardButton(text='Лимитные ордера', callback_data="clb_open_orders")], InlineKeyboardButton(text='Ордера', callback_data="clb_open_orders")],
back_btn_to_main back_btn_to_main
]) ])
def create_trades_inline_keyboard(trades): def create_trades_inline_keyboard(trades):
builder = InlineKeyboardBuilder() builder = InlineKeyboardBuilder()
for trade in trades: for trade in trades:
@@ -174,6 +178,7 @@ def create_trades_inline_keyboard(trades):
builder.adjust(2) builder.adjust(2)
return builder.as_markup() return builder.as_markup()
def create_trades_inline_keyboard_limits(trades): def create_trades_inline_keyboard_limits(trades):
builder = InlineKeyboardBuilder() builder = InlineKeyboardBuilder()
for trade in trades: for trade in trades:
@@ -190,12 +195,14 @@ def create_close_deal_markup(symbol: str) -> InlineKeyboardMarkup:
back_btn_to_main back_btn_to_main
]) ])
def create_close_limit_markup(symbol: str) -> InlineKeyboardMarkup: def create_close_limit_markup(symbol: str) -> InlineKeyboardMarkup:
return InlineKeyboardMarkup(inline_keyboard=[ return InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text="Закрыть лимитный ордер", callback_data=f"close_limit:{symbol}")], [InlineKeyboardButton(text="Закрыть ордер", callback_data=f"close_limit:{symbol}")],
back_btn_to_main back_btn_to_main
]) ])
timer_markup = InlineKeyboardMarkup(inline_keyboard=[ timer_markup = InlineKeyboardMarkup(inline_keyboard=[
[InlineKeyboardButton(text="Установить таймер", callback_data="clb_set_timer")], [InlineKeyboardButton(text="Установить таймер", callback_data="clb_set_timer")],
[InlineKeyboardButton(text="Удалить таймер", callback_data="clb_delete_timer")], [InlineKeyboardButton(text="Удалить таймер", callback_data="clb_delete_timer")],

View File

@@ -11,7 +11,7 @@ from sqlalchemy import select, insert
logging.config.dictConfig(LOGGING_CONFIG) logging.config.dictConfig(LOGGING_CONFIG)
logger = logging.getLogger("models") logger = logging.getLogger("models")
engine = create_async_engine(url='sqlite+aiosqlite:///db.sqlite3') engine = create_async_engine(url='sqlite+aiosqlite:///data/db.sqlite3')
async_session = async_sessionmaker(engine) async_session = async_sessionmaker(engine)
@@ -148,12 +148,15 @@ class User_Main_Settings(Base):
switch_state = mapped_column(String(10), default='По направлению') switch_state = mapped_column(String(10), default='По направлению')
size_leverage = mapped_column(Integer(), default=1) size_leverage = mapped_column(Integer(), default=1)
starting_quantity = mapped_column(Integer(), default=1) starting_quantity = mapped_column(Integer(), default=1)
base_quantity = mapped_column(Integer(), default=1)
martingale_factor = mapped_column(Integer(), default=1) martingale_factor = mapped_column(Integer(), default=1)
martingale_step = mapped_column(Integer(), default=1) martingale_step = mapped_column(Integer(), default=1)
maximal_quantity = mapped_column(Integer(), default=10) maximal_quantity = mapped_column(Integer(), default=10)
entry_order_type = mapped_column(String(10), default='Market') entry_order_type = mapped_column(String(10), default='Market')
limit_order_price = mapped_column(Numeric(18, 15), nullable=True) limit_order_price = mapped_column(Numeric(18, 15), nullable=True, default=0)
trigger_price = mapped_column(Numeric(18, 15), nullable=True, default=0)
last_side = mapped_column(String(10), default='Buy') last_side = mapped_column(String(10), default='Buy')
trading_start_stop = mapped_column(Integer(), default=0)
class User_Risk_Management_Settings(Base): class User_Risk_Management_Settings(Base):
@@ -304,3 +307,10 @@ async def async_main():
if not result.first(): if not result.first():
logger.info("Заполение таблицы последнего направления") logger.info("Заполение таблицы последнего направления")
await conn.execute(User_Main_Settings.__table__.insert().values(last_side=side)) await conn.execute(User_Main_Settings.__table__.insert().values(last_side=side))
order_type = ['Limit', 'Market']
for typ in order_type:
result = await conn.execute(select(User_Main_Settings).where(User_Main_Settings.entry_order_type == typ))
if not result.first():
logger.info("Заполение таблицы типов ордеров")
await conn.execute(User_Main_Settings.__table__.insert().values(entry_order_type=typ))

View File

@@ -318,8 +318,11 @@ async def get_user_main_settings(tg_id):
'maximal_quantity': user.maximal_quantity, 'maximal_quantity': user.maximal_quantity,
'entry_order_type': user.entry_order_type, 'entry_order_type': user.entry_order_type,
'limit_order_price': user.limit_order_price, 'limit_order_price': user.limit_order_price,
'trigger_price': user.trigger_price,
'martingale_step': user.martingale_step, 'martingale_step': user.martingale_step,
'last_side': user.last_side, 'last_side': user.last_side,
'trading_start_stop': user.trading_start_stop,
'base_quantity': user.base_quantity,
} }
return data return data
@@ -368,15 +371,23 @@ async def update_size_leverange(tg_id, num):
async def update_starting_quantity(tg_id, num): async def update_starting_quantity(tg_id, num):
"""Обновить размер левеража пользователя.""" """Обновить размер начальной ставки пользователя."""
async with async_session() as session: async with async_session() as session:
await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(starting_quantity=num)) await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(starting_quantity=num))
await session.commit() await session.commit()
async def update_base_quantity(tg_id, num):
"""Обновить размер следующей ставки пользователя."""
async with async_session() as session:
await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(base_quantity=num))
await session.commit()
async def update_martingale_factor(tg_id, num): async def update_martingale_factor(tg_id, num):
"""Обновить размер левеража пользователя.""" """Обновить шаг мартингейла пользователя."""
async with async_session() as session: async with async_session() as session:
await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(martingale_factor=num)) await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(martingale_factor=num))
@@ -384,7 +395,7 @@ async def update_martingale_factor(tg_id, num):
async def update_maximal_quantity(tg_id, num): async def update_maximal_quantity(tg_id, num):
"""Обновить размер левеража пользователя.""" """Обновить размер максимальной ставки пользователя."""
async with async_session() as session: async with async_session() as session:
await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(maximal_quantity=num)) await session.execute(update(UMS).where(UMS.tg_id == tg_id).values(maximal_quantity=num))
@@ -428,6 +439,31 @@ async def update_entry_order_type(tg_id, order_type):
await session.commit() await session.commit()
async def update_trigger_price(tg_id, price):
"""Обновить условную цену пользователя."""
async with async_session() as session:
await session.execute(
update(UMS)
.where(UMS.tg_id == tg_id)
.values(trigger_price=str(price))
)
await session.commit()
async def get_trigger_price(tg_id):
"""Получить условную цену пользователя как float, либо None."""
async with async_session() as session:
result = await session.execute(
select(UMS.trigger_price)
.where(UMS.tg_id == tg_id)
)
price = result.scalar_one_or_none()
if price:
try:
return float(price)
except ValueError:
return None
return None
async def get_limit_price(tg_id): async def get_limit_price(tg_id):
"""Получить лимитную цену пользователя как float, либо None.""" """Получить лимитную цену пользователя как float, либо None."""
async with async_session() as session: async with async_session() as session:

View File

@@ -5,7 +5,7 @@ from aiogram.types import Message, CallbackQuery
from aiogram.fsm.context import FSMContext from aiogram.fsm.context import FSMContext
import app.telegram.database.requests as rq import app.telegram.database.requests as rq
from app.states.States import condition_settings from app.states.States import condition_settings
from app.states.States import state_limit_price, state_update_entry_type, state_trigger_price
from logger_helper.logger_helper import LOGGING_CONFIG from logger_helper.logger_helper import LOGGING_CONFIG
logging.config.dictConfig(LOGGING_CONFIG) logging.config.dictConfig(LOGGING_CONFIG)
@@ -23,42 +23,28 @@ async def reg_new_user_default_condition_settings(id):
async def main_settings_message(id, message): async def main_settings_message(id, message):
data = await rq.get_user_main_settings(id)
entry_order_type = data['entry_order_type']
if entry_order_type == "Market":
entry_order_type_rus = "Маркет"
elif entry_order_type == "Limit":
entry_order_type_rus = "Лимит"
else:
entry_order_type_rus = "Условный"
trigger_price = data['trigger_price'] or 0.0
limit_price = data['limit_order_price'] or 0.0
tg_id = id
trigger = await rq.get_for_registration_trigger(tg_id)
text = f""" <b>Условия запуска</b> text = f""" <b>Условия запуска</b>
<b>- Режим торговли:</b> {trigger}
<b>- Таймер: </b> установить таймер / удалить таймер <b>- Таймер: </b> установить таймер / удалить таймер
<b>- Тип позиции:</b> {entry_order_type_rus}
<b>- Триггер цена: </b> {trigger_price:,.4f}
<b>- Лимит цена: </b> {limit_price:,.4f}
""" """
await message.answer(text=text, parse_mode='html', reply_markup=inline_markup.condition_settings_markup) await message.answer(text=text, parse_mode='html', reply_markup=inline_markup.condition_settings_markup)
async def trigger_message(id, message, state: FSMContext):
await state.set_state(condition_settings.trigger)
text = '''
<b>- Автоматический:</b> торговля будет происходить в рамках серии ставок.
<b>- Ручной:</b> торговля будет происходить только в ручном режиме.
<em>- Выберите тип триггера:</em>'''
await message.answer(text=text, parse_mode='html', reply_markup=inline_markup.trigger_markup)
@condition_settings_router.callback_query(F.data == "clb_trigger_manual")
async def trigger_manual_callback(callback: CallbackQuery, state: FSMContext):
await state.set_state(condition_settings.trigger)
await rq.update_trigger(tg_id=callback.from_user.id, trigger="Ручной")
await main_settings_message(callback.from_user.id, callback.message)
await callback.answer()
@condition_settings_router.callback_query(F.data == "clb_trigger_auto")
async def trigger_manual_callback(callback: CallbackQuery, state: FSMContext):
await state.set_state(condition_settings.trigger)
await rq.update_trigger(tg_id=callback.from_user.id, trigger="Автоматический")
await main_settings_message(callback.from_user.id, callback.message)
await callback.answer()
async def timer_message(id, message: Message, state: FSMContext): async def timer_message(id, message: Message, state: FSMContext):
await state.set_state(condition_settings.timer) await state.set_state(condition_settings.timer)
@@ -105,6 +91,85 @@ async def delete_timer_callback(callback: CallbackQuery, state: FSMContext):
await callback.answer() await callback.answer()
@condition_settings_router.callback_query(F.data == 'clb_update_entry_type')
async def update_entry_type_message(callback: CallbackQuery, state: FSMContext) -> None:
"""
Запрашивает у пользователя тип входа в позицию (Market или Limit).
"""
await state.set_state(state_update_entry_type.entry_type)
await callback.message.answer("Выберите тип входа в позицию:", reply_markup=inline_markup.entry_order_type_markup)
await callback.answer()
@condition_settings_router.callback_query(lambda c: c.data and c.data.startswith('entry_order_type:'))
async def entry_order_type_callback(callback: CallbackQuery, state: FSMContext) -> None:
"""
Обработка выбора типа входа в позицию.
Если Limit, запрашивает цену лимитного ордера.
Если Market — обновляет настройки.
"""
order_type = callback.data.split(':')[1]
if order_type not in ['Market', 'Limit']:
await callback.answer("Ошибка выбора", show_alert=True)
return
if order_type == 'Limit':
order_type_rus = 'Лимит'
else:
order_type_rus = 'Маркет'
try:
await state.update_data(entry_order_type=order_type)
await rq.update_entry_order_type(callback.from_user.id, order_type)
await callback.message.answer(f"Выбран тип входа {order_type_rus}",
reply_markup=inline_markup.back_to_condition_settings)
await callback.answer()
except Exception as e:
logger.error("Произошла ошибка при обновлении типа входа в позицию: %s", e)
await callback.message.answer("Произошла ошибка при обновлении типа входа в позицию",
reply_markup=inline_markup.back_to_condition_settings)
await state.clear()
@condition_settings_router.callback_query(F.data == 'clb_change_limit_price')
async def set_limit_price_callback(callback: CallbackQuery, state: FSMContext) -> None:
await state.set_state(state_limit_price.price)
await callback.message.answer("Введите цену лимитного ордера:", reply_markup=inline_markup.cancel)
await callback.answer()
@condition_settings_router.message(state_limit_price.price)
async def process_limit_price_input(message: Message, state: FSMContext) -> None:
try:
price = float(message.text)
await state.update_data(price=price)
await rq.update_limit_price(tg_id=message.from_user.id, price=price)
await message.answer(f"Цена лимитного ордера установлена: {price}", reply_markup=inline_markup.back_to_condition_settings)
await state.clear()
except ValueError:
await message.reply("Пожалуйста, введите корректную цену.")
@condition_settings_router.callback_query(F.data == 'clb_change_trigger_price')
async def change_trigger_price_callback(callback: CallbackQuery, state: FSMContext) -> None:
await state.set_state(state_trigger_price.price)
await callback.message.answer("Введите цену триггера:", reply_markup=inline_markup.cancel)
await callback.answer()
@condition_settings_router.message(state_trigger_price.price)
async def process_trigger_price_input(message: Message, state: FSMContext) -> None:
try:
price = float(message.text)
await state.update_data(price=price)
await rq.update_trigger_price(tg_id=message.from_user.id, price=price)
await message.answer(f"Цена триггера установлена: {price}", reply_markup=inline_markup.back_to_condition_settings)
await state.clear()
except ValueError:
await message.reply("Пожалуйста, введите корректную цену.")
async def filter_volatility_message(message, state): async def filter_volatility_message(message, state):
text = '''Фильтр волатильности text = '''Фильтр волатильности

View File

@@ -2,11 +2,9 @@
import logging.config import logging.config
import app.telegram.Keyboards.inline_keyboards as inline_markup import app.telegram.Keyboards.inline_keyboards as inline_markup
from pybit.unified_trading import HTTP
import app.telegram.database.requests as rq import app.telegram.database.requests as rq
from aiogram.types import Message, CallbackQuery from aiogram.types import Message, CallbackQuery
from app.services.Bybit.functions.price_symbol import get_price
from app.services.Bybit.functions.Futures import safe_float, calculate_total_budget, get_bybit_client from app.services.Bybit.functions.Futures import safe_float, calculate_total_budget, get_bybit_client
from app.states.States import update_main_settings from app.states.States import update_main_settings
from logger_helper.logger_helper import LOGGING_CONFIG from logger_helper.logger_helper import LOGGING_CONFIG
@@ -40,9 +38,6 @@ async def main_settings_message(id, message):
starting_quantity = safe_float((data_main_stgs or {}).get('starting_quantity')) starting_quantity = safe_float((data_main_stgs or {}).get('starting_quantity'))
martingale_factor = safe_float((data_main_stgs or {}).get('martingale_factor')) martingale_factor = safe_float((data_main_stgs or {}).get('martingale_factor'))
fee_info = client.get_fee_rates(category='linear', symbol=symbol) fee_info = client.get_fee_rates(category='linear', symbol=symbol)
leverage = safe_float((data_main_stgs or {}).get('size_leverage'))
price = await get_price(tg_id, symbol=symbol)
entry_price = safe_float(price)
if commission_fee == "Да": if commission_fee == "Да":
commission_fee_percent = safe_float(fee_info['result']['list'][0]['takerFeeRate']) commission_fee_percent = safe_float(fee_info['result']['list'][0]['takerFeeRate'])
@@ -54,8 +49,6 @@ async def main_settings_message(id, message):
martingale_factor=martingale_factor, martingale_factor=martingale_factor,
max_steps=max_martingale_steps, max_steps=max_martingale_steps,
commission_fee_percent=commission_fee_percent, commission_fee_percent=commission_fee_percent,
leverage=leverage,
current_price=entry_price,
) )
await message.answer(f"""<b>Основные настройки</b> await message.answer(f"""<b>Основные настройки</b>
@@ -65,7 +58,7 @@ async def main_settings_message(id, message):
<b>- Направление последней сделки:</b> {data['last_side']} <b>- Направление последней сделки:</b> {data['last_side']}
<b>- Тип маржи:</b> {data['margin_type']} <b>- Тип маржи:</b> {data['margin_type']}
<b>- Размер кредитного плеча:</b> х{data['size_leverage']} <b>- Размер кредитного плеча:</b> х{data['size_leverage']}
<b>- Начальная ставка:</b> {data['starting_quantity']} <b>- Ставка:</b> {data['starting_quantity']}
<b>- Коэффициент мартингейла:</b> {data['martingale_factor']} <b>- Коэффициент мартингейла:</b> {data['martingale_factor']}
<b>- Текущий шаг:</b> {data['martingale_step']} <b>- Текущий шаг:</b> {data['martingale_step']}
<b>- Максимальное количество ставок в серии:</b> {data['maximal_quantity']} <b>- Максимальное количество ставок в серии:</b> {data['maximal_quantity']}
@@ -128,7 +121,8 @@ async def state_trading_mode(callback: CallbackQuery, state):
await state.clear() await state.clear()
except Exception as e: except Exception as e:
logger.error(e) logger.error("Ошибка при обновлении режима торговли: %s", e)
async def switch_mode_enabled_message(message, state): async def switch_mode_enabled_message(message, state):
@@ -235,8 +229,9 @@ async def state_martingale_factor(message: Message, state):
await state.clear() await state.clear()
else: else:
val = data['martingale_factor']
await message.answer( await message.answer(
f'⛔️ Ошибка: ваше значение ({data['martingale_factor']}) или выше лимита (100) или вы вводите неверные символы') f"⛔️ Ошибка: ваше значение ({val}) или выше лимита (100) или вы вводите неверные символы")
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
@@ -266,26 +261,7 @@ async def state_margin_type(callback: CallbackQuery, state):
callback_data = callback.data callback_data = callback.data
if callback_data in ['margin_type_isolated', 'margin_type_cross']: if callback_data in ['margin_type_isolated', 'margin_type_cross']:
tg_id = callback.from_user.id tg_id = callback.from_user.id
api_key = await rq.get_bybit_api_key(tg_id)
secret_key = await rq.get_bybit_secret_key(tg_id)
data_settings = await rq.get_user_main_settings(tg_id) data_settings = await rq.get_user_main_settings(tg_id)
symbol = await rq.get_symbol(tg_id)
client = HTTP(api_key=api_key, api_secret=secret_key)
try:
active_positions = client.get_positions(category='linear', settleCoin="USDT")
positions = active_positions.get('result', {}).get('list', [])
except Exception as e:
logger.error(f"error: {e}")
positions = []
for pos in positions:
size = pos.get('size')
if float(size) > 0:
await callback.answer(
"⚠️ Маржинальный режим нельзя менять при открытой позиции"
)
return
try: try:
match callback.data: match callback.data:
@@ -306,7 +282,7 @@ async def state_margin_type(callback: CallbackQuery, state):
await state.clear() await state.clear()
except Exception as e: except Exception as e:
logger.error(f"error: {e}") logger.error("Ошибка при изменении типа маржи: %s", e)
else: else:
await callback.answer() await callback.answer()
await main_settings_message(callback.from_user.id, callback.message) await main_settings_message(callback.from_user.id, callback.message)
@@ -317,7 +293,7 @@ async def state_margin_type(callback: CallbackQuery, state):
async def starting_quantity_message(message, state): async def starting_quantity_message(message, state):
await state.set_state(update_main_settings.starting_quantity) await state.set_state(update_main_settings.starting_quantity)
await message.edit_text("Введите <b>начальную ставку:</b>", parse_mode='html', await message.edit_text("Введите <b>ставку:</b>", parse_mode='html',
reply_markup=inline_markup.back_btn_list_settings_markup) reply_markup=inline_markup.back_btn_list_settings_markup)
@@ -332,11 +308,12 @@ async def state_starting_quantity(message: Message, state):
await message.answer(f"✅ Изменено: {data_settings['starting_quantity']}{data['starting_quantity']}") await message.answer(f"✅ Изменено: {data_settings['starting_quantity']}{data['starting_quantity']}")
await rq.update_starting_quantity(message.from_user.id, data['starting_quantity']) await rq.update_starting_quantity(message.from_user.id, data['starting_quantity'])
await rq.update_base_quantity(tg_id=message.from_user.id, num=data['starting_quantity'])
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
await state.clear() await state.clear()
else: else:
await message.answer(f'⛔️ Ошибка: вы вводите неверные символы') await message.answer("⛔️ Ошибка: вы вводите неверные символы")
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
@@ -344,7 +321,7 @@ async def state_starting_quantity(message: Message, state):
async def maximum_quantity_message(message, state): async def maximum_quantity_message(message, state):
await state.set_state(update_main_settings.maximal_quantity) await state.set_state(update_main_settings.maximal_quantity)
await message.edit_text("Введите <b>максимальное количество серии ставок:</b>", parse_mode='html', await message.edit_text("Введите максимальное количество серии ставок:",
reply_markup=inline_markup.back_btn_list_settings_markup) reply_markup=inline_markup.back_btn_list_settings_markup)
@@ -363,7 +340,9 @@ async def state_maximal_quantity(message: Message, state):
await state.clear() await state.clear()
else: else:
val = data['maximal_quantity']
await message.answer( await message.answer(
f'⛔️ Ошибка: ваше значение ({data['maximal_quantity']}) или выше лимита (100) или вы вводите неверные символы') f'⛔️ Ошибка: ваше значение ({val}) или выше лимита (100) или вы вводите неверные символы')
logger.error(f'⛔️ Ошибка: ваше значение ({val}) или выше лимита (100) или вы вводите неверные символы')
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)

View File

@@ -56,8 +56,9 @@ async def state_price_profit(message: Message, state):
await state.clear() await state.clear()
else: else:
val = data['price_profit']
await message.answer( await message.answer(
f'⛔️ Ошибка: ваше значение ({data['price_profit']}%) или выше лимита (100) или вы вводите неверные символы') f'⛔️ Ошибка: ваше значение ({val}%) или выше лимита (100) или вы вводите неверные символы')
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
@@ -107,8 +108,9 @@ async def state_price_loss(message: Message, state):
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
await state.clear() await state.clear()
else: else:
val = data['price_loss']
await message.answer( await message.answer(
f'⛔️ Ошибка: ваше значение ({data["price_loss"]}%) выше лимита (100) или содержит неверные символы') f'⛔️ Ошибка: ваше значение ({val}%) выше лимита (100) или содержит неверные символы')
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)
@@ -135,8 +137,9 @@ async def state_max_risk_deal(message: Message, state):
await state.clear() await state.clear()
else: else:
val = data['max_risk_deal']
await message.answer( await message.answer(
f'⛔️ Ошибка: ваше значение ({data['max_risk_deal']}%) или выше лимита (100) или вы вводите неверные символы') f'⛔️ Ошибка: ваше значение ({val}%) или выше лимита (100) или вы вводите неверные символы')
await main_settings_message(message.from_user.id, message) await main_settings_message(message.from_user.id, message)

View File

@@ -211,7 +211,7 @@ async def clb_main_settings_msg(callback: CallbackQuery, state: FSMContext) -> N
case 'clb_change_maximum_quantity': case 'clb_change_maximum_quantity':
await func_main_settings.maximum_quantity_message(callback.message, state) await func_main_settings.maximum_quantity_message(callback.message, state)
except Exception as e: except Exception as e:
logger.error(f"Error callback in main_settings match-case: {e}") logger.error("Error callback in main_settings match-case: %s", e)
list_risk_management_settings = ['clb_change_price_profit', list_risk_management_settings = ['clb_change_price_profit',
@@ -243,7 +243,7 @@ async def clb_risk_management_settings_msg(callback: CallbackQuery, state: FSMCo
case 'commission_fee': case 'commission_fee':
await func_rmanagement_settings.commission_fee_message(callback.message, state) await func_rmanagement_settings.commission_fee_message(callback.message, state)
except Exception as e: except Exception as e:
logger.error(f"Error callback in risk_management match-case: {e}") logger.error("Error callback in risk_management match-case: %s", e)
list_condition_settings = ['clb_change_mode', list_condition_settings = ['clb_change_mode',
@@ -284,7 +284,7 @@ async def clb_condition_settings_msg(callback: CallbackQuery, state: FSMContext)
case 'clb_change_ai_analytics': case 'clb_change_ai_analytics':
await func_condition_settings.ai_analytics_message(callback.message, state) await func_condition_settings.ai_analytics_message(callback.message, state)
except Exception as e: except Exception as e:
logger.error(f"Error callback in main_settings match-case: {e}") logger.error("Error callback in main_settings match-case: %s", e)
list_additional_settings = ['clb_change_save_pattern', list_additional_settings = ['clb_change_save_pattern',
@@ -313,4 +313,4 @@ async def clb_additional_settings_msg(callback: CallbackQuery, state: FSMContext
case 'clb_change_notifications': case 'clb_change_notifications':
await func_additional_settings.notifications_message(callback.message, state) await func_additional_settings.notifications_message(callback.message, state)
except Exception as e: except Exception as e:
logger.error(f"Error callback in additional_settings match-case: {e}") logger.error("Error callback in additional_settings match-case: %s", e)

0
data/__init__.py Normal file
View File

View File

@@ -0,0 +1,17 @@
[Unit]
Description=Telegram chat-robot: @stcs_cryptobot
Wants=network.target
After=syslog.target network-online.target
[Service]
ExecStart=sudo -u www-data /usr/bin/python3 /var/www/stcs/BybitBot_API.py
PIDFile=/var/run/python/stcs.pid
RemainAfterExit=no
RuntimeMaxSec=3600s
Restart=always
RestartSec=5s
[Install]
WantedBy=multi-user.target

View File

@@ -1,35 +0,0 @@
2025-08-23 12:57:26 - main - INFO - Bot is off
2025-08-23 13:04:01 - main - INFO - Bot is off
2025-08-23 13:25:04 - main - INFO - Bot is off
2025-08-23 13:26:24 - main - INFO - Bot is off
2025-08-23 13:28:36 - main - INFO - Bot is off
2025-08-23 13:29:29 - main - INFO - Bot is off
2025-08-23 13:30:48 - main - INFO - Bot is off
2025-08-23 13:31:43 - main - INFO - Bot is off
2025-08-23 13:33:10 - main - INFO - Bot is off
2025-08-23 13:34:59 - main - INFO - Bot is off
2025-08-23 13:36:15 - main - INFO - Bot is off
2025-08-23 13:49:17 - main - INFO - Bot is off
2025-08-23 13:50:22 - main - INFO - Bot is on
2025-08-23 13:51:30 - main - INFO - Bot is off
2025-08-23 13:51:37 - main - INFO - Bot is on
2025-08-23 13:52:12 - main - INFO - Bot is off
2025-08-23 13:57:48 - main - INFO - Bot is on
2025-08-23 14:05:36 - main - INFO - Bot is off
2025-08-23 14:05:43 - main - INFO - Bot is on
2025-08-23 14:06:03 - main - INFO - Bot is off
2025-08-23 14:06:46 - main - INFO - Bot is on
2025-08-23 14:07:04 - requests - INFO - Bybit был успешно подключен
2025-08-23 14:07:43 - requests - INFO - Новый пользователь был добавлен в бд
2025-08-23 14:07:43 - requests - INFO - Основные настройки нового пользователя были заполнены
2025-08-23 14:07:43 - requests - INFO - Риск-Менеджмент настройки нового пользователя были заполнены
2025-08-23 14:07:43 - requests - INFO - Условные настройки нового пользователя были заполнены
2025-08-23 14:07:43 - requests - INFO - Дополнительные настройки нового пользователя были заполнены
2025-08-23 14:23:31 - main - INFO - Bot is off
2025-08-23 14:23:39 - main - INFO - Bot is on
2025-08-23 14:28:13 - main - INFO - Bot is off
2025-08-23 14:28:19 - main - INFO - Bot is on
2025-08-23 14:28:26 - requests - INFO - Получение риск-менеджмента настроек пользователя 899674724
2025-08-23 14:28:26 - requests - INFO - Получение риск-менеджмента настроек пользователя 899674724
2025-08-23 14:29:12 - requests - INFO - Получение риск-менеджмента настроек пользователя 899674724
2025-08-23 14:29:34 - main - INFO - Bot is off

View File

@@ -4,6 +4,7 @@ aiohappyeyeballs==2.6.1
aiohttp==3.12.15 aiohttp==3.12.15
aiosignal==1.4.0 aiosignal==1.4.0
aiosqlite==0.21.0 aiosqlite==0.21.0
alembic==1.16.5
annotated-types==0.7.0 annotated-types==0.7.0
attrs==25.3.0 attrs==25.3.0
black==25.1.0 black==25.1.0
@@ -20,7 +21,9 @@ greenlet==3.2.4
idna==3.10 idna==3.10
isort==6.0.1 isort==6.0.1
magic-filter==1.0.12 magic-filter==1.0.12
Mako==1.3.10
mando==0.7.1 mando==0.7.1
MarkupSafe==3.0.2
mccabe==0.7.0 mccabe==0.7.0
multidict==6.6.4 multidict==6.6.4
mypy_extensions==1.1.0 mypy_extensions==1.1.0