Compare commits
	
		
			5 Commits
		
	
	
		
			58a4c6af06
			...
			aebcc9dff2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| aebcc9dff2 | |||
| 
						 | 
					e2f9478971 | ||
| 
						 | 
					4f0668970f | ||
| 
						 | 
					4c9901c14a | ||
| 
						 | 
					17dba19078 | 
@@ -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,8 +102,6 @@ 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")
 | 
				
			||||||
@@ -126,8 +124,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 +136,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 +147,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
 | 
				
			||||||
    return None
 | 
					    return None
 | 
				
			||||||
@@ -172,7 +164,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 +181,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 +198,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)
 | 
				
			||||||
@@ -248,12 +233,12 @@ 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':
 | 
				
			||||||
 | 
					            side = data_main_stgs.get("last_side")
 | 
				
			||||||
 | 
					        else:
 | 
				
			||||||
 | 
					            side = "Buy" if trading_mode == "Long" else "Sell"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if trading_mode == 'Switch':
 | 
					        if pnl < 0:
 | 
				
			||||||
                side = data_main_stgs.get("last_side")
 | 
					 | 
				
			||||||
            else:
 | 
					 | 
				
			||||||
                side = "Buy" if trading_mode == "Long" else "Sell"
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
            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)
 | 
				
			||||||
@@ -262,6 +247,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 +261,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(
 | 
				
			||||||
                "❗️ Прибыль достигнута, шаг мартингейла сброшен."
 | 
					                "❗️ Прибыль достигнута, шаг мартингейла сброшен. "
 | 
				
			||||||
 | 
					                "Возврат к начальной ставке."
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -348,8 +337,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 +346,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 +396,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
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -465,6 +438,8 @@ async def open_position(
 | 
				
			|||||||
                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 +455,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 +473,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 +508,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)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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 пропущен.")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -10,6 +10,7 @@ from app.services.Bybit.functions.Futures import (close_user_trade, set_take_pro
 | 
				
			|||||||
                                                  get_active_positions_by_symbol, get_active_orders_by_symbol,
 | 
					                                                  get_active_positions_by_symbol, get_active_orders_by_symbol,
 | 
				
			||||||
                                                  get_active_positions, get_active_orders, cancel_all_tp_sl_orders,
 | 
					                                                  get_active_positions, get_active_orders, cancel_all_tp_sl_orders,
 | 
				
			||||||
                                                  open_position, close_trade_after_delay, safe_float,
 | 
					                                                  open_position, close_trade_after_delay, safe_float,
 | 
				
			||||||
 | 
					                                                  calculate_total_budget, get_bybit_client,
 | 
				
			||||||
                                                  )
 | 
					                                                  )
 | 
				
			||||||
from app.services.Bybit.functions.balance import get_balance
 | 
					from app.services.Bybit.functions.balance import get_balance
 | 
				
			||||||
import app.telegram.Keyboards.inline_keyboards as inline_markup
 | 
					import app.telegram.Keyboards.inline_keyboards as inline_markup
 | 
				
			||||||
@@ -17,6 +18,7 @@ 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_entry_type, state_update_symbol, state_limit_price,
 | 
				
			||||||
                               SetTP_SL_State, CloseTradeTimerState)
 | 
					                               SetTP_SL_State, CloseTradeTimerState)
 | 
				
			||||||
@@ -144,14 +146,14 @@ async def entry_order_type_callback(callback: CallbackQuery, state: FSMContext)
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    if order_type == 'Limit':
 | 
					    if order_type == 'Limit':
 | 
				
			||||||
        await state.set_state(state_limit_price.price)
 | 
					        await state.set_state(state_limit_price.price)
 | 
				
			||||||
        await callback.message.answer("Введите цену лимитного ордера:", reply_markup=inline_markup.cancel)
 | 
					        await callback.message.answer("Введите цену:", reply_markup=inline_markup.cancel)
 | 
				
			||||||
        await callback.answer()
 | 
					        await callback.answer()
 | 
				
			||||||
        return
 | 
					        return
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        await state.update_data(entry_order_type=order_type)
 | 
					        await state.update_data(entry_order_type=order_type)
 | 
				
			||||||
        await rq.update_entry_order_type(callback.from_user.id, order_type)
 | 
					        await rq.update_entry_order_type(callback.from_user.id, order_type)
 | 
				
			||||||
        await callback.message.answer(f"Выбран тип входа в позицию: {order_type}",
 | 
					        await callback.message.answer("Выбран тип входа в позицию по текущей цене:",
 | 
				
			||||||
                                      reply_markup=inline_markup.start_trading_markup)
 | 
					                                      reply_markup=inline_markup.start_trading_markup)
 | 
				
			||||||
        await callback.answer()
 | 
					        await callback.answer()
 | 
				
			||||||
    except Exception as e:
 | 
					    except Exception as e:
 | 
				
			||||||
