Spaces:
Paused
Paused
Upload folder using huggingface_hub
Browse files- .env +15 -0
- .env-example +13 -0
- Dockerfile +13 -0
- INSTALL.bat +11 -0
- INSTALL.sh +48 -0
- START.bat +9 -0
- START.sh +10 -0
- bot/__init__.py +0 -0
- bot/__pycache__/__init__.cpython-310.pyc +0 -0
- bot/__pycache__/launcher.cpython-310.pyc +0 -0
- bot/__pycache__/utils.cpython-310.pyc +0 -0
- bot/config/__init__.py +0 -0
- bot/config/__pycache__/__init__.cpython-310.pyc +0 -0
- bot/config/__pycache__/headers.cpython-310.pyc +0 -0
- bot/config/__pycache__/logger.cpython-310.pyc +0 -0
- bot/config/__pycache__/settings.cpython-310.pyc +0 -0
- bot/config/headers.py +12 -0
- bot/config/logger.py +15 -0
- bot/config/settings.py +79 -0
- bot/core/__init__.py +0 -0
- bot/core/__pycache__/__init__.cpython-310.pyc +0 -0
- bot/core/__pycache__/api.cpython-310.pyc +0 -0
- bot/core/__pycache__/bot.cpython-310.pyc +0 -0
- bot/core/__pycache__/errors.cpython-310.pyc +0 -0
- bot/core/__pycache__/models.cpython-310.pyc +0 -0
- bot/core/__pycache__/utils.cpython-310.pyc +0 -0
- bot/core/api.py +320 -0
- bot/core/api_js_helpers/__init__.py +0 -0
- bot/core/api_js_helpers/__pycache__/__init__.cpython-310.pyc +0 -0
- bot/core/api_js_helpers/__pycache__/bet_counter.cpython-310.pyc +0 -0
- bot/core/api_js_helpers/__pycache__/upgrader.cpython-310.pyc +0 -0
- bot/core/api_js_helpers/bet_counter.py +48 -0
- bot/core/api_js_helpers/upgrader.py +90 -0
- bot/core/bot.py +427 -0
- bot/core/errors.py +3 -0
- bot/core/models.py +219 -0
- bot/core/utils.py +24 -0
- bot/helper/__pycache__/utils.cpython-310.pyc +0 -0
- bot/helper/utils.py +89 -0
- bot/launcher.py +129 -0
- bot/utils.py +34 -0
- data.json +0 -0
- docker-compose.yml +14 -0
- main.py +15 -0
- proxies.txt +5 -0
- requirements.txt +11 -0
- sessions/yasir.session +0 -0
- youtube.json +25 -0
.env
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
API_ID=25514329
|
3 |
+
API_HASH=2b15eb2dd52e935d6b091432b71cfc78
|
4 |
+
|
5 |
+
TAPS_ENABLED=True
|
6 |
+
TAPS_PER_SECOND=[20, 30]
|
7 |
+
PVP_ENABLED=False
|
8 |
+
PVP_LEAGUE=
|
9 |
+
PVP_STRATEGY=aggressive
|
10 |
+
PVP_COUNT=10
|
11 |
+
SLEEP_BETWEEN_START=
|
12 |
+
ERRORS_BEFORE_STOP=
|
13 |
+
USE_PROXY_FROM_FILE=
|
14 |
+
REF_ID=hero5413217144
|
15 |
+
MONEY_TO_SAVE=100
|
.env-example
ADDED
@@ -0,0 +1,13 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
API_ID=
|
2 |
+
API_HASH=
|
3 |
+
|
4 |
+
TAPS_ENABLED=
|
5 |
+
TAPS_PER_SECOND=
|
6 |
+
PVP_ENABLED=
|
7 |
+
PVP_LEAGUE=
|
8 |
+
PVP_STRATEGY=
|
9 |
+
PVP_COUNT=
|
10 |
+
SLEEP_BETWEEN_START=
|
11 |
+
ERRORS_BEFORE_STOP=
|
12 |
+
USE_PROXY_FROM_FILE=
|
13 |
+
REF_ID=
|
Dockerfile
ADDED
@@ -0,0 +1,13 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
FROM python:3.10
|
2 |
+
|
3 |
+
WORKDIR /app
|
4 |
+
|
5 |
+
COPY requirements.txt requirements.txt
|
6 |
+
|
7 |
+
RUN pip3 install --upgrade pip setuptools wheel
|
8 |
+
RUN pip3 install --no-warn-script-location --no-cache-dir -r requirements.txt
|
9 |
+
|
10 |
+
COPY . .
|
11 |
+
|
12 |
+
ENTRYPOINT ["python3", "main.py"]
|
13 |
+
CMD ["-a", "2"]
|
INSTALL.bat
ADDED
@@ -0,0 +1,11 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
@echo off
|
2 |
+
echo Creating virtual environment...
|
3 |
+
python -m venv venv
|
4 |
+
echo Activating virtual environment...
|
5 |
+
call venv\Scripts\activate
|
6 |
+
echo Installing dependencies...
|
7 |
+
pip install -r requirements.txt
|
8 |
+
echo Copying .env-example to .env...
|
9 |
+
copy .env-example .env
|
10 |
+
echo Please edit the .env file to add your API_ID and API_HASH.
|
11 |
+
pause
|
INSTALL.sh
ADDED
@@ -0,0 +1,48 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
#!/bin/bash
|
2 |
+
|
3 |
+
install_python() {
|
4 |
+
echo "Select the Python version to install:"
|
5 |
+
echo "1) Python 3.10"
|
6 |
+
echo "2) Python 3.11"
|
7 |
+
echo "3) Python 3.12"
|
8 |
+
read -p "Enter the number of your choice: " choice
|
9 |
+
|
10 |
+
case $choice in
|
11 |
+
1) version="3.10" ;;
|
12 |
+
2) version="3.11" ;;
|
13 |
+
3) version="3.12" ;;
|
14 |
+
*) echo "Invalid choice"; exit 1 ;;
|
15 |
+
esac
|
16 |
+
|
17 |
+
if command -v apt-get &> /dev/null; then
|
18 |
+
sudo apt-get update
|
19 |
+
sudo apt-get install -y python$version python$version-venv python$version-pip
|
20 |
+
elif command -v yum &> /dev/null; then
|
21 |
+
sudo yum install -y https://repo.ius.io/ius-release-el$(rpm -E %{rhel}).rpm
|
22 |
+
sudo yum install -y python$version python$version-venv python$version-pip
|
23 |
+
elif command -v dnf &> /dev/null; then
|
24 |
+
sudo dnf install -y python$version python$version-venv python$version-pip
|
25 |
+
else
|
26 |
+
echo "Package manager not supported. Please install Python manually."
|
27 |
+
exit 1
|
28 |
+
fi
|
29 |
+
}
|
30 |
+
|
31 |
+
if ! command -v python3 &> /dev/null; then
|
32 |
+
install_python
|
33 |
+
fi
|
34 |
+
|
35 |
+
echo "Creating virtual environment..."
|
36 |
+
python3 -m venv venv
|
37 |
+
|
38 |
+
echo "Activating virtual environment..."
|
39 |
+
source venv/bin/activate
|
40 |
+
|
41 |
+
echo "Installing dependencies..."
|
42 |
+
pip install -r requirements.txt
|
43 |
+
|
44 |
+
echo "Copying .env-example to .env..."
|
45 |
+
cp .env-example .env
|
46 |
+
|
47 |
+
echo "Please edit the .env file to add your API_ID and API_HASH."
|
48 |
+
read -p "Press any key to continue..."
|
START.bat
ADDED
@@ -0,0 +1,9 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
@echo off
|
2 |
+
echo Activating virtual environment...
|
3 |
+
title MuskEmpire
|
4 |
+
call venv\Scripts\activate
|
5 |
+
echo Starting git pull
|
6 |
+
git pull
|
7 |
+
echo Starting the bot...
|
8 |
+
python main.py -a 2
|
9 |
+
pause
|
START.sh
ADDED
@@ -0,0 +1,10 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
#!/bin/bash
|
2 |
+
|
3 |
+
echo "Activating virtual environment..."
|
4 |
+
source venv/bin/activate
|
5 |
+
|
6 |
+
echo "Starting the bot..."
|
7 |
+
python main.py
|
8 |
+
|
9 |
+
echo "Press any key to continue..."
|
10 |
+
read -n 1 -s
|
bot/__init__.py
ADDED
File without changes
|
bot/__pycache__/__init__.cpython-310.pyc
ADDED
Binary file (131 Bytes). View file
|
|
bot/__pycache__/launcher.cpython-310.pyc
ADDED
Binary file (4.55 kB). View file
|
|
bot/__pycache__/utils.cpython-310.pyc
ADDED
Binary file (1.4 kB). View file
|
|
bot/config/__init__.py
ADDED
File without changes
|
bot/config/__pycache__/__init__.cpython-310.pyc
ADDED
Binary file (138 Bytes). View file
|
|
bot/config/__pycache__/headers.cpython-310.pyc
ADDED
Binary file (647 Bytes). View file
|
|
bot/config/__pycache__/logger.cpython-310.pyc
ADDED
Binary file (539 Bytes). View file
|
|
bot/config/__pycache__/settings.cpython-310.pyc
ADDED
Binary file (3.54 kB). View file
|
|
bot/config/headers.py
ADDED
@@ -0,0 +1,12 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
headers = {
|
2 |
+
"Accept": "*/*",
|
3 |
+
"Accept-Language": "ru,ru-RU;q=0.9,en-US;q=0.8,en;q=0.7",
|
4 |
+
"Content-Type": "application/json",
|
5 |
+
"Origin": "https://game.xempire.io",
|
6 |
+
"Referer": "https://game.xempire.io/",
|
7 |
+
"Sec-Fetch-Dest": "empty",
|
8 |
+
"Sec-Fetch-Mode": "cors",
|
9 |
+
"Sec-Fetch-Site": "same-site",
|
10 |
+
"User-Agent": "Mozilla/5.0 (Linux; Android 9; SM-N971N Build/PQ3B.190801.07101020; wv) AppleWebKit/537.36 (KHTML, like Gecko) Version/4.0 Chrome/124.0.6367.82 Safari/537.36",
|
11 |
+
"X-Requested-With": "org.telegram.messenger.web",
|
12 |
+
}
|
bot/config/logger.py
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import sys
|
2 |
+
|
3 |
+
from loguru import logger
|
4 |
+
|
5 |
+
logger.remove()
|
6 |
+
|
7 |
+
logger_str_format = (
|
8 |
+
"<white>{time:YYYY-MM-DD HH:mm:ss}</white> | "
|
9 |
+
"<level>{level: <8}</level> | "
|
10 |
+
"<c><b>{line: <5}</b></c>| "
|
11 |
+
"<c><b>{extra[session_name]: <7}</b></c> | "
|
12 |
+
"<white><b>{message}</b></white>"
|
13 |
+
)
|
14 |
+
logger.add(sink=sys.stdout, format=logger_str_format, colorize=True)
|
15 |
+
log = logger.bind(session_name="GLOBAL").opt(colors=True)
|
bot/config/settings.py
ADDED
@@ -0,0 +1,79 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from enum import Enum
|
2 |
+
|
3 |
+
from pydantic import Field
|
4 |
+
from pydantic_settings import BaseSettings, SettingsConfigDict
|
5 |
+
|
6 |
+
logo = """
|
7 |
+
|
8 |
+
███ ███ ██ ██ ███████ ██ ██ ███████ ███ ███ ██████ ██ ██████ ███████
|
9 |
+
████ ████ ██ ██ ██ ██ ██ ██ ████ ████ ██ ██ ██ ██ ██ ██
|
10 |
+
██ ████ ██ ██ ██ ███████ █████ █████ ██ ████ ██ ██████ ██ ██████ █████
|
11 |
+
██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██ ██
|
12 |
+
██ ██ ██████ ███████ ██ ██ ███████ ██ ██ ██ ██ ██ ██ ███████
|
13 |
+
|
14 |
+
"""
|
15 |
+
|
16 |
+
|
17 |
+
class Strategy(str, Enum):
|
18 |
+
flexible = "flexible"
|
19 |
+
protective = "protective"
|
20 |
+
aggressive = "aggressive"
|
21 |
+
random = "random"
|
22 |
+
|
23 |
+
|
24 |
+
class League(str, Enum):
|
25 |
+
bronze = "bronze"
|
26 |
+
silver = "silver"
|
27 |
+
gold = "gold"
|
28 |
+
platina = "platina"
|
29 |
+
diamond = "diamond"
|
30 |
+
|
31 |
+
|
32 |
+
class Settings(BaseSettings):
|
33 |
+
model_config = SettingsConfigDict(env_file=".env", env_ignore_empty=True, extra="allow")
|
34 |
+
|
35 |
+
API_ID: int
|
36 |
+
API_HASH: str
|
37 |
+
|
38 |
+
LOGIN_TIMEOUT: int = 3600
|
39 |
+
|
40 |
+
TAPS_ENABLED: bool = True
|
41 |
+
TAPS_PER_SECOND: list[int] = [20, 30]
|
42 |
+
AUTO_UPGRADE_HERO: bool = True
|
43 |
+
PVP_ENABLED: bool = True
|
44 |
+
PVP_LEAGUE: League = League.bronze
|
45 |
+
PVP_STRATEGY: Strategy = Strategy.random
|
46 |
+
PVP_COUNT: int = 5
|
47 |
+
|
48 |
+
SLEEP_BETWEEN_START: list[int] = [10, 20]
|
49 |
+
SESSION_AC_DELAY: int = 10
|
50 |
+
ERRORS_BEFORE_STOP: int = 5
|
51 |
+
USE_PROXY_FROM_FILE: bool = False
|
52 |
+
ADD_LOCAL_MACHINE_AS_IP: bool = False
|
53 |
+
|
54 |
+
RANDOM_SLEEP_TIME: int = 8
|
55 |
+
SKILL_WEIGHT: float = 0
|
56 |
+
MAX_SKILL_UPGRADE_COSTS: int = 5e9
|
57 |
+
|
58 |
+
MONEY_TO_SAVE: int = 1_000_000
|
59 |
+
|
60 |
+
AUTO_UPGRADE_MINING: bool = True
|
61 |
+
MAX_MINING_UPGRADE_LEVEL: int = 30
|
62 |
+
MAX_MINING_ENERGY_RECOVERY_UPGRADE_LEVEL: int = 60
|
63 |
+
MINING_ENERGY_SKILLS: list[str] = ["energy_capacity", "energy_recovery", "profit_per_tap_power"]
|
64 |
+
MAX_MINING_UPGRADE_COSTS: int = 5_000_000
|
65 |
+
|
66 |
+
SKIP_IMPROVE_DISCIPLINE_BUG: bool = Field(
|
67 |
+
default=False,
|
68 |
+
description="Skip improve discipline bug for eror "
|
69 |
+
"{'success': False, 'error': 'invalid key improve_discipline'}",
|
70 |
+
)
|
71 |
+
SKIP_TO_UPGRADE_SKILLS: list = Field([], description='Skip upgrade skills. For example: ["Уборщик", "Рекрутер,HR"]')
|
72 |
+
|
73 |
+
BOT_SLEEP_TIME: list[int] = [3000, 3500]
|
74 |
+
REF_ID: str = "hero1092379081"
|
75 |
+
base_url: str = "https://game.muskempire.io/"
|
76 |
+
bot_name: str = "empirebot"
|
77 |
+
|
78 |
+
|
79 |
+
config = Settings()
|
bot/core/__init__.py
ADDED
File without changes
|
bot/core/__pycache__/__init__.cpython-310.pyc
ADDED
Binary file (136 Bytes). View file
|
|
bot/core/__pycache__/api.cpython-310.pyc
ADDED
Binary file (11.5 kB). View file
|
|
bot/core/__pycache__/bot.cpython-310.pyc
ADDED
Binary file (16.4 kB). View file
|
|
bot/core/__pycache__/errors.cpython-310.pyc
ADDED
Binary file (464 Bytes). View file
|
|
bot/core/__pycache__/models.cpython-310.pyc
ADDED
Binary file (6.96 kB). View file
|
|
bot/core/__pycache__/utils.cpython-310.pyc
ADDED
Binary file (900 Bytes). View file
|
|
bot/core/api.py
ADDED
@@ -0,0 +1,320 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import json
|
3 |
+
import random
|
4 |
+
from datetime import datetime
|
5 |
+
from typing import NamedTuple
|
6 |
+
from urllib.parse import parse_qs
|
7 |
+
|
8 |
+
import aiohttp
|
9 |
+
from aiocache import Cache, cached
|
10 |
+
from better_proxy import Proxy
|
11 |
+
from pyrogram import Client, errors
|
12 |
+
from pyrogram.errors import FloodWait, RPCError, UserAlreadyParticipant
|
13 |
+
from pyrogram.raw.functions import account
|
14 |
+
from pyrogram.raw.functions.messages import RequestAppWebView
|
15 |
+
from pyrogram.raw.types import InputBotAppShortName, InputNotifyPeer, InputPeerNotifySettings
|
16 |
+
from pytz import UTC
|
17 |
+
|
18 |
+
from bot.config.logger import log
|
19 |
+
from bot.config.settings import config
|
20 |
+
from bot.helper.utils import error_handler, handle_request
|
21 |
+
|
22 |
+
from .errors import TapsError
|
23 |
+
from .models import FundHelper, Profile, PvpData, UserDataAfter
|
24 |
+
from .utils import num_prettier
|
25 |
+
|
26 |
+
|
27 |
+
class TgWebData(NamedTuple):
|
28 |
+
hash: str
|
29 |
+
request_data: dict
|
30 |
+
|
31 |
+
|
32 |
+
class CryptoBotApi:
|
33 |
+
def __init__(self, tg_client: Client):
|
34 |
+
self.session_name = tg_client.name
|
35 |
+
self.tg_client = tg_client
|
36 |
+
self.user_id = None
|
37 |
+
self.api_url = "https://api.xempire.io"
|
38 |
+
self.need_quiz = False
|
39 |
+
self.need_rebus = False
|
40 |
+
self.rebus_key = ""
|
41 |
+
self.errors = 0
|
42 |
+
self.logger = log.bind(session_name=self.session_name)
|
43 |
+
self._peer = None
|
44 |
+
|
45 |
+
async def get_tg_web_data(self, proxy: str | None) -> TgWebData:
|
46 |
+
if proxy:
|
47 |
+
proxy = Proxy.from_str(proxy)
|
48 |
+
proxy_dict = {
|
49 |
+
"scheme": proxy.protocol,
|
50 |
+
"hostname": proxy.host,
|
51 |
+
"port": proxy.port,
|
52 |
+
"username": proxy.login,
|
53 |
+
"password": proxy.password,
|
54 |
+
}
|
55 |
+
else:
|
56 |
+
proxy_dict = None
|
57 |
+
|
58 |
+
self.tg_client.proxy = proxy_dict
|
59 |
+
|
60 |
+
try:
|
61 |
+
async with self.tg_client:
|
62 |
+
if not self._peer:
|
63 |
+
try:
|
64 |
+
self._peer = await self.tg_client.resolve_peer(config.bot_name)
|
65 |
+
except FloodWait as error:
|
66 |
+
self.logger.warning(f"FloodWait error: {error} | Retry in {error.value} seconds")
|
67 |
+
await asyncio.sleep(delay=error.value)
|
68 |
+
# update in session db peer ids to fix this errors˚
|
69 |
+
async for dialog in self.tg_client.get_dialogs():
|
70 |
+
if dialog.chat and dialog.chat.username and dialog.chat.username == config.bot_name:
|
71 |
+
break
|
72 |
+
self._peer = await self.tg_client.resolve_peer(config.bot_name)
|
73 |
+
|
74 |
+
web_view = await self.tg_client.invoke(
|
75 |
+
RequestAppWebView(
|
76 |
+
peer=self._peer,
|
77 |
+
app=InputBotAppShortName(bot_id=self._peer, short_name="game"),
|
78 |
+
platform="android",
|
79 |
+
write_allowed=True,
|
80 |
+
start_param=config.REF_ID,
|
81 |
+
)
|
82 |
+
)
|
83 |
+
tg_web_data = parse_qs(web_view.url.split("#")[1]).get("tgWebAppData")[0]
|
84 |
+
query_params = parse_qs(tg_web_data)
|
85 |
+
return TgWebData(
|
86 |
+
request_data={
|
87 |
+
"data": {
|
88 |
+
"chatId": "",
|
89 |
+
"chatInstance": tg_web_data,
|
90 |
+
"chatType": query_params.get("chat_type")[0],
|
91 |
+
"initData": tg_web_data,
|
92 |
+
"platform": "android",
|
93 |
+
"startParam": config.REF_ID,
|
94 |
+
},
|
95 |
+
},
|
96 |
+
hash=query_params.get("hash")[0],
|
97 |
+
)
|
98 |
+
|
99 |
+
except RuntimeError as error:
|
100 |
+
raise error from error
|
101 |
+
except FloodWait as error:
|
102 |
+
log.warning(f"{self.session_name} | FloodWait error: {error} | Retry in {error.value} seconds")
|
103 |
+
await asyncio.sleep(delay=error.value)
|
104 |
+
raise
|
105 |
+
except Exception as error:
|
106 |
+
log.error(f"{self.session_name} | Authorization error: {error}")
|
107 |
+
await asyncio.sleep(delay=3)
|
108 |
+
raise
|
109 |
+
|
110 |
+
async def join_and_archive_channel(self, channel_name: str) -> None:
|
111 |
+
try:
|
112 |
+
async with self.tg_client:
|
113 |
+
try:
|
114 |
+
chat = await self.tg_client.join_chat(channel_name)
|
115 |
+
self.logger.info(f"Successfully joined to <g>{chat.title}</g>")
|
116 |
+
except UserAlreadyParticipant:
|
117 |
+
self.logger.info(f"Chat <y>{channel_name}</y> already joined")
|
118 |
+
chat = await self.tg_client.get_chat(channel_name)
|
119 |
+
except RPCError:
|
120 |
+
self.logger.error(f"Channel <y>{channel_name}</y> not found")
|
121 |
+
raise
|
122 |
+
|
123 |
+
await self.sleeper()
|
124 |
+
peer = await self.tg_client.resolve_peer(chat.id)
|
125 |
+
|
126 |
+
await self.tg_client.invoke(
|
127 |
+
account.UpdateNotifySettings(
|
128 |
+
peer=InputNotifyPeer(peer=peer), settings=InputPeerNotifySettings(mute_until=2147483647)
|
129 |
+
)
|
130 |
+
)
|
131 |
+
self.logger.info(f"Successfully muted chat <g>{chat.title}</g> for channel <y>{channel_name}</y>")
|
132 |
+
await self.sleeper()
|
133 |
+
await self.tg_client.archive_chats(chat_ids=[chat.id])
|
134 |
+
self.logger.info(f"Channel <g>{chat.title}</g> successfully archived for channel <y>{channel_name}</y>")
|
135 |
+
|
136 |
+
except errors.FloodWait as e:
|
137 |
+
self.logger.error(f"Waiting {e.value} seconds before the next attempt.")
|
138 |
+
await asyncio.sleep(e.value)
|
139 |
+
raise
|
140 |
+
|
141 |
+
async def sleeper(self, delay: int = config.RANDOM_SLEEP_TIME, additional_delay: int = 4) -> None:
|
142 |
+
await asyncio.sleep(random.random() * delay + additional_delay)
|
143 |
+
|
144 |
+
@error_handler()
|
145 |
+
@handle_request("/telegram/auth")
|
146 |
+
async def login(self, *, response_json: dict, json_body: dict) -> bool:
|
147 |
+
if response_json.get("success", False):
|
148 |
+
self.logger.success("Login successful")
|
149 |
+
return True
|
150 |
+
return False
|
151 |
+
|
152 |
+
@error_handler()
|
153 |
+
@handle_request("/dbs", json_body={"data": {"dbs": ["all"]}})
|
154 |
+
async def get_dbs(self, *, response_json: dict) -> dict:
|
155 |
+
return response_json["data"]
|
156 |
+
|
157 |
+
@error_handler()
|
158 |
+
@handle_request("/hero/balance/sync", json_body={"data": {}})
|
159 |
+
async def syn_hero_balance(self, *, response_json: dict) -> Profile:
|
160 |
+
self._update_money_balance(response_json)
|
161 |
+
self.logger.info(
|
162 |
+
f"Level: <blue>{self.level}</blue> | "
|
163 |
+
f"Balance: <y>{num_prettier(self.balance)}</y> | "
|
164 |
+
f"Money per hour: <g>{num_prettier(self.mph)}</g>"
|
165 |
+
)
|
166 |
+
return Profile(**response_json["data"])
|
167 |
+
|
168 |
+
@error_handler()
|
169 |
+
@handle_request("/user/data/all", json_body={"data": {}})
|
170 |
+
async def get_profile_full(self, *, response_json: dict) -> dict:
|
171 |
+
return dict(**response_json["data"])
|
172 |
+
|
173 |
+
@error_handler()
|
174 |
+
@handle_request("/user/data/after", json_body={"data": {"lang": "en"}})
|
175 |
+
async def user_data_after(self, *, response_json: dict) -> UserDataAfter:
|
176 |
+
return UserDataAfter(**response_json["data"])
|
177 |
+
|
178 |
+
@error_handler()
|
179 |
+
@handle_request("/hero/bonus/offline/claim")
|
180 |
+
async def get_offline_bonus(self, *, response_json: dict) -> None:
|
181 |
+
self._update_money_balance(response_json)
|
182 |
+
self.logger.success(f"Offline bonus claimed: <y>+{num_prettier(self.user_profile.offline_bonus)}</y>")
|
183 |
+
|
184 |
+
@error_handler()
|
185 |
+
@handle_request("/quests/daily/claim")
|
186 |
+
async def daily_reward(self, *, response_json: dict, json_body: dict) -> None:
|
187 |
+
self._update_money_balance(response_json)
|
188 |
+
|
189 |
+
@error_handler()
|
190 |
+
@handle_request("/quests/claim")
|
191 |
+
async def quest_reward_claim(self, *, response_json: dict, json_body: dict) -> bool:
|
192 |
+
self._update_money_balance(response_json)
|
193 |
+
return True
|
194 |
+
|
195 |
+
@error_handler()
|
196 |
+
@handle_request("/quests/daily/progress/claim")
|
197 |
+
async def daily_quest_reward(self, *, response_json: dict, json_body: dict) -> None:
|
198 |
+
self._update_money_balance(response_json)
|
199 |
+
|
200 |
+
@error_handler()
|
201 |
+
@handle_request("/quests/daily/progress/all")
|
202 |
+
async def all_daily_quests(self, *, response_json: dict) -> dict:
|
203 |
+
return response_json["data"]
|
204 |
+
|
205 |
+
@error_handler()
|
206 |
+
@handle_request("/quests/check")
|
207 |
+
async def quest_check(self, *, response_json: dict, json_body: dict) -> bool:
|
208 |
+
await self.sleeper()
|
209 |
+
await self.quest_reward_claim(json_body=json_body)
|
210 |
+
|
211 |
+
@error_handler()
|
212 |
+
@handle_request("/friends/claim")
|
213 |
+
async def friend_reward(self, *, response_json: dict, json_body: dict) -> None:
|
214 |
+
self._update_money_balance(response_json)
|
215 |
+
|
216 |
+
@error_handler()
|
217 |
+
@handle_request("/hero/action/tap")
|
218 |
+
async def api_perform_taps(self, *, response_json: dict, json_body: dict) -> int:
|
219 |
+
if (error_msg := response_json.get("error")) and "take some rest" in error_msg:
|
220 |
+
raise TapsError(error_msg)
|
221 |
+
data = self._update_money_balance(response_json)
|
222 |
+
self.tapped_today = data.get("tappedToday", 0)
|
223 |
+
return int(data["hero"]["earns"]["task"]["energy"])
|
224 |
+
|
225 |
+
@cached(ttl=2 * 60 * 60, cache=Cache.MEMORY)
|
226 |
+
@error_handler()
|
227 |
+
@handle_request(
|
228 |
+
"https://raw.githubusercontent.com/testingstrategy/musk_daily/main/daily.json",
|
229 |
+
method="GET",
|
230 |
+
full_url=True,
|
231 |
+
)
|
232 |
+
async def get_helper(self, *, response_json: str) -> FundHelper | dict:
|
233 |
+
response_json = json.loads(response_json)
|
234 |
+
return FundHelper(
|
235 |
+
funds=response_json.get(str(datetime.now(UTC).date()), {}).get("funds", set()),
|
236 |
+
**response_json,
|
237 |
+
)
|
238 |
+
|
239 |
+
@error_handler()
|
240 |
+
@handle_request("/fund/info")
|
241 |
+
async def get_funds_info(self, *, response_json: dict) -> dict:
|
242 |
+
return response_json["data"]
|
243 |
+
|
244 |
+
@error_handler()
|
245 |
+
@handle_request("/box/list", json_body={})
|
246 |
+
async def get_box_list(self, *, response_json: dict) -> dict:
|
247 |
+
return response_json["data"] or {}
|
248 |
+
|
249 |
+
@error_handler()
|
250 |
+
@handle_request("/box/open")
|
251 |
+
async def box_open(self, *, response_json: dict, json_body: dict) -> list:
|
252 |
+
return response_json["data"]
|
253 |
+
|
254 |
+
@error_handler()
|
255 |
+
@handle_request("/pvp/info")
|
256 |
+
async def get_pvp_info(self, *, response_json: dict) -> dict:
|
257 |
+
return response_json["data"]
|
258 |
+
|
259 |
+
@error_handler()
|
260 |
+
@handle_request("/pvp/fight/start")
|
261 |
+
async def get_pvp_fight(self, *, response_json: dict, json_body: dict) -> PvpData | None:
|
262 |
+
if response_json["data"].get("opponent"):
|
263 |
+
return PvpData(**response_json["data"])
|
264 |
+
return None
|
265 |
+
|
266 |
+
@error_handler()
|
267 |
+
@handle_request("/pvp/claim")
|
268 |
+
async def get_pvp_claim(self, *, response_json: dict) -> None:
|
269 |
+
if response_json.get("success"):
|
270 |
+
self._update_money_balance(response_json)
|
271 |
+
|
272 |
+
@error_handler()
|
273 |
+
@handle_request(
|
274 |
+
"/settings/save",
|
275 |
+
json_body={
|
276 |
+
"data": {
|
277 |
+
"id": None,
|
278 |
+
"music": False,
|
279 |
+
"sound": True,
|
280 |
+
"vibrate": True,
|
281 |
+
"animations": True,
|
282 |
+
"darkTheme": True,
|
283 |
+
"lang": "en",
|
284 |
+
}
|
285 |
+
},
|
286 |
+
)
|
287 |
+
async def sent_eng_settings(self, *, response_json: dict) -> None: ...
|
288 |
+
|
289 |
+
@error_handler()
|
290 |
+
@handle_request("/fund/invest")
|
291 |
+
async def invest(self, *, response_json: dict, json_body: dict) -> None:
|
292 |
+
data = self._update_money_balance(response_json)
|
293 |
+
for fnd in data["funds"]:
|
294 |
+
if fnd["fundKey"] == json_body["data"]["fund"]:
|
295 |
+
money = fnd["moneyProfit"]
|
296 |
+
money_str = (
|
297 |
+
f"Win: <y>+{num_prettier(money)}</y>" if money > 0 else f"Loss: <red>{num_prettier(money)}</red>"
|
298 |
+
)
|
299 |
+
self.logger.success(f"Invest completed: {money_str}")
|
300 |
+
break
|
301 |
+
|
302 |
+
@error_handler()
|
303 |
+
@handle_request("/skills/improve")
|
304 |
+
async def skills_improve(self, *, response_json: dict, json_body: dict) -> None:
|
305 |
+
self._update_money_balance(response_json)
|
306 |
+
|
307 |
+
async def check_proxy(self, proxy: Proxy) -> None:
|
308 |
+
try:
|
309 |
+
response = await self.http_client.get(url="https://httpbin.org/ip", timeout=aiohttp.ClientTimeout(10))
|
310 |
+
ip = (await response.json()).get("origin")
|
311 |
+
self.logger.info(f"Proxy IP: {ip}")
|
312 |
+
except Exception:
|
313 |
+
self.logger.exception(f"Proxy: {proxy}")
|
314 |
+
|
315 |
+
def _update_money_balance(self, response_json: dict) -> dict:
|
316 |
+
response_json = response_json["data"]
|
317 |
+
self.balance = int(response_json["hero"]["money"])
|
318 |
+
self.level = int(response_json["hero"]["level"])
|
319 |
+
self.mph = int(response_json["hero"]["moneyPerHour"])
|
320 |
+
return response_json
|
bot/core/api_js_helpers/__init__.py
ADDED
File without changes
|
bot/core/api_js_helpers/__pycache__/__init__.cpython-310.pyc
ADDED
Binary file (151 Bytes). View file
|
|
bot/core/api_js_helpers/__pycache__/bet_counter.cpython-310.pyc
ADDED
Binary file (1.72 kB). View file
|
|
bot/core/api_js_helpers/__pycache__/upgrader.cpython-310.pyc
ADDED
Binary file (3.24 kB). View file
|
|
bot/core/api_js_helpers/bet_counter.py
ADDED
@@ -0,0 +1,48 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
class BetCounter:
|
2 |
+
bet_steps_count = 7
|
3 |
+
|
4 |
+
def __init__(self, obj):
|
5 |
+
self.obj = obj
|
6 |
+
|
7 |
+
def min_bet(self):
|
8 |
+
multiplier = 2
|
9 |
+
if self.obj.level < 3:
|
10 |
+
multiplier = 5
|
11 |
+
elif self.obj.level < 6:
|
12 |
+
multiplier = 4
|
13 |
+
elif self.obj.level < 10:
|
14 |
+
multiplier = 3
|
15 |
+
|
16 |
+
calculated_bet = self.smart_zero_round(self.obj.mph * multiplier / (self.bet_steps_count * 3))
|
17 |
+
return calculated_bet or 100
|
18 |
+
|
19 |
+
def max_bet(self):
|
20 |
+
return self.min_bet() * self.bet_steps_count
|
21 |
+
|
22 |
+
def smart_zero_round(self, amount: int):
|
23 |
+
def round_to_nearest(value, base=100):
|
24 |
+
return round(value / base) * base
|
25 |
+
|
26 |
+
if amount < 100:
|
27 |
+
return round_to_nearest(amount, 50)
|
28 |
+
elif amount < 1000:
|
29 |
+
return round_to_nearest(amount, 100)
|
30 |
+
elif amount < 10000:
|
31 |
+
return round_to_nearest(amount, 1000)
|
32 |
+
elif amount < 100000:
|
33 |
+
return round_to_nearest(amount, 10000)
|
34 |
+
elif amount < 1000000:
|
35 |
+
return round_to_nearest(amount, 100000)
|
36 |
+
elif amount < 10000000:
|
37 |
+
return round_to_nearest(amount, 1000000)
|
38 |
+
elif amount < 100000000:
|
39 |
+
return round_to_nearest(amount, 10000000)
|
40 |
+
else:
|
41 |
+
return round_to_nearest(amount, 1000)
|
42 |
+
|
43 |
+
def calculate_bet(self) -> int:
|
44 |
+
max_bet_ = self.max_bet()
|
45 |
+
min_bet_ = self.min_bet()
|
46 |
+
while max_bet_ > self.obj.balance:
|
47 |
+
max_bet_ -= min_bet_
|
48 |
+
return max_bet_
|
bot/core/api_js_helpers/upgrader.py
ADDED
@@ -0,0 +1,90 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import math
|
2 |
+
|
3 |
+
|
4 |
+
class Calculator:
|
5 |
+
def get_price(self, data, level):
|
6 |
+
return self.calculate_formula(data.priceFormula, level, data.priceBasic, data.priceFormulaK) if level else 0
|
7 |
+
|
8 |
+
def get_profit(self, data, level):
|
9 |
+
return (
|
10 |
+
self.calculate_formula(data.profitFormula, level, data.profitBasic, data.priceFormulaK, data)
|
11 |
+
if level
|
12 |
+
else 0
|
13 |
+
)
|
14 |
+
|
15 |
+
def calculate_formula(self, formula, level, base_value, formula_coefficient, data=None):
|
16 |
+
result = base_value
|
17 |
+
if formula == "fnCompound":
|
18 |
+
result = self.fn_compound(level, base_value, formula_coefficient)
|
19 |
+
elif formula == "fnLogarithmic":
|
20 |
+
result = self.fn_logarithmic(level, base_value)
|
21 |
+
elif formula == "fnLinear":
|
22 |
+
result = self.fn_linear(level, base_value)
|
23 |
+
elif formula == "fnQuadratic":
|
24 |
+
result = self.fn_quadratic(level, base_value)
|
25 |
+
elif formula == "fnCubic":
|
26 |
+
result = self.fn_cubic(level, base_value)
|
27 |
+
elif formula == "fnExponential":
|
28 |
+
result = self.fn_exponential(level, base_value, formula_coefficient)
|
29 |
+
elif formula == "fnPayback":
|
30 |
+
result = self.fn_payback(level, data)
|
31 |
+
|
32 |
+
return self.smart_round(result)
|
33 |
+
|
34 |
+
def smart_round(self, value):
|
35 |
+
def round_to(value, factor=100):
|
36 |
+
return round(value / factor) * factor
|
37 |
+
|
38 |
+
if value < 50:
|
39 |
+
return round(value)
|
40 |
+
elif value < 100:
|
41 |
+
return round_to(value, 5)
|
42 |
+
elif value < 500:
|
43 |
+
return round_to(value, 25)
|
44 |
+
elif value < 1000:
|
45 |
+
return round_to(value, 50)
|
46 |
+
elif value < 5000:
|
47 |
+
return round_to(value, 100)
|
48 |
+
elif value < 10000:
|
49 |
+
return round_to(value, 200)
|
50 |
+
elif value < 100000:
|
51 |
+
return round_to(value, 500)
|
52 |
+
elif value < 500000:
|
53 |
+
return round_to(value, 1000)
|
54 |
+
elif value < 1000000:
|
55 |
+
return round_to(value, 5000)
|
56 |
+
elif value < 50000000:
|
57 |
+
return round_to(value, 10000)
|
58 |
+
elif value < 100000000:
|
59 |
+
return round_to(value, 50000)
|
60 |
+
else:
|
61 |
+
return round_to(value, 100000)
|
62 |
+
|
63 |
+
def fn_linear(self, level, base_value):
|
64 |
+
return base_value * level
|
65 |
+
|
66 |
+
def fn_quadratic(self, level, base_value):
|
67 |
+
return base_value * level * level
|
68 |
+
|
69 |
+
def fn_cubic(self, level, base_value):
|
70 |
+
return base_value * level * level * level
|
71 |
+
|
72 |
+
def fn_exponential(self, level, base_value, coefficient):
|
73 |
+
return base_value * math.pow(coefficient / 10, level)
|
74 |
+
|
75 |
+
def fn_logarithmic(self, level, base_value):
|
76 |
+
return base_value * math.log2(level + 1)
|
77 |
+
|
78 |
+
def fn_compound(self, level, base_value, coefficient):
|
79 |
+
compound_rate = coefficient / 100
|
80 |
+
return base_value * math.pow(1 + compound_rate, level - 1)
|
81 |
+
|
82 |
+
def fn_payback(self, level, data):
|
83 |
+
accumulated = [0]
|
84 |
+
for current_level in range(1, level + 1):
|
85 |
+
previous_accumulated = accumulated[current_level - 1]
|
86 |
+
current_price = self.get_price(data, current_level)
|
87 |
+
current_profit = data.profitBasic + data.profitFormulaK * (current_level - 1)
|
88 |
+
smart_rounded_value = self.smart_round(previous_accumulated + current_price / current_profit)
|
89 |
+
accumulated.append(smart_rounded_value)
|
90 |
+
return accumulated[level]
|
bot/core/bot.py
ADDED
@@ -0,0 +1,427 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import math
|
3 |
+
import random
|
4 |
+
import time
|
5 |
+
from collections.abc import Generator
|
6 |
+
from datetime import datetime
|
7 |
+
from enum import Enum
|
8 |
+
|
9 |
+
import aiohttp
|
10 |
+
from aiohttp_proxy import ProxyConnector
|
11 |
+
from aiohttp_socks import ProxyConnector as SocksProxyConnector
|
12 |
+
from pyrogram import Client
|
13 |
+
from pytz import UTC
|
14 |
+
|
15 |
+
from bot.config.headers import headers
|
16 |
+
from bot.config.logger import log
|
17 |
+
from bot.config.settings import Strategy, config
|
18 |
+
from bot.core.api_js_helpers.bet_counter import BetCounter
|
19 |
+
|
20 |
+
from .api import CryptoBotApi
|
21 |
+
from .errors import TapsError
|
22 |
+
from .models import DbSkill, DbSkills, Profile, ProfileData, SessionData, SkillLevel
|
23 |
+
from .utils import load_codes_from_files, num_prettier
|
24 |
+
|
25 |
+
|
26 |
+
class CryptoBot(CryptoBotApi):
|
27 |
+
def __init__(self, tg_client: Client, additional_data: dict) -> None:
|
28 |
+
super().__init__(tg_client)
|
29 |
+
self.temporary_stop_taps_time = 0
|
30 |
+
self.bet_calculator = BetCounter(self)
|
31 |
+
self.pvp_count = config.PVP_COUNT
|
32 |
+
self.authorized = False
|
33 |
+
self.settings_was_set = False
|
34 |
+
self.sleep_time = config.BOT_SLEEP_TIME
|
35 |
+
self.additional_data: SessionData = SessionData.model_validate(
|
36 |
+
{k: v for d in additional_data for k, v in d.items()}
|
37 |
+
)
|
38 |
+
|
39 |
+
async def claim_daily_reward(self) -> None:
|
40 |
+
for day, status in self.data_after.daily_rewards.items():
|
41 |
+
if status == "canTake":
|
42 |
+
await self.daily_reward(json_body={"data": str(day)})
|
43 |
+
self.logger.success("Daily reward claimed")
|
44 |
+
return
|
45 |
+
|
46 |
+
async def perform_taps(self, profile: Profile) -> None:
|
47 |
+
self.logger.info("Taps started")
|
48 |
+
energy = profile.energy
|
49 |
+
while True:
|
50 |
+
taps_per_second = random.randint(*config.TAPS_PER_SECOND)
|
51 |
+
seconds = random.randint(5, 8)
|
52 |
+
earned_money = profile.money_per_tap * taps_per_second * seconds
|
53 |
+
energy_spent = math.ceil(earned_money / 2)
|
54 |
+
energy -= energy_spent
|
55 |
+
if energy < 0:
|
56 |
+
self.logger.info("Taps stopped (not enough energy)")
|
57 |
+
break
|
58 |
+
await asyncio.sleep(delay=seconds)
|
59 |
+
try:
|
60 |
+
json_data = {
|
61 |
+
"data": {
|
62 |
+
"data": {"task": {"amount": earned_money, "currentEnergy": energy}},
|
63 |
+
"seconds": seconds,
|
64 |
+
}
|
65 |
+
}
|
66 |
+
energy = await self.api_perform_taps(json_body=json_data)
|
67 |
+
self.logger.success(
|
68 |
+
f"Earned money: <y>+{num_prettier(earned_money)}</y> | Energy left: <blue>{num_prettier(energy)}</blue>"
|
69 |
+
)
|
70 |
+
except TapsError as e:
|
71 |
+
self.logger.warning(f"Taps stopped (<red>{e.message}</red>)")
|
72 |
+
self.temporary_stop_taps_time = time.monotonic() + 60 * 60 * 3
|
73 |
+
break
|
74 |
+
|
75 |
+
async def execute_and_claim_daily_quest(self) -> None:
|
76 |
+
helper_data = await self.get_helper()
|
77 |
+
helper_data.youtube.update(load_codes_from_files())
|
78 |
+
all_daily_quests = await self.all_daily_quests()
|
79 |
+
for key, value in all_daily_quests.items():
|
80 |
+
desc = value["description"]
|
81 |
+
if (
|
82 |
+
value["type"] == "youtube"
|
83 |
+
and not value["isRewarded"]
|
84 |
+
and (code := helper_data.youtube.get(value["description"])) is not None
|
85 |
+
):
|
86 |
+
await self.daily_quest_reward(json_body={"data": {"quest": key, "code": str(code)}})
|
87 |
+
self.logger.info(f"Quest <g>{desc}</g> claimed")
|
88 |
+
elif desc:
|
89 |
+
self.logger.info(f"Quest not executed: \n<r>{desc}</r>")
|
90 |
+
if not value["isRewarded"] and value["isComplete"] and not value["url"]:
|
91 |
+
await self.daily_quest_reward(json_body={"data": {"quest": key, "code": None}})
|
92 |
+
self.logger.info(f"Quest <g>{key}</g> claimed")
|
93 |
+
|
94 |
+
async def claim_all_executed_quest(self) -> None:
|
95 |
+
for i in self.data_after.quests:
|
96 |
+
if not i["isRewarded"]:
|
97 |
+
if config.SKIP_IMPROVE_DISCIPLINE_BUG and i["key"] == "improve_discipline":
|
98 |
+
continue
|
99 |
+
await self.quest_reward_claim(json_body={"data": [i["key"], None]})
|
100 |
+
self.logger.info(f'Quest <g>{i["key"]}</g> claimed ')
|
101 |
+
|
102 |
+
def random_pvp_count(self) -> int:
|
103 |
+
return random.randint(config.PVP_COUNT, config.PVP_COUNT * 2)
|
104 |
+
|
105 |
+
async def _perform_pvp(self, league: dict, strategy: str) -> None:
|
106 |
+
self.pvp_count = self.random_pvp_count()
|
107 |
+
self.logger.info(
|
108 |
+
f"PvP negotiations started | League: <blue>{league['key']}</blue> | Strategy: <g>{strategy}</g>"
|
109 |
+
)
|
110 |
+
res = await self.get_pvp_info()
|
111 |
+
await self.sleeper()
|
112 |
+
if res.get("fight"):
|
113 |
+
await self.get_pvp_claim()
|
114 |
+
await self.sleeper()
|
115 |
+
current_strategy = strategy
|
116 |
+
money = 0
|
117 |
+
while self.pvp_count > 0:
|
118 |
+
if self.balance < int(league["maxContract"]):
|
119 |
+
money_str = (
|
120 |
+
f"Profit: <y>+{num_prettier(money)}</y>"
|
121 |
+
if money >= 0
|
122 |
+
else f"Loss: <red>-{num_prettier(money)}</red>"
|
123 |
+
)
|
124 |
+
self.logger.info(f"PvP negotiations stopped (<red>not enough money</red>). Pvp profit: {money_str}")
|
125 |
+
break
|
126 |
+
|
127 |
+
if strategy == "random":
|
128 |
+
current_strategy = random.choice(self.strategies)
|
129 |
+
self.logger.info("Searching opponent...")
|
130 |
+
current_strategy = current_strategy.value if isinstance(current_strategy, Enum) else current_strategy
|
131 |
+
json_data = {"data": {"league": league["key"], "strategy": current_strategy}}
|
132 |
+
response_json = await self.get_pvp_fight(json_body=json_data)
|
133 |
+
if response_json is None:
|
134 |
+
await self.sleeper(delay=10, additional_delay=5)
|
135 |
+
continue
|
136 |
+
|
137 |
+
fight = response_json.fight
|
138 |
+
opponent_strategy = (
|
139 |
+
fight.player2Strategy if fight.player1 == self.user_profile.user_id else fight.player1Strategy
|
140 |
+
)
|
141 |
+
if fight.winner == self.user_profile.user_id:
|
142 |
+
money += fight.moneyProfit
|
143 |
+
log_part = f"You <g>WIN</g> (<y>+{num_prettier(fight.moneyProfit)})</y>"
|
144 |
+
else:
|
145 |
+
money -= fight.moneyContract
|
146 |
+
log_part = f"You <red>LOSE</red> (<y>-{num_prettier(fight.moneyProfit)}</y>)"
|
147 |
+
self.logger.success(
|
148 |
+
f"Contract sum: <y>{num_prettier(fight.moneyContract)}</y> | "
|
149 |
+
f"Your strategy: <c>{current_strategy}</c> | "
|
150 |
+
f"Opponent strategy: <blue>{opponent_strategy}</blue> | "
|
151 |
+
f"{log_part}"
|
152 |
+
)
|
153 |
+
await self.sleeper(additional_delay=10)
|
154 |
+
await self.get_pvp_claim()
|
155 |
+
self.pvp_count -= 1
|
156 |
+
await self.sleeper()
|
157 |
+
|
158 |
+
self.logger.info(
|
159 |
+
"Total money after all pvp:"
|
160 |
+
+ (f"<i><g>+{num_prettier(money)}</g></i>" if money >= 0 else f"<i><red>{num_prettier(money)}</red></i>")
|
161 |
+
)
|
162 |
+
self.pvp_count = config.PVP_COUNT
|
163 |
+
|
164 |
+
async def get_friend_reward(self) -> None:
|
165 |
+
for friend in [friend for friend in self.data_after.friends if friend["bonusToTake"] > 0]:
|
166 |
+
await self.friend_reward(json_body={"data": friend["id"]})
|
167 |
+
self.logger.info(
|
168 |
+
f"Friend <g>{friend['name']}</g> claimed money <y>{num_prettier(friend['bonusToTake'])}</y>"
|
169 |
+
)
|
170 |
+
await self.sleeper()
|
171 |
+
|
172 |
+
async def solve_quiz_and_rebus(self) -> None:
|
173 |
+
for quest in self.dbs["dbQuests"]:
|
174 |
+
quest_key = quest["key"]
|
175 |
+
if quest["requiredLevel"] > self.user_profile.level:
|
176 |
+
continue
|
177 |
+
if "t.me" in (link := quest.get("actionUrl")) and not self._is_event_solved(quest_key):
|
178 |
+
if len(link.split("/")) > 4 or "muskempire" in link:
|
179 |
+
continue
|
180 |
+
if quest["checkType"] != "fakeCheck":
|
181 |
+
link = link if "/+" in link else link.split("/")[-1]
|
182 |
+
await self.join_and_archive_channel(link)
|
183 |
+
await self.quest_check(json_body={"data": [quest_key]})
|
184 |
+
self.logger.info(
|
185 |
+
f'Claimed <g>{quest["title"]}</g> Reward: <y>+{num_prettier(quest["rewardMoney"])}</y>quest'
|
186 |
+
)
|
187 |
+
if any(i in quest_key for i in ("riddle", "rebus", "tg_story")) and not self._is_event_solved(quest_key):
|
188 |
+
await self.quest_check(json_body={"data": [quest_key, quest["checkData"]]})
|
189 |
+
self.logger.info(f"Was solved <g>{quest['title']}</g>")
|
190 |
+
|
191 |
+
def _is_event_solved(self, quest_key: str) -> bool:
|
192 |
+
return self.data_after.quests and any(i["key"] == quest_key for i in self.data_after.quests)
|
193 |
+
|
194 |
+
async def set_funds(self) -> None:
|
195 |
+
helper_data = await self.get_helper()
|
196 |
+
if helper_data.funds:
|
197 |
+
current_invest = await self.get_funds_info()
|
198 |
+
already_funded = {i["fundKey"] for i in current_invest["funds"]}
|
199 |
+
for fund in list(helper_data.funds - already_funded)[: 3 - len(already_funded)]:
|
200 |
+
if self.balance > (amount := self.bet_calculator.calculate_bet()):
|
201 |
+
self.logger.info(f"Investing <y>{num_prettier(amount)}</y> to fund <blue>{fund}</blue>")
|
202 |
+
await self.invest(json_body={"data": {"fund": fund, "money": amount}})
|
203 |
+
else:
|
204 |
+
self.logger.info("Not enough money for invest")
|
205 |
+
|
206 |
+
async def starting_pvp(self) -> None:
|
207 |
+
if self.dbs:
|
208 |
+
league_data = None
|
209 |
+
for league in self.dbs["dbNegotiationsLeague"]:
|
210 |
+
if league["key"] == config.PVP_LEAGUE:
|
211 |
+
league_data = league
|
212 |
+
break
|
213 |
+
|
214 |
+
if league_data is not None:
|
215 |
+
if self.level >= int(league_data["requiredLevel"]):
|
216 |
+
self.strategies = [strategy["key"] for strategy in self.dbs["dbNegotiationsStrategy"]]
|
217 |
+
if Strategy.random == config.PVP_STRATEGY or config.PVP_STRATEGY in self.strategies:
|
218 |
+
await self._perform_pvp(
|
219 |
+
league=league_data,
|
220 |
+
strategy=config.PVP_STRATEGY.value,
|
221 |
+
)
|
222 |
+
else:
|
223 |
+
config.PVP_ENABLED = False
|
224 |
+
self.logger.warning("PVP_STRATEGY param is invalid. PvP negotiations disabled.")
|
225 |
+
else:
|
226 |
+
config.PVP_ENABLED = False
|
227 |
+
self.logger.warning(
|
228 |
+
f"Your level is too low for the {config.PVP_LEAGUE} league. PvP negotiations disabled."
|
229 |
+
)
|
230 |
+
else:
|
231 |
+
config.PVP_ENABLED = False
|
232 |
+
self.logger.warning("PVP_LEAGUE param is invalid. PvP negotiations disabled.")
|
233 |
+
else:
|
234 |
+
self.logger.warning("Database is missing. PvP negotiations will be skipped this time.")
|
235 |
+
|
236 |
+
async def upgrade_hero(self) -> None:
|
237 |
+
available_skill = list(self._get_available_skills())
|
238 |
+
if config.AUTO_UPGRADE_HERO:
|
239 |
+
await self._upgrade_hero_skill(available_skill)
|
240 |
+
if config.AUTO_UPGRADE_MINING:
|
241 |
+
await self._upgrade_mining_skill(available_skill)
|
242 |
+
|
243 |
+
async def get_box_rewards(self) -> None:
|
244 |
+
boxes = await self.get_box_list()
|
245 |
+
for key, box_count in boxes.items():
|
246 |
+
for _ in range(box_count):
|
247 |
+
res = await self.box_open(json_body={"data": key})
|
248 |
+
self.logger.info(f"Box <g>{key}</g> Was looted: <y>{res['loot']}</y>")
|
249 |
+
|
250 |
+
async def _upgrade_mining_skill(self, available_skill: list[DbSkill]) -> None:
|
251 |
+
for skill in [skill for skill in available_skill if skill.category == "mining"]:
|
252 |
+
if (
|
253 |
+
skill.key in config.MINING_ENERGY_SKILLS
|
254 |
+
and skill.next_level <= config.MAX_MINING_ENERGY_RECOVERY_UPGRADE_LEVEL
|
255 |
+
or (
|
256 |
+
skill.next_level <= config.MAX_MINING_UPGRADE_LEVEL
|
257 |
+
or skill.skill_price <= config.MAX_MINING_UPGRADE_COSTS
|
258 |
+
)
|
259 |
+
):
|
260 |
+
await self._upgrade_skill(skill)
|
261 |
+
|
262 |
+
def _is_enough_money_for_upgrade(self, skill: DbSkill) -> bool:
|
263 |
+
return (self.balance - skill.skill_price) >= config.MONEY_TO_SAVE
|
264 |
+
|
265 |
+
async def _upgrade_hero_skill(self, available_skill: list[DbSkill]) -> None:
|
266 |
+
for skill in sorted(
|
267 |
+
[skill for skill in available_skill if skill.weight],
|
268 |
+
key=lambda x: x.weight,
|
269 |
+
reverse=True,
|
270 |
+
):
|
271 |
+
if skill.title in config.SKIP_TO_UPGRADE_SKILLS:
|
272 |
+
continue
|
273 |
+
# if skill.weight >= config.SKILL_WEIGHT or skill.skill_price <= config.MAX_SKILL_UPGRADE_COSTS:
|
274 |
+
if skill.weight >= config.SKILL_WEIGHT:
|
275 |
+
await self._upgrade_skill(skill)
|
276 |
+
|
277 |
+
async def _upgrade_skill(self, skill: DbSkill) -> None:
|
278 |
+
if self._is_enough_money_for_upgrade(skill):
|
279 |
+
try:
|
280 |
+
await self.skills_improve(json_body={"data": skill.key})
|
281 |
+
self.logger.info(
|
282 |
+
f"Skill: <blue>{skill.title}</blue> upgraded to level: <c>{skill.next_level}</c> "
|
283 |
+
f"Profit: <y>{num_prettier(skill.skill_profit)}</y> "
|
284 |
+
f"Costs: <blue>{num_prettier(skill.skill_price)}</blue> "
|
285 |
+
f"Money stay: <y>{num_prettier(self.balance)}</y> "
|
286 |
+
f"Skill weight <magenta>{skill.weight:.5f}</magenta>"
|
287 |
+
)
|
288 |
+
await self.sleeper()
|
289 |
+
except ValueError:
|
290 |
+
self.logger.exception(f"Failed to upgrade skill: {skill}")
|
291 |
+
raise
|
292 |
+
|
293 |
+
def _get_available_skills(self) -> Generator[DbSkill, None, None]:
|
294 |
+
for skill in DbSkills(**self.dbs).dbSkills:
|
295 |
+
self._calkulate_skill_requirements(skill)
|
296 |
+
if self._is_available_to_upgrade_skills(skill):
|
297 |
+
yield skill
|
298 |
+
|
299 |
+
def _calkulate_skill_requirements(self, skill: DbSkill) -> None:
|
300 |
+
skill.next_level = (
|
301 |
+
self.data_after.skills[skill.key]["level"] + 1 if self.data_after.skills.get(skill.key) else 1
|
302 |
+
)
|
303 |
+
skill.skill_profit = skill.calculate_profit(skill.next_level)
|
304 |
+
skill.skill_price = skill.price_for_level(skill.next_level)
|
305 |
+
skill.weight = skill.skill_profit / skill.skill_price
|
306 |
+
skill.progress_time = skill.get_skill_time(self.data_after)
|
307 |
+
|
308 |
+
def _is_available_to_upgrade_skills(self, skill: DbSkill) -> bool:
|
309 |
+
# check the current skill is still in the process of improvement
|
310 |
+
if skill.progress_time and skill.progress_time.timestamp() + 60 > datetime.now(UTC).timestamp():
|
311 |
+
return False
|
312 |
+
if skill.next_level > skill.maxLevel:
|
313 |
+
return False
|
314 |
+
skill_requirements = skill.get_level_by_skill_level(skill.next_level)
|
315 |
+
if not skill_requirements:
|
316 |
+
return True
|
317 |
+
return (
|
318 |
+
len(self.data_after.friends) >= skill_requirements.requiredFriends
|
319 |
+
and self.user_profile.level >= skill_requirements.requiredHeroLevel
|
320 |
+
and self._is_can_learn_skill(skill_requirements)
|
321 |
+
)
|
322 |
+
|
323 |
+
def _is_can_learn_skill(self, level: SkillLevel) -> bool:
|
324 |
+
if not level.requiredSkills:
|
325 |
+
return True
|
326 |
+
for skill, level in level.requiredSkills.items():
|
327 |
+
if skill not in self.data_after.skills:
|
328 |
+
return False
|
329 |
+
if self.data_after.skills[skill]["level"] >= level:
|
330 |
+
return True
|
331 |
+
return False
|
332 |
+
|
333 |
+
async def login_to_app(self, proxy: str | None) -> bool:
|
334 |
+
if self.authorized:
|
335 |
+
return True
|
336 |
+
tg_web_data = await self.get_tg_web_data(proxy=proxy)
|
337 |
+
self.http_client.headers["Api-Key"] = tg_web_data.hash
|
338 |
+
if await self.login(json_body=tg_web_data.request_data):
|
339 |
+
self.authorized = True
|
340 |
+
return True
|
341 |
+
return False
|
342 |
+
|
343 |
+
async def run(self, proxy: str | None) -> None:
|
344 |
+
proxy = proxy or self.additional_data.proxy
|
345 |
+
if proxy and "socks" in proxy:
|
346 |
+
proxy_conn = SocksProxyConnector.from_url(proxy)
|
347 |
+
elif proxy:
|
348 |
+
proxy_conn = ProxyConnector.from_url(proxy)
|
349 |
+
else:
|
350 |
+
proxy_conn = None
|
351 |
+
|
352 |
+
async with aiohttp.ClientSession(
|
353 |
+
headers=headers,
|
354 |
+
connector=proxy_conn,
|
355 |
+
timeout=aiohttp.ClientTimeout(total=60),
|
356 |
+
) as http_client:
|
357 |
+
self.http_client = http_client
|
358 |
+
if proxy:
|
359 |
+
await self.check_proxy(proxy=proxy)
|
360 |
+
|
361 |
+
while True:
|
362 |
+
if self.errors >= config.ERRORS_BEFORE_STOP:
|
363 |
+
self.logger.error("Bot stopped (too many errors)")
|
364 |
+
break
|
365 |
+
try:
|
366 |
+
if await self.login_to_app(proxy):
|
367 |
+
# if not self.settings_was_set:
|
368 |
+
# await self.sent_eng_settings()
|
369 |
+
data = await self.get_profile_full()
|
370 |
+
self.dbs = data["dbData"]
|
371 |
+
await self.get_box_rewards()
|
372 |
+
|
373 |
+
self.user_profile: ProfileData = ProfileData(**data)
|
374 |
+
if self.user_profile.offline_bonus > 0:
|
375 |
+
await self.get_offline_bonus()
|
376 |
+
|
377 |
+
profile = await self.syn_hero_balance()
|
378 |
+
|
379 |
+
config.MONEY_TO_SAVE = self.bet_calculator.max_bet()
|
380 |
+
self.logger.info(f"Max bet for funds saved: <y>{num_prettier(config.MONEY_TO_SAVE)}</y>")
|
381 |
+
|
382 |
+
self.data_after = await self.user_data_after()
|
383 |
+
|
384 |
+
await self.claim_daily_reward()
|
385 |
+
|
386 |
+
await self.execute_and_claim_daily_quest()
|
387 |
+
|
388 |
+
await self.syn_hero_balance()
|
389 |
+
|
390 |
+
await self.get_friend_reward()
|
391 |
+
|
392 |
+
if config.TAPS_ENABLED and profile.energy and time.monotonic() > self.temporary_stop_taps_time:
|
393 |
+
await self.perform_taps(profile)
|
394 |
+
|
395 |
+
await self.set_funds()
|
396 |
+
await self.solve_quiz_and_rebus()
|
397 |
+
|
398 |
+
await self.claim_all_executed_quest()
|
399 |
+
|
400 |
+
await self.syn_hero_balance()
|
401 |
+
|
402 |
+
await self.upgrade_hero()
|
403 |
+
|
404 |
+
if config.PVP_ENABLED:
|
405 |
+
await self.starting_pvp()
|
406 |
+
await self.syn_hero_balance()
|
407 |
+
sleep_time = random.randint(*config.BOT_SLEEP_TIME)
|
408 |
+
self.logger.info(f"Sleep minutes {sleep_time // 60} minutes")
|
409 |
+
await asyncio.sleep(sleep_time)
|
410 |
+
|
411 |
+
except RuntimeError as error:
|
412 |
+
raise error from error
|
413 |
+
except Exception:
|
414 |
+
self.errors += 1
|
415 |
+
self.authorized = False
|
416 |
+
self.logger.exception("Unknown error")
|
417 |
+
await self.sleeper(additional_delay=self.errors * 8)
|
418 |
+
else:
|
419 |
+
self.errors = 0
|
420 |
+
self.authorized = False
|
421 |
+
|
422 |
+
|
423 |
+
async def run_bot(tg_client: Client, proxy: str | None, additional_data: dict) -> None:
|
424 |
+
try:
|
425 |
+
await CryptoBot(tg_client=tg_client, additional_data=additional_data).run(proxy=proxy)
|
426 |
+
except RuntimeError:
|
427 |
+
log.bind(session_name=tg_client.name).exception("Session error")
|
bot/core/errors.py
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
class TapsError(Exception):
|
2 |
+
def __init__(self, message: str) -> None:
|
3 |
+
self.message = message
|
bot/core/models.py
ADDED
@@ -0,0 +1,219 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from datetime import datetime
|
2 |
+
from typing import Any
|
3 |
+
|
4 |
+
from pydantic import AliasPath, BaseModel, Field, field_validator
|
5 |
+
from pytz import UTC
|
6 |
+
|
7 |
+
from bot.core.api_js_helpers.upgrader import Calculator
|
8 |
+
|
9 |
+
|
10 |
+
class SkillLevel(BaseModel):
|
11 |
+
level: int
|
12 |
+
title: str
|
13 |
+
requiredSkills: dict | list
|
14 |
+
requiredHeroLevel: int
|
15 |
+
requiredFriends: int
|
16 |
+
desc: str
|
17 |
+
|
18 |
+
|
19 |
+
class DbSkill(BaseModel):
|
20 |
+
key: str
|
21 |
+
title: str
|
22 |
+
category: str
|
23 |
+
subCategory: str
|
24 |
+
priceBasic: int
|
25 |
+
priceFormula: str
|
26 |
+
priceFormulaK: int
|
27 |
+
profitBasic: int
|
28 |
+
profitFormula: str
|
29 |
+
profitFormulaK: int
|
30 |
+
maxLevel: int
|
31 |
+
timeBasic: str
|
32 |
+
timeFormula: str
|
33 |
+
timeFormulaK: str
|
34 |
+
desc: str
|
35 |
+
special: str
|
36 |
+
levels: list[SkillLevel]
|
37 |
+
next_level: int = 1
|
38 |
+
skill_profit: int = 0
|
39 |
+
skill_price: int = 0
|
40 |
+
weight: int = 0
|
41 |
+
progress_time: datetime | None = None
|
42 |
+
|
43 |
+
def __init__(self, /, **data: Any) -> None:
|
44 |
+
super().__init__(**data)
|
45 |
+
self._calculator = Calculator()
|
46 |
+
|
47 |
+
def get_level_by_skill_level(self, level: int) -> SkillLevel | None:
|
48 |
+
if not self.levels or self.levels[0].level > level:
|
49 |
+
return None
|
50 |
+
|
51 |
+
for index, skill_level in enumerate(self.levels):
|
52 |
+
if skill_level.level <= level:
|
53 |
+
if index + 1 == len(self.levels):
|
54 |
+
return skill_level
|
55 |
+
if self.levels[index + 1].level > level:
|
56 |
+
return skill_level
|
57 |
+
|
58 |
+
return None
|
59 |
+
|
60 |
+
def calculate_profit(self, level: int) -> int:
|
61 |
+
return self._calculator.get_profit(self, level)
|
62 |
+
|
63 |
+
def price_for_level(self, level: int) -> int:
|
64 |
+
return self._calculator.get_price(self, level)
|
65 |
+
|
66 |
+
def get_skill_time(self, data_after: "UserDataAfter") -> None | datetime:
|
67 |
+
if finish_time := data_after.skills.get(self.key, {}).get("finishUpgradeDate"):
|
68 |
+
return datetime.strptime(finish_time, "%Y-%m-%d %H:%M:%S").replace(tzinfo=UTC)
|
69 |
+
return None
|
70 |
+
|
71 |
+
|
72 |
+
class DbSkills(BaseModel):
|
73 |
+
dbSkills: list[DbSkill]
|
74 |
+
|
75 |
+
|
76 |
+
class ProfileData(BaseModel):
|
77 |
+
user_id: int = Field(validation_alias=AliasPath("profile", "id"))
|
78 |
+
money: int = Field(validation_alias=AliasPath("hero", "money"))
|
79 |
+
level: int = Field(validation_alias=AliasPath("hero", "level"))
|
80 |
+
money_per_hour: int = Field(validation_alias=AliasPath("hero", "moneyPerHour"))
|
81 |
+
offline_bonus: int = Field(validation_alias=AliasPath("hero", "offlineBonus"))
|
82 |
+
|
83 |
+
|
84 |
+
class UserDataAfter(BaseModel):
|
85 |
+
daily_rewards: dict = Field(validation_alias=AliasPath("dailyRewards"))
|
86 |
+
quests: list = Field(validation_alias=AliasPath("quests"))
|
87 |
+
friends: list = Field(validation_alias=AliasPath("friends"))
|
88 |
+
skills: dict | list = Field(
|
89 |
+
description="all user learned skills",
|
90 |
+
examples=[
|
91 |
+
{"desks": {"level": 6, "lastUpgradeDate": "2024-07-30 19:20:32", "finishUpgradeDate": None}},
|
92 |
+
{
|
93 |
+
"empathy": {
|
94 |
+
"level": 6,
|
95 |
+
"lastUpgradeDate": "2024-07-30 19:21:36",
|
96 |
+
"finishUpgradeDate": "2024-07-30 19:22:13",
|
97 |
+
}
|
98 |
+
},
|
99 |
+
],
|
100 |
+
)
|
101 |
+
|
102 |
+
@field_validator("skills")
|
103 |
+
@classmethod
|
104 |
+
def check_skills(cls, v: Any) -> dict:
|
105 |
+
return v or {}
|
106 |
+
|
107 |
+
|
108 |
+
class Profile(BaseModel):
|
109 |
+
money_per_tap: int = Field(validation_alias=AliasPath("hero", "earns", "task", "moneyPerTap"))
|
110 |
+
limit: int = Field(validation_alias=AliasPath("hero", "earns", "task", "limit"))
|
111 |
+
energy: int = Field(validation_alias=AliasPath("hero", "earns", "task", "energy"))
|
112 |
+
energy_recovery: int = Field(validation_alias=AliasPath("hero", "earns", "task", "recoveryPerSecond"))
|
113 |
+
|
114 |
+
money: int = Field(validation_alias=AliasPath("hero", "money"))
|
115 |
+
level: int = Field(validation_alias=AliasPath("hero", "level"))
|
116 |
+
money_per_hour: int = Field(validation_alias=AliasPath("hero", "moneyPerHour"))
|
117 |
+
|
118 |
+
|
119 |
+
class Fight(BaseModel):
|
120 |
+
league: str
|
121 |
+
moneyProfit: int
|
122 |
+
player1: int
|
123 |
+
moneyContract: int
|
124 |
+
player1Strategy: str
|
125 |
+
player1Level: int
|
126 |
+
player1Rewarded: bool
|
127 |
+
player2: int
|
128 |
+
player2Strategy: str
|
129 |
+
player2Rewarded: bool
|
130 |
+
winner: int
|
131 |
+
|
132 |
+
|
133 |
+
class PvpData(BaseModel):
|
134 |
+
opponent: dict | None
|
135 |
+
fight: Fight | None
|
136 |
+
|
137 |
+
|
138 |
+
class FundHelper(BaseModel):
|
139 |
+
funds: set = Field(default_factory=set)
|
140 |
+
youtube: dict
|
141 |
+
|
142 |
+
|
143 |
+
class Skills(BaseModel):
|
144 |
+
skills: dict
|
145 |
+
|
146 |
+
|
147 |
+
if __name__ == "__main__":
|
148 |
+
data = {
|
149 |
+
"hero": {
|
150 |
+
"id": 1092379081,
|
151 |
+
"level": 12,
|
152 |
+
"exp": 207156950,
|
153 |
+
"money": 8154952,
|
154 |
+
"moneyUpdateDate": "2024-08-05 07:02:28",
|
155 |
+
"lastOfflineBonusDate": "2024-08-05 07:02:28",
|
156 |
+
"moneyPerHour": 7943050,
|
157 |
+
"energyUpdateDate": "2024-08-05 07:02:28",
|
158 |
+
"tax": 20,
|
159 |
+
"pvpMatch": 1143,
|
160 |
+
"pvpWin": 646,
|
161 |
+
"pvpLose": 497,
|
162 |
+
"pvpMatchesDaily": 123,
|
163 |
+
"pvpMatchesDailyDay": "2024-08-04",
|
164 |
+
"earns": {
|
165 |
+
"task": {"moneyPerTap": 21, "limit": 9500, "energy": 9500, "recoveryPerSecond": 14},
|
166 |
+
"sell": {"moneyPerTap": 20, "limit": 6600, "energy": 6600, "recoveryPerSecond": 14},
|
167 |
+
},
|
168 |
+
"dailyRewardLastDate": "2024-08-04 11:15:56",
|
169 |
+
"dailyRewardLastIndex": 7,
|
170 |
+
"onboarding": [
|
171 |
+
"9040",
|
172 |
+
"41",
|
173 |
+
"30",
|
174 |
+
"40",
|
175 |
+
"9000",
|
176 |
+
"90",
|
177 |
+
"70",
|
178 |
+
"10720",
|
179 |
+
"9020",
|
180 |
+
"10015",
|
181 |
+
"9010",
|
182 |
+
"80",
|
183 |
+
"50",
|
184 |
+
"10700",
|
185 |
+
"60",
|
186 |
+
"9030",
|
187 |
+
"20",
|
188 |
+
"51",
|
189 |
+
"1",
|
190 |
+
],
|
191 |
+
"updateDate": "2024-07-19 16:39:56.91573",
|
192 |
+
"userId": 1092379081,
|
193 |
+
},
|
194 |
+
"fight": {
|
195 |
+
"id": "08999015-6835-44ed-b00e-3ac529b62285",
|
196 |
+
"league": "bronze",
|
197 |
+
"moneyContract": 6600,
|
198 |
+
"moneyProfit": None,
|
199 |
+
"searchTime": None,
|
200 |
+
"player1": 1092379081,
|
201 |
+
"player1Strategy": "aggressive",
|
202 |
+
"player1Level": 12,
|
203 |
+
"player1Rewarded": False,
|
204 |
+
"player2": None,
|
205 |
+
"player2Strategy": None,
|
206 |
+
"player2Rewarded": False,
|
207 |
+
"winner": None,
|
208 |
+
"draw": [],
|
209 |
+
"updateDate": None,
|
210 |
+
"creationDate": "2024-08-05 07:02:35",
|
211 |
+
},
|
212 |
+
"opponent": None,
|
213 |
+
}
|
214 |
+
PvpData(**data)
|
215 |
+
|
216 |
+
|
217 |
+
class SessionData(BaseModel):
|
218 |
+
user_agent: str = Field(validation_alias="User-Agent")
|
219 |
+
proxy: str | None = None
|
bot/core/utils.py
ADDED
@@ -0,0 +1,24 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import json
|
2 |
+
from functools import lru_cache
|
3 |
+
from pathlib import Path
|
4 |
+
|
5 |
+
|
6 |
+
@lru_cache
|
7 |
+
def load_codes_from_files() -> dict:
|
8 |
+
with Path("youtube.json").open("r", encoding="utf-8") as file:
|
9 |
+
return json.load(file)
|
10 |
+
|
11 |
+
|
12 |
+
def num_prettier(num: int) -> str:
|
13 |
+
number = abs(num)
|
14 |
+
if number >= (comparer := 1e12):
|
15 |
+
prettier_num = f"{number / comparer:.1f}T"
|
16 |
+
elif number >= (comparer := 1e9):
|
17 |
+
prettier_num = f"{number / comparer:.1f}B"
|
18 |
+
elif number >= (comparer := 1e6):
|
19 |
+
prettier_num = f"{number / comparer:.1f}M"
|
20 |
+
elif number >= (comparer := 1e3):
|
21 |
+
prettier_num = f"{number / comparer:.1f}k"
|
22 |
+
else:
|
23 |
+
prettier_num = str(number)
|
24 |
+
return f"-{prettier_num}" if num < 0 else prettier_num
|
bot/helper/__pycache__/utils.cpython-310.pyc
ADDED
Binary file (3.03 kB). View file
|
|
bot/helper/utils.py
ADDED
@@ -0,0 +1,89 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import hashlib
|
3 |
+
import json
|
4 |
+
import random
|
5 |
+
from collections.abc import Callable
|
6 |
+
from functools import wraps
|
7 |
+
from time import time
|
8 |
+
|
9 |
+
import aiohttp
|
10 |
+
from loguru import logger
|
11 |
+
|
12 |
+
|
13 |
+
def error_handler(delay=3):
|
14 |
+
def decorator(func):
|
15 |
+
@wraps(func)
|
16 |
+
async def wrapper(*args, **kwargs):
|
17 |
+
try:
|
18 |
+
return await func(*args, **kwargs)
|
19 |
+
except Exception as error:
|
20 |
+
logger.error(f"Error in {func.__name__}: {error}")
|
21 |
+
await asyncio.sleep(random.randint(delay, delay * 2))
|
22 |
+
raise
|
23 |
+
|
24 |
+
return wrapper
|
25 |
+
|
26 |
+
return decorator
|
27 |
+
|
28 |
+
|
29 |
+
def handle_request(
|
30 |
+
endpoint: str,
|
31 |
+
full_url: bool = False,
|
32 |
+
method: str = "POST",
|
33 |
+
raise_for_status: bool = True,
|
34 |
+
json_body: dict | None = None,
|
35 |
+
):
|
36 |
+
def decorator(func: Callable) -> Callable:
|
37 |
+
@wraps(func)
|
38 |
+
async def wrapper(self, *args, **kwargs):
|
39 |
+
url = endpoint if full_url else self.api_url + endpoint
|
40 |
+
if method.upper() == "POST":
|
41 |
+
_json_body = kwargs.get("json_body") or json_body or {}
|
42 |
+
set_sign_headers(http_client=self.http_client, data=_json_body)
|
43 |
+
response = await self.http_client.post(url, json=_json_body)
|
44 |
+
elif method.upper() == "GET":
|
45 |
+
response = await self.http_client.get(url)
|
46 |
+
else:
|
47 |
+
msg = "Unsupported HTTP method"
|
48 |
+
raise ValueError(msg)
|
49 |
+
if raise_for_status:
|
50 |
+
response.raise_for_status()
|
51 |
+
|
52 |
+
content_type = response.headers.get("Content-Type", "")
|
53 |
+
if "application/json" in content_type:
|
54 |
+
response_data = await response.json()
|
55 |
+
elif "text/" in content_type:
|
56 |
+
response_data = await response.text()
|
57 |
+
else:
|
58 |
+
response_data = await response.read()
|
59 |
+
return await func(self, response_json=response_data, **kwargs)
|
60 |
+
|
61 |
+
return wrapper
|
62 |
+
|
63 |
+
return decorator
|
64 |
+
|
65 |
+
|
66 |
+
def set_sign_headers(http_client: aiohttp.ClientSession, data: dict) -> None:
|
67 |
+
time_string = str(int(time()))
|
68 |
+
json_string = json.dumps(data)
|
69 |
+
hash_object = hashlib.md5()
|
70 |
+
hash_object.update(f"{time_string}_{json_string}".encode())
|
71 |
+
hash_string = hash_object.hexdigest()
|
72 |
+
http_client.headers["Api-Time"] = time_string
|
73 |
+
http_client.headers["Api-Hash"] = hash_string
|
74 |
+
|
75 |
+
|
76 |
+
def error_handler(delay=3):
|
77 |
+
def decorator(func):
|
78 |
+
@wraps(func)
|
79 |
+
async def wrapper(self, *args, **kwargs):
|
80 |
+
try:
|
81 |
+
return await func(self, *args, **kwargs)
|
82 |
+
except Exception as error:
|
83 |
+
self.logger.error(f"Error in {func.__name__}: {error}")
|
84 |
+
await asyncio.sleep(random.randint(delay, delay * 2))
|
85 |
+
raise
|
86 |
+
|
87 |
+
return wrapper
|
88 |
+
|
89 |
+
return decorator
|
bot/launcher.py
ADDED
@@ -0,0 +1,129 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import random
|
3 |
+
from argparse import ArgumentParser
|
4 |
+
from itertools import cycle
|
5 |
+
from pathlib import Path
|
6 |
+
from typing import NamedTuple
|
7 |
+
|
8 |
+
from better_proxy import Proxy
|
9 |
+
from pyrogram import Client
|
10 |
+
|
11 |
+
from bot.config.logger import log
|
12 |
+
from bot.config.settings import config, logo
|
13 |
+
from bot.core.bot import run_bot
|
14 |
+
from bot.utils import get_session_profiles
|
15 |
+
|
16 |
+
start_text = """
|
17 |
+
Select an action:
|
18 |
+
1. Create session
|
19 |
+
2. Run bot
|
20 |
+
"""
|
21 |
+
|
22 |
+
|
23 |
+
class SessionData(NamedTuple):
|
24 |
+
tg_client: Client
|
25 |
+
session_data: dict
|
26 |
+
|
27 |
+
|
28 |
+
def get_session_names() -> list[str]:
|
29 |
+
return [file.stem for file in sorted(Path("sessions").glob("*.session"))]
|
30 |
+
|
31 |
+
|
32 |
+
async def register_sessions() -> None:
|
33 |
+
session_name = input("\nEnter the session name (press Enter to exit): ")
|
34 |
+
if not session_name:
|
35 |
+
return
|
36 |
+
|
37 |
+
sessions_path = Path("sessions")
|
38 |
+
if not sessions_path.exists():
|
39 |
+
sessions_path.mkdir()
|
40 |
+
|
41 |
+
session = Client(
|
42 |
+
name=session_name,
|
43 |
+
api_id=config.API_ID,
|
44 |
+
api_hash=config.API_HASH,
|
45 |
+
workdir="sessions/",
|
46 |
+
)
|
47 |
+
|
48 |
+
async with session:
|
49 |
+
user_data = await session.get_me()
|
50 |
+
log.success(
|
51 |
+
f"Session added successfully: {user_data.username or user_data.id} | "
|
52 |
+
f"{user_data.first_name or ''} {user_data.last_name or ''}"
|
53 |
+
)
|
54 |
+
|
55 |
+
|
56 |
+
def get_proxies() -> [str | None]:
|
57 |
+
if config.USE_PROXY_FROM_FILE:
|
58 |
+
with Path("proxies.txt").open(encoding="utf-8") as file:
|
59 |
+
return [Proxy.from_str(proxy=row.strip()).as_url for row in file if row.strip()]
|
60 |
+
return None
|
61 |
+
|
62 |
+
|
63 |
+
async def get_tg_clients() -> list[SessionData]:
|
64 |
+
session_names = get_session_names()
|
65 |
+
|
66 |
+
if not session_names:
|
67 |
+
msg = "Not found session files"
|
68 |
+
raise FileNotFoundError(msg)
|
69 |
+
session_profiles = get_session_profiles(session_names)
|
70 |
+
return [
|
71 |
+
SessionData(
|
72 |
+
tg_client=Client(
|
73 |
+
name=session_name,
|
74 |
+
api_id=config.API_ID,
|
75 |
+
api_hash=config.API_HASH,
|
76 |
+
workdir="sessions/",
|
77 |
+
),
|
78 |
+
session_data=session_profiles[session_name],
|
79 |
+
)
|
80 |
+
for session_name in session_names
|
81 |
+
]
|
82 |
+
|
83 |
+
|
84 |
+
async def run_bot_with_delay(tg_client: Client, proxy: str | None, additional_data: dict, session_index: int) -> None:
|
85 |
+
delay = session_index * config.SESSION_AC_DELAY + random.randint(*config.SLEEP_BETWEEN_START)
|
86 |
+
log.bind(session_name=tg_client.name).info(f"Wait {delay} seconds before start")
|
87 |
+
await asyncio.sleep(delay)
|
88 |
+
await run_bot(tg_client=tg_client, proxy=proxy, additional_data=additional_data)
|
89 |
+
|
90 |
+
|
91 |
+
async def run_clients(session_data: list[SessionData]) -> None:
|
92 |
+
proxies = get_proxies() or [None]
|
93 |
+
if config.ADD_LOCAL_MACHINE_AS_IP:
|
94 |
+
proxies.append(None)
|
95 |
+
proxy_cycle = cycle(proxies)
|
96 |
+
await asyncio.gather(
|
97 |
+
*[
|
98 |
+
run_bot_with_delay(
|
99 |
+
tg_client=s_data.tg_client,
|
100 |
+
proxy=next(proxy_cycle),
|
101 |
+
additional_data=s_data.session_data,
|
102 |
+
session_index=index,
|
103 |
+
)
|
104 |
+
for index, s_data in enumerate(session_data)
|
105 |
+
]
|
106 |
+
)
|
107 |
+
|
108 |
+
|
109 |
+
async def start() -> None:
|
110 |
+
print(logo)
|
111 |
+
parser = ArgumentParser()
|
112 |
+
parser.add_argument("-a", "--action", type=int, choices=[1, 2], help="Action to perform (1 or 2)")
|
113 |
+
log.info(f"Detected {len(get_session_names())} sessions | {len(proxy) if (proxy := get_proxies()) else 0} proxies")
|
114 |
+
action = parser.parse_args().action
|
115 |
+
|
116 |
+
if not action:
|
117 |
+
print(start_text)
|
118 |
+
while True:
|
119 |
+
action = input("> ").strip()
|
120 |
+
if action.isdigit() and action in ["1", "2"]:
|
121 |
+
action = int(action)
|
122 |
+
break
|
123 |
+
log.warning("Action must be a number (1 or 2)")
|
124 |
+
|
125 |
+
if action == 1:
|
126 |
+
await register_sessions()
|
127 |
+
elif action == 2:
|
128 |
+
session_data = await get_tg_clients()
|
129 |
+
await run_clients(session_data=session_data)
|
bot/utils.py
ADDED
@@ -0,0 +1,34 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import json
|
2 |
+
from pathlib import Path
|
3 |
+
|
4 |
+
from fake_useragent import UserAgent
|
5 |
+
|
6 |
+
|
7 |
+
def read_session_profiles(sessions: list[str]) -> dict | None:
|
8 |
+
file_path = Path("session_profile.json")
|
9 |
+
if not file_path.exists():
|
10 |
+
return None
|
11 |
+
|
12 |
+
try:
|
13 |
+
with file_path.open(encoding="utf-8") as file:
|
14 |
+
data = json.load(file)
|
15 |
+
return data if all(session in data and len(data[session]) >= 2 for session in sessions) else None
|
16 |
+
except (OSError, json.JSONDecodeError):
|
17 |
+
return None
|
18 |
+
|
19 |
+
|
20 |
+
def get_session_profiles(sessions: list[str]) -> dict:
|
21 |
+
session_profiles = read_session_profiles(sessions)
|
22 |
+
if session_profiles is None:
|
23 |
+
session_profiles = {}
|
24 |
+
ua_generator = UserAgent(browsers=["safari"], os=["ios"], platforms=["mobile", "tablet"])
|
25 |
+
|
26 |
+
for session in sessions:
|
27 |
+
inner = session_profiles.setdefault(session, [])
|
28 |
+
inner.append({"User-Agent": ua_generator.random})
|
29 |
+
inner.append({"proxy": None})
|
30 |
+
|
31 |
+
with Path("session_profile.json").open("w", encoding="utf-8") as file:
|
32 |
+
json.dump(session_profiles, file, ensure_ascii=False, indent=4)
|
33 |
+
|
34 |
+
return session_profiles
|
data.json
ADDED
File without changes
|
docker-compose.yml
ADDED
@@ -0,0 +1,14 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
version: '3'
|
2 |
+
services:
|
3 |
+
bot:
|
4 |
+
container_name: 'MuskEmpireBot'
|
5 |
+
stop_signal: SIGINT
|
6 |
+
build:
|
7 |
+
context: .
|
8 |
+
working_dir: /app
|
9 |
+
volumes:
|
10 |
+
- .:/app
|
11 |
+
entrypoint: "python3 main.py"
|
12 |
+
command: ["-a", "2"]
|
13 |
+
restart: unless-stopped
|
14 |
+
env_file: .env
|
main.py
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
|
3 |
+
from bot import launcher
|
4 |
+
from bot.config.logger import log
|
5 |
+
|
6 |
+
|
7 |
+
async def main() -> None:
|
8 |
+
await launcher.start()
|
9 |
+
|
10 |
+
|
11 |
+
if __name__ == "__main__":
|
12 |
+
try:
|
13 |
+
asyncio.run(main())
|
14 |
+
except KeyboardInterrupt:
|
15 |
+
log.info("Bot stopped by user")
|
proxies.txt
ADDED
@@ -0,0 +1,5 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
type://user:pass@ip:port
|
2 |
+
type://user:pass:ip:port
|
3 |
+
type://ip:port:user:pass
|
4 |
+
type://ip:port@user:pass
|
5 |
+
type://ip:port
|
requirements.txt
ADDED
@@ -0,0 +1,11 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
pydantic-settings==2.3.4
|
2 |
+
aiohttp==3.10.2
|
3 |
+
better-proxy==1.2.0
|
4 |
+
loguru==0.7.2
|
5 |
+
Pyrogram==2.0.106
|
6 |
+
TgCrypto==1.2.5
|
7 |
+
aiocache==0.12.2
|
8 |
+
pytz==2024.1
|
9 |
+
fake-useragent==1.5.1
|
10 |
+
aiohttp-socks==0.9.0
|
11 |
+
aiohttp-proxy==0.1.2
|
sessions/yasir.session
ADDED
Binary file (28.7 kB). View file
|
|
youtube.json
ADDED
@@ -0,0 +1,25 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
{
|
2 |
+
"From Queens to billionaire: the real story of Donald Trump. Controversy, power, wealth, and success": 2007,
|
3 |
+
"How сloud storage works? Episode 28": 75928,
|
4 |
+
"How biomimicry is changing the World. Episode 27": 35527,
|
5 |
+
"The Bezos phenomenon. The billionaire's path from the garage to space.": 0,
|
6 |
+
"How will AI change the world? Episode 29": 34329,
|
7 |
+
"What does a CEO really do? Episode 32": 12932,
|
8 |
+
"Marketing magic: turning Ordinary into Extraordinary. Episode 35": 15235,
|
9 |
+
"The secrets of multicurrency transfers. Episode 38": 22938,
|
10 |
+
"Who’s behind your crypto transfers? Episode 39": 44139,
|
11 |
+
"Check out the guide on how to connect your wallet and complete a test transaction. Once you've reviewed it, connect your wallet to participate in the Airdrop!": 3,
|
12 |
+
"How Sam Altman Is changing the AI world and what’s behind his popularity?": 2023,
|
13 |
+
"Bioengineering: medical miracles of tomorrow. Episode 37": 72837,
|
14 |
+
"Turning Creativity Into Cash: Intellectual Property Valuation. Episode 40": 65340,
|
15 |
+
"Corporate Anthropology: the art of Improving team work. Episode 41": 10941,
|
16 |
+
"The truth about Inflation. Episode 42": 83242,
|
17 |
+
"Mining secrets: How digital currency Is really made? Episode 43": 87843,
|
18 |
+
"Broker secrets: how they make your money work? Episode 44": 62944,
|
19 |
+
"From dorm room to tech Icon: Zuckerberg's story!": 2011,
|
20 |
+
"Why personal branding is the key to success? Episode 47": 24147,
|
21 |
+
"The power of strategic thinking: Think long-term, win big! Episode 50": 89350,
|
22 |
+
"Steve Jobs: Innovator, leader, enigma. The truth about the man behind Apple!": 1989,
|
23 |
+
"The Hidden Power and Harm of Monopolies. Episode 51": 37251,
|
24 |
+
"Reshoring: why Global businesses are coming home? Episode 49": 52649
|
25 |
+
}
|