forked from kodorvan/stcs
187 lines
5.0 KiB
Python
187 lines
5.0 KiB
Python
import logging.config
|
|
|
|
from app.bybit import get_bybit_client
|
|
from logger_helper.logger_helper import LOGGING_CONFIG
|
|
|
|
logging.config.dictConfig(LOGGING_CONFIG)
|
|
logger = logging.getLogger("helper_functions")
|
|
|
|
|
|
def safe_float(val) -> float:
|
|
"""
|
|
Function to safely convert string to float
|
|
"""
|
|
try:
|
|
if val is None or val == "":
|
|
return 0.0
|
|
return float(val)
|
|
except (ValueError, TypeError):
|
|
logger.error("Error converting value to float: %s", val)
|
|
return 0.0
|
|
|
|
|
|
def is_number(value: str) -> bool:
|
|
"""
|
|
Checks if a given string represents a number.
|
|
|
|
Args:
|
|
value (str): The string to check.
|
|
|
|
Returns:
|
|
bool: True if the string represents a number, False otherwise.
|
|
"""
|
|
try:
|
|
# Convert the string to a float
|
|
num = float(value)
|
|
# Check if the number is positive
|
|
if num < 0:
|
|
return False
|
|
# Check if the string contains "+" or "-"
|
|
if "+" in value or "-" in value:
|
|
return False
|
|
# Check if the string contains only digits
|
|
allowed_chars = set("0123456789.")
|
|
if not all(ch in allowed_chars for ch in value):
|
|
return False
|
|
return True
|
|
except ValueError:
|
|
return False
|
|
|
|
|
|
def is_int(value: str) -> bool:
|
|
"""
|
|
Checks if a given string represents an integer.
|
|
|
|
Args:
|
|
value (str): The string to check.
|
|
|
|
Returns:
|
|
bool: True if the string represents an integer, False otherwise.
|
|
"""
|
|
# Check if the string contains only digits
|
|
if not value.isdigit():
|
|
return False
|
|
# Convert the string to an integer
|
|
num = int(value)
|
|
return num > 0
|
|
|
|
|
|
def is_int_for_timer(value: str) -> bool | int:
|
|
"""
|
|
Checks if a given string represents an integer for timer.
|
|
|
|
Args:
|
|
value (str): The string to check.
|
|
|
|
Returns:
|
|
bool: True if the string represents an integer, False otherwise.
|
|
"""
|
|
# Check if the string contains only digits
|
|
try:
|
|
num = int(value)
|
|
|
|
if num >= 0:
|
|
return num
|
|
else:
|
|
return False
|
|
except ValueError:
|
|
return False
|
|
|
|
|
|
def get_base_currency(symbol: str) -> str:
|
|
"""
|
|
Extracts the base currency from a symbol string.
|
|
|
|
Args:
|
|
symbol (str): The symbol string to extract the base currency from.
|
|
|
|
Returns:
|
|
str: The base currency extracted from the symbol string.
|
|
"""
|
|
if symbol.endswith("USDT"):
|
|
return symbol[:-4]
|
|
return symbol
|
|
|
|
|
|
def safe_int(value, default=0) -> int:
|
|
"""
|
|
Integer conversion with default value.
|
|
"""
|
|
try:
|
|
return int(value)
|
|
except (ValueError, TypeError):
|
|
return default
|
|
|
|
|
|
def format_value(value) -> str:
|
|
"""
|
|
Function to format value
|
|
"""
|
|
if not value or value.strip() == "":
|
|
return "Нет данных"
|
|
return value
|
|
|
|
|
|
def check_limit_price(limit_price, min_price, max_price) -> str | None:
|
|
"""
|
|
Function to check limit price
|
|
"""
|
|
if limit_price < min_price:
|
|
return "Limit price is out min price"
|
|
if limit_price > max_price:
|
|
return "Limit price is out max price"
|
|
return None
|
|
|
|
|
|
async def get_liquidation_price(
|
|
tg_id: int, symbol: str, entry_price: float, leverage: float
|
|
) -> tuple[float, float]:
|
|
"""
|
|
Function to get liquidation price
|
|
"""
|
|
try:
|
|
client = await get_bybit_client(tg_id=tg_id)
|
|
get_risk_info = client.get_risk_limit(category="linear", symbol=symbol)
|
|
risk_list = get_risk_info.get("result", {}).get("list", [])
|
|
risk_level = risk_list[0] if risk_list else {}
|
|
maintenance_margin_rate = safe_float(risk_level.get("maintenanceMargin"))
|
|
|
|
liq_price_long = entry_price * (1 - 1 / leverage + maintenance_margin_rate)
|
|
liq_price_short = entry_price * (1 + 1 / leverage - maintenance_margin_rate)
|
|
|
|
liq_price = liq_price_long, liq_price_short
|
|
|
|
return liq_price
|
|
except Exception as e:
|
|
logger.error("Error getting liquidation price: %s", e)
|
|
return 0, 0
|
|
|
|
|
|
async def calculate_total_budget(
|
|
quantity, martingale_factor, max_steps, commission_fee_percent
|
|
) -> float:
|
|
"""
|
|
Calculate the total budget for a series of trading steps.
|
|
|
|
Args:
|
|
quantity (float): The initial quantity of the asset.
|
|
martingale_factor (float): The factor by which the quantity is multiplied for each step.
|
|
max_steps (int): The maximum number of trading steps.
|
|
commission_fee_percent (float): The commission fee percentage.
|
|
|
|
Returns:
|
|
float: The total budget for the series of trading steps.
|
|
"""
|
|
total = 0
|
|
for step in range(max_steps):
|
|
set_quantity = quantity * (martingale_factor**step)
|
|
if commission_fee_percent == 0:
|
|
# Commission fee is not added to the position size
|
|
r_quantity = set_quantity
|
|
else:
|
|
# Commission fee is added to the position size
|
|
r_quantity = set_quantity * (1 + 2 * commission_fee_percent)
|
|
|
|
total += r_quantity
|
|
return total
|