@@ -181,7 +183,7 @@ async def set_limit_price(message: Message, state: FSMContext) -> None:
 | 
				
			|||||||
    await rq.update_entry_order_type(message.from_user.id, 'Limit')
 | 
					    await rq.update_entry_order_type(message.from_user.id, 'Limit')
 | 
				
			||||||
    await rq.update_limit_price(message.from_user.id, price)
 | 
					    await rq.update_limit_price(message.from_user.id, price)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await message.answer(f"Цена лимитного ордера установлена: {price}", reply_markup=inline_markup.start_trading_markup)
 | 
					    await message.answer(f"Триггер цена установлена: {price}", reply_markup=inline_markup.start_trading_markup)
 | 
				
			||||||
    await state.clear()
 | 
					    await state.clear()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -196,11 +198,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 +230,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 +294,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:
 | 
				
			||||||
@@ -505,9 +541,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 +583,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()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -559,4 +604,4 @@ async def cancel(callback: CallbackQuery, state: FSMContext) -> None:
 | 
				
			|||||||
        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:
 | 
					    except Exception as e:
 | 
				
			||||||
        logger.error("Ошибка при обработке отмены: %s", e)
 | 
					        logger.error("Ошибка при обработке отмены: %s", e)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -61,8 +61,8 @@ 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
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
@@ -79,7 +79,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')],
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -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,14 @@ 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)
 | 
				
			||||||
    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):
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -320,6 +320,8 @@ async def get_user_main_settings(tg_id):
 | 
				
			|||||||
                'limit_order_price': user.limit_order_price,
 | 
					                'limit_order_price': user.limit_order_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 +370,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 +394,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))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -582,4 +592,4 @@ async def set_last_series_info(tg_id: int, last_side: str):
 | 
				
			|||||||
                    last_side=last_side,
 | 
					                    last_side=last_side,
 | 
				
			||||||
                )
 | 
					                )
 | 
				
			||||||
                session.add(new_entry)
 | 
					                session.add(new_entry)
 | 
				
			||||||
        await session.commit()
 | 
					        await session.commit()
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -23,42 +23,12 @@ async def reg_new_user_default_condition_settings(id):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
async def main_settings_message(id, message):
 | 
					async def main_settings_message(id, message):
 | 
				
			||||||
 | 
					 | 
				
			||||||
    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> установить таймер / удалить таймер
 | 
				
			||||||
"""
 | 
					"""
 | 
				
			||||||
    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)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -140,4 +110,4 @@ async def ai_analytics_message(message, state):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    Описание... '''
 | 
					    Описание... '''
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    await message.answer(text=text, parse_mode='html', reply_markup=inline_markup.buttons_yes_no_markup)
 | 
					    await message.answer(text=text, parse_mode='html', reply_markup=inline_markup.buttons_yes_no_markup)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,7 +6,6 @@ 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 +39,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 +50,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 +59,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']}   
 | 
				
			||||||
@@ -267,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("Ошибка при получении активных позиций: %s", 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:
 | 
				
			||||||
@@ -318,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)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -333,6 +308,7 @@ 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()
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										0
									
								
								data/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								data/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										1
									
								
								logger_helper/loggers/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								logger_helper/loggers/.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -1 +0,0 @@
 | 
				
			|||||||
*.log
 | 
					 | 
				
			||||||
@@ -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
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user