Spaces:
Running
Running
Upload 93 files
Browse filesThis view is limited to 50 files because it contains too many changes.
See raw diff
- .gitignore +164 -0
- Dockerfile +29 -0
- HellBot/__init__.py +60 -0
- HellBot/__main__.py +33 -0
- HellBot/core/__init__.py +19 -0
- HellBot/core/clients.py +232 -0
- HellBot/core/config.py +155 -0
- HellBot/core/database.py +584 -0
- HellBot/core/initializer.py +94 -0
- HellBot/core/logger.py +19 -0
- HellBot/functions/__init__.py +0 -0
- HellBot/functions/admins.py +24 -0
- HellBot/functions/convert.py +98 -0
- HellBot/functions/driver.py +318 -0
- HellBot/functions/formatter.py +94 -0
- HellBot/functions/images.py +389 -0
- HellBot/functions/media.py +192 -0
- HellBot/functions/paste.py +49 -0
- HellBot/functions/scraping.py +528 -0
- HellBot/functions/sticker.py +135 -0
- HellBot/functions/templates.py +466 -0
- HellBot/functions/tools.py +139 -0
- HellBot/functions/utility.py +241 -0
- HellBot/plugins/__init__.py +0 -0
- HellBot/plugins/bot/__init__.py +29 -0
- HellBot/plugins/bot/bot.py +60 -0
- HellBot/plugins/bot/callbacks.py +279 -0
- HellBot/plugins/bot/forcesub.py +226 -0
- HellBot/plugins/bot/inline.py +39 -0
- HellBot/plugins/bot/sessions.py +183 -0
- HellBot/plugins/bot/users.py +82 -0
- HellBot/plugins/btnsG.py +106 -0
- HellBot/plugins/btnsK.py +45 -0
- HellBot/plugins/decorator.py +67 -0
- HellBot/plugins/help.py +132 -0
- HellBot/plugins/user/__init__.py +16 -0
- HellBot/plugins/user/admins.py +510 -0
- HellBot/plugins/user/afk.py +157 -0
- HellBot/plugins/user/anime.py +184 -0
- HellBot/plugins/user/antiflood.py +200 -0
- HellBot/plugins/user/archiver.py +99 -0
- HellBot/plugins/user/autopost.py +143 -0
- HellBot/plugins/user/blacklist.py +78 -0
- HellBot/plugins/user/bot.py +164 -0
- HellBot/plugins/user/carbon.py +73 -0
- HellBot/plugins/user/climate.py +140 -0
- HellBot/plugins/user/clone.py +106 -0
- HellBot/plugins/user/convert.py +216 -0
- HellBot/plugins/user/core.py +321 -0
- HellBot/plugins/user/downloads.py +85 -0
.gitignore
ADDED
@@ -0,0 +1,164 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Byte-compiled / optimized / DLL files
|
2 |
+
__pycache__/
|
3 |
+
*.py[cod]
|
4 |
+
*$py.class
|
5 |
+
|
6 |
+
# C extensions
|
7 |
+
*.so
|
8 |
+
|
9 |
+
# Distribution / packaging
|
10 |
+
.Python
|
11 |
+
build/
|
12 |
+
develop-eggs/
|
13 |
+
dist/
|
14 |
+
downloads/
|
15 |
+
eggs/
|
16 |
+
.eggs/
|
17 |
+
lib/
|
18 |
+
lib64/
|
19 |
+
parts/
|
20 |
+
sdist/
|
21 |
+
var/
|
22 |
+
wheels/
|
23 |
+
share/python-wheels/
|
24 |
+
*.egg-info/
|
25 |
+
.installed.cfg
|
26 |
+
*.egg
|
27 |
+
MANIFEST
|
28 |
+
|
29 |
+
# PyInstaller
|
30 |
+
# Usually these files are written by a python script from a template
|
31 |
+
# before PyInstaller builds the exe, so as to inject date/other infos into it.
|
32 |
+
*.manifest
|
33 |
+
*.spec
|
34 |
+
|
35 |
+
# Installer logs
|
36 |
+
pip-log.txt
|
37 |
+
pip-delete-this-directory.txt
|
38 |
+
|
39 |
+
# Unit test / coverage reports
|
40 |
+
htmlcov/
|
41 |
+
.tox/
|
42 |
+
.nox/
|
43 |
+
.coverage
|
44 |
+
.coverage.*
|
45 |
+
.cache
|
46 |
+
nosetests.xml
|
47 |
+
coverage.xml
|
48 |
+
*.cover
|
49 |
+
*.py,cover
|
50 |
+
.hypothesis/
|
51 |
+
.pytest_cache/
|
52 |
+
cover/
|
53 |
+
|
54 |
+
# Translations
|
55 |
+
*.mo
|
56 |
+
*.pot
|
57 |
+
|
58 |
+
# Django stuff:
|
59 |
+
*.log
|
60 |
+
local_settings.py
|
61 |
+
db.sqlite3
|
62 |
+
db.sqlite3-journal
|
63 |
+
|
64 |
+
# Flask stuff:
|
65 |
+
instance/
|
66 |
+
.webassets-cache
|
67 |
+
|
68 |
+
# Scrapy stuff:
|
69 |
+
.scrapy
|
70 |
+
|
71 |
+
# Sphinx documentation
|
72 |
+
docs/_build/
|
73 |
+
|
74 |
+
# PyBuilder
|
75 |
+
.pybuilder/
|
76 |
+
target/
|
77 |
+
|
78 |
+
# Jupyter Notebook
|
79 |
+
.ipynb_checkpoints
|
80 |
+
|
81 |
+
# IPython
|
82 |
+
profile_default/
|
83 |
+
ipython_config.py
|
84 |
+
|
85 |
+
# pyenv
|
86 |
+
# For a library or package, you might want to ignore these files since the code is
|
87 |
+
# intended to run in multiple environments; otherwise, check them in:
|
88 |
+
# .python-version
|
89 |
+
|
90 |
+
# pipenv
|
91 |
+
# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
|
92 |
+
# However, in case of collaboration, if having platform-specific dependencies or dependencies
|
93 |
+
# having no cross-platform support, pipenv may install dependencies that don't work, or not
|
94 |
+
# install all needed dependencies.
|
95 |
+
#Pipfile.lock
|
96 |
+
|
97 |
+
# poetry
|
98 |
+
# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
|
99 |
+
# This is especially recommended for binary packages to ensure reproducibility, and is more
|
100 |
+
# commonly ignored for libraries.
|
101 |
+
# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
|
102 |
+
#poetry.lock
|
103 |
+
|
104 |
+
# pdm
|
105 |
+
# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
|
106 |
+
#pdm.lock
|
107 |
+
# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
|
108 |
+
# in version control.
|
109 |
+
# https://pdm.fming.dev/#use-with-ide
|
110 |
+
.pdm.toml
|
111 |
+
|
112 |
+
# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
|
113 |
+
__pypackages__/
|
114 |
+
|
115 |
+
# Celery stuff
|
116 |
+
celerybeat-schedule
|
117 |
+
celerybeat.pid
|
118 |
+
|
119 |
+
# SageMath parsed files
|
120 |
+
*.sage.py
|
121 |
+
|
122 |
+
# Environments
|
123 |
+
.env
|
124 |
+
.venv
|
125 |
+
env/
|
126 |
+
venv/
|
127 |
+
ENV/
|
128 |
+
env.bak/
|
129 |
+
venv.bak/
|
130 |
+
|
131 |
+
# Spyder project settings
|
132 |
+
.spyderproject
|
133 |
+
.spyproject
|
134 |
+
|
135 |
+
# Rope project settings
|
136 |
+
.ropeproject
|
137 |
+
|
138 |
+
# mkdocs documentation
|
139 |
+
/site
|
140 |
+
|
141 |
+
# mypy
|
142 |
+
.mypy_cache/
|
143 |
+
.dmypy.json
|
144 |
+
dmypy.json
|
145 |
+
|
146 |
+
# Pyre type checker
|
147 |
+
.pyre/
|
148 |
+
|
149 |
+
# pytype static type analyzer
|
150 |
+
.pytype/
|
151 |
+
|
152 |
+
# Cython debug symbols
|
153 |
+
cython_debug/
|
154 |
+
|
155 |
+
# PyCharm
|
156 |
+
# JetBrains specific template is maintained in a separate JetBrains.gitignore that can
|
157 |
+
# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
|
158 |
+
# and can be added to the global gitignore or merged into this file. For a more nuclear
|
159 |
+
# option (not recommended) you can uncomment the following to ignore the entire idea folder.
|
160 |
+
#.idea/
|
161 |
+
HellBot.session
|
162 |
+
HellBot.session-journal
|
163 |
+
test.py
|
164 |
+
.vscode/
|
Dockerfile
ADDED
@@ -0,0 +1,29 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
FROM python:3.9.5-buster
|
2 |
+
|
3 |
+
RUN apt-get update
|
4 |
+
|
5 |
+
RUN apt-get install --no-install-recommends -y python3-dev python3-pip python3-virtualenv git mediainfo nano ffmpeg unzip
|
6 |
+
|
7 |
+
COPY ./reqs.txt /Hellbot/reqs.txt
|
8 |
+
|
9 |
+
RUN pip3 install --no-cache-dir -U -r /Hellbot/reqs.txt
|
10 |
+
|
11 |
+
WORKDIR /Hellbot
|
12 |
+
|
13 |
+
RUN pip3 install -U pip
|
14 |
+
|
15 |
+
COPY . .
|
16 |
+
|
17 |
+
RUN chown -R 1000:0 .
|
18 |
+
RUN chmod 777 .
|
19 |
+
RUN chown -R 1000:0 /usr
|
20 |
+
RUN chmod 777 /usr
|
21 |
+
|
22 |
+
EXPOSE 7860
|
23 |
+
|
24 |
+
# Add a script for periodic redeployment
|
25 |
+
COPY start.sh /Hellbot/start.sh
|
26 |
+
RUN chmod +x /Hellbot/start.sh
|
27 |
+
|
28 |
+
# Run the restart script
|
29 |
+
CMD ["bash", "-c", "python3 server.py & /Hellbot/start.sh"]
|
HellBot/__init__.py
ADDED
@@ -0,0 +1,60 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import time
|
3 |
+
from platform import python_version
|
4 |
+
|
5 |
+
import heroku3
|
6 |
+
from pyrogram import __version__ as pyrogram_version
|
7 |
+
|
8 |
+
from .core import LOGS, Config
|
9 |
+
|
10 |
+
START_TIME = time.time()
|
11 |
+
|
12 |
+
|
13 |
+
__version__ = {
|
14 |
+
"hellbot": "3.0",
|
15 |
+
"pyrogram": pyrogram_version,
|
16 |
+
"python": python_version(),
|
17 |
+
}
|
18 |
+
|
19 |
+
|
20 |
+
try:
|
21 |
+
if Config.HEROKU_APIKEY is not None and Config.HEROKU_APPNAME is not None:
|
22 |
+
HEROKU_APP = heroku3.from_key(Config.HEROKU_APIKEY).apps()[
|
23 |
+
Config.HEROKU_APPNAME
|
24 |
+
]
|
25 |
+
else:
|
26 |
+
HEROKU_APP = None
|
27 |
+
except Exception as e:
|
28 |
+
LOGS.error(f"Heroku Api - {e}")
|
29 |
+
HEROKU_APP = None
|
30 |
+
|
31 |
+
|
32 |
+
if Config.API_HASH is None:
|
33 |
+
LOGS.error("Please set your API_HASH !")
|
34 |
+
quit(1)
|
35 |
+
|
36 |
+
if Config.API_ID == 0:
|
37 |
+
LOGS.error("Please set your API_ID !")
|
38 |
+
quit(1)
|
39 |
+
|
40 |
+
if Config.BOT_TOKEN is None:
|
41 |
+
LOGS.error("Please set your BOT_TOKEN !")
|
42 |
+
quit(1)
|
43 |
+
|
44 |
+
if Config.DATABASE_URL is None:
|
45 |
+
LOGS.error("Please set your DATABASE_URL !")
|
46 |
+
quit(1)
|
47 |
+
|
48 |
+
if Config.LOGGER_ID == 0:
|
49 |
+
LOGS.error("Please set your LOGGER_ID !")
|
50 |
+
quit(1)
|
51 |
+
|
52 |
+
if Config.OWNER_ID == 0:
|
53 |
+
LOGS.error("Please set your OWNER_ID !")
|
54 |
+
quit(1)
|
55 |
+
|
56 |
+
if not os.path.isdir(Config.DWL_DIR):
|
57 |
+
os.makedirs(Config.DWL_DIR)
|
58 |
+
|
59 |
+
if not os.path.isdir(Config.TEMP_DIR):
|
60 |
+
os.makedirs(Config.TEMP_DIR)
|
HellBot/__main__.py
ADDED
@@ -0,0 +1,33 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import idle
|
2 |
+
|
3 |
+
from Hellbot import __version__
|
4 |
+
from Hellbot.core import (
|
5 |
+
Config,
|
6 |
+
ForcesubSetup,
|
7 |
+
GachaBotsSetup,
|
8 |
+
TemplateSetup,
|
9 |
+
UserSetup,
|
10 |
+
db,
|
11 |
+
hellbot,
|
12 |
+
)
|
13 |
+
from Hellbot.functions.tools import initialize_git
|
14 |
+
from Hellbot.functions.utility import BList, Flood, TGraph
|
15 |
+
|
16 |
+
|
17 |
+
async def main():
|
18 |
+
await hellbot.startup()
|
19 |
+
await db.connect()
|
20 |
+
await UserSetup()
|
21 |
+
await ForcesubSetup()
|
22 |
+
await GachaBotsSetup()
|
23 |
+
await TemplateSetup()
|
24 |
+
await Flood.updateFromDB()
|
25 |
+
await BList.updateBlacklists()
|
26 |
+
await TGraph.setup()
|
27 |
+
await initialize_git(Config.PLUGINS_REPO)
|
28 |
+
await hellbot.start_message(__version__)
|
29 |
+
await idle()
|
30 |
+
|
31 |
+
|
32 |
+
if __name__ == "__main__":
|
33 |
+
hellbot.run(main())
|
HellBot/core/__init__.py
ADDED
@@ -0,0 +1,19 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from .clients import hellbot
|
2 |
+
from .config import ENV, Config, Limits, Symbols
|
3 |
+
from .database import db
|
4 |
+
from .initializer import ForcesubSetup, GachaBotsSetup, TemplateSetup, UserSetup
|
5 |
+
from .logger import LOGS
|
6 |
+
|
7 |
+
__all__ = [
|
8 |
+
"hellbot",
|
9 |
+
"ENV",
|
10 |
+
"Config",
|
11 |
+
"Limits",
|
12 |
+
"Symbols",
|
13 |
+
"db",
|
14 |
+
"ForcesubSetup",
|
15 |
+
"GachaBotsSetup",
|
16 |
+
"TemplateSetup",
|
17 |
+
"UserSetup",
|
18 |
+
"LOGS",
|
19 |
+
]
|
HellBot/core/clients.py
ADDED
@@ -0,0 +1,232 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import glob
|
3 |
+
import importlib
|
4 |
+
import os
|
5 |
+
import sys
|
6 |
+
from pathlib import Path
|
7 |
+
|
8 |
+
import pyroaddon # pylint: disable=unused-import
|
9 |
+
from pyrogram import Client
|
10 |
+
from pyrogram.enums import ParseMode
|
11 |
+
from pyrogram.types import InlineKeyboardButton, InlineKeyboardMarkup, Message
|
12 |
+
|
13 |
+
from .config import ENV, Config, Symbols
|
14 |
+
from .database import db
|
15 |
+
from .logger import LOGS
|
16 |
+
|
17 |
+
|
18 |
+
class HellClient(Client):
|
19 |
+
def __init__(self) -> None:
|
20 |
+
self.users: list[Client] = []
|
21 |
+
self.bot: Client = Client(
|
22 |
+
name="HellBot",
|
23 |
+
api_id=Config.API_ID,
|
24 |
+
api_hash=Config.API_HASH,
|
25 |
+
bot_token=Config.BOT_TOKEN,
|
26 |
+
plugins=dict(root="Hellbot.plugins.bot"),
|
27 |
+
)
|
28 |
+
|
29 |
+
async def start_user(self) -> None:
|
30 |
+
sessions = await db.get_all_sessions()
|
31 |
+
for i, session in enumerate(sessions):
|
32 |
+
try:
|
33 |
+
client = Client(
|
34 |
+
name=f"HellUser#{i + 1}",
|
35 |
+
api_id=Config.API_ID,
|
36 |
+
api_hash=Config.API_HASH,
|
37 |
+
session_string=session["session"],
|
38 |
+
)
|
39 |
+
await client.start()
|
40 |
+
me = await client.get_me()
|
41 |
+
self.users.append(client)
|
42 |
+
LOGS.info(
|
43 |
+
f"{Symbols.arrow_right * 2} Started User {i + 1}: '{me.first_name}' {Symbols.arrow_left * 2}"
|
44 |
+
)
|
45 |
+
is_in_logger = await self.validate_logger(client)
|
46 |
+
if not is_in_logger:
|
47 |
+
LOGS.warning(
|
48 |
+
f"Client #{i+1}: '{me.first_name}' is not in Logger Group! Check and add manually for proper functioning."
|
49 |
+
)
|
50 |
+
try:
|
51 |
+
await client.join_chat("https://t.me/+wQyUMn4891Q2OTVh") # Channel
|
52 |
+
except:
|
53 |
+
pass
|
54 |
+
# try:
|
55 |
+
# await client.join_chat("https://t.me/+P4Ekwk7P7Rk3NzA9") # Group
|
56 |
+
# except:
|
57 |
+
# pass
|
58 |
+
except Exception as e:
|
59 |
+
LOGS.error(f"{i + 1}: {e}")
|
60 |
+
continue
|
61 |
+
|
62 |
+
async def start_bot(self) -> None:
|
63 |
+
await self.bot.start()
|
64 |
+
me = await self.bot.get_me()
|
65 |
+
LOGS.info(
|
66 |
+
f"{Symbols.arrow_right * 2} Started HellBot Client: '{me.username}' {Symbols.arrow_left * 2}"
|
67 |
+
)
|
68 |
+
|
69 |
+
async def load_plugin(self) -> None:
|
70 |
+
count = 0
|
71 |
+
files = glob.glob("Hellbot/plugins/user/*.py")
|
72 |
+
unload = await db.get_env(ENV.unload_plugins) or ""
|
73 |
+
unload = unload.split(" ")
|
74 |
+
for file in files:
|
75 |
+
with open(file) as f:
|
76 |
+
path = Path(f.name)
|
77 |
+
shortname = path.stem.replace(".py", "")
|
78 |
+
if shortname in unload:
|
79 |
+
os.remove(Path(f"Hellbot/plugins/user/{shortname}.py"))
|
80 |
+
continue
|
81 |
+
if shortname.startswith("__"):
|
82 |
+
continue
|
83 |
+
fpath = Path(f"Hellbot/plugins/user/{shortname}.py")
|
84 |
+
name = "Hellbot.plugins.user." + shortname
|
85 |
+
spec = importlib.util.spec_from_file_location(name, fpath)
|
86 |
+
load = importlib.util.module_from_spec(spec)
|
87 |
+
spec.loader.exec_module(load)
|
88 |
+
sys.modules["Hellbot.plugins.user." + shortname] = load
|
89 |
+
count += 1
|
90 |
+
f.close()
|
91 |
+
LOGS.info(
|
92 |
+
f"{Symbols.bullet * 3} Loaded User Plugin: '{count}' {Symbols.bullet * 3}"
|
93 |
+
)
|
94 |
+
|
95 |
+
async def validate_logger(self, client: Client) -> bool:
|
96 |
+
try:
|
97 |
+
await client.get_chat_member(Config.LOGGER_ID, "me")
|
98 |
+
return True
|
99 |
+
except Exception:
|
100 |
+
return await self.join_logger(client)
|
101 |
+
|
102 |
+
async def join_logger(self, client: Client) -> bool:
|
103 |
+
try:
|
104 |
+
invite_link = await self.bot.export_chat_invite_link(Config.LOGGER_ID)
|
105 |
+
await client.join_chat(invite_link)
|
106 |
+
return True
|
107 |
+
except Exception:
|
108 |
+
return False
|
109 |
+
|
110 |
+
async def start_message(self, version: dict) -> None:
|
111 |
+
await self.bot.send_animation(
|
112 |
+
Config.LOGGER_ID,
|
113 |
+
"https://te.legra.ph/file/8deca5343c64d9db9401f.mp4",
|
114 |
+
f"**{Symbols.check_mark} 𝖧𝖾𝗅𝗅𝖡𝗈𝗍 𝗂𝗌 𝗇𝗈𝗐 𝖮𝗇𝗅𝗂𝗇𝖾!**\n\n"
|
115 |
+
f"**{Symbols.triangle_right} 𝖢𝗅𝗂𝖾𝗇𝗍𝗌:** `{len(self.users)}`\n"
|
116 |
+
f"**{Symbols.triangle_right} 𝖯𝗅𝗎𝗀𝗂𝗇𝗌:** `{len(Config.CMD_MENU)}`\n"
|
117 |
+
f"**{Symbols.triangle_right} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽𝗌:** `{len(Config.CMD_INFO)}`\n"
|
118 |
+
f"**{Symbols.triangle_right} 𝖲𝗍𝖺𝗇 𝖴𝗌𝖾𝗋𝗌:** `{len(Config.STAN_USERS)}`\n"
|
119 |
+
f"**{Symbols.triangle_right} 𝖠𝗎𝗍𝗁 𝖴𝗌𝖾𝗋𝗌:** `{len(Config.AUTH_USERS)}`\n\n"
|
120 |
+
f"**{Symbols.triangle_right} 𝖧𝖾𝗅𝗅𝖡𝗈𝗍 𝖵𝖾𝗋𝗌𝗂𝗈𝗇:** `{version['hellbot']}`\n"
|
121 |
+
f"**{Symbols.triangle_right} 𝖯𝗒𝗋𝗈𝗀𝗋𝖺𝗆 𝖵𝖾𝗋𝗌𝗂𝗈𝗇:** `{version['pyrogram']}`\n"
|
122 |
+
f"**{Symbols.triangle_right} 𝖯𝗒𝗍𝗁𝗈𝗇 𝖵𝖾𝗋𝗌𝗂𝗈𝗇:** `{version['python']}`\n\n"
|
123 |
+
f"**</> @HellBot_Networks**",
|
124 |
+
parse_mode=ParseMode.MARKDOWN,
|
125 |
+
disable_notification=True,
|
126 |
+
reply_markup=InlineKeyboardMarkup(
|
127 |
+
[
|
128 |
+
[
|
129 |
+
InlineKeyboardButton("💫 Start Me", url=f"https://t.me/{self.bot.me.username}?start=start"),
|
130 |
+
InlineKeyboardButton("💖 Repo", url="https://github.com/The-HellBot/HellBot"),
|
131 |
+
],
|
132 |
+
[
|
133 |
+
InlineKeyboardButton("🍀 HellBot Networks 🍀", url="https://t.me/hellbot_networks"),
|
134 |
+
],
|
135 |
+
]
|
136 |
+
),
|
137 |
+
)
|
138 |
+
|
139 |
+
async def startup(self) -> None:
|
140 |
+
LOGS.info(
|
141 |
+
f"{Symbols.bullet * 3} Starting HellBot Client & User {Symbols.bullet * 3}"
|
142 |
+
)
|
143 |
+
await self.start_bot()
|
144 |
+
await self.start_user()
|
145 |
+
await self.load_plugin()
|
146 |
+
|
147 |
+
|
148 |
+
class CustomMethods(HellClient):
|
149 |
+
async def input(self, message: Message) -> str:
|
150 |
+
"""Get the input from the user"""
|
151 |
+
if len(message.command) < 2:
|
152 |
+
output = ""
|
153 |
+
|
154 |
+
else:
|
155 |
+
try:
|
156 |
+
output = message.text.split(" ", 1)[1].strip() or ""
|
157 |
+
except IndexError:
|
158 |
+
output = ""
|
159 |
+
|
160 |
+
return output
|
161 |
+
|
162 |
+
async def edit(
|
163 |
+
self,
|
164 |
+
message: Message,
|
165 |
+
text: str,
|
166 |
+
parse_mode: ParseMode = ParseMode.DEFAULT,
|
167 |
+
no_link_preview: bool = True,
|
168 |
+
) -> Message:
|
169 |
+
"""Edit or Reply to a message, if possible"""
|
170 |
+
if message.from_user and message.from_user.id in Config.STAN_USERS:
|
171 |
+
if message.reply_to_message:
|
172 |
+
return await message.reply_to_message.reply_text(
|
173 |
+
text,
|
174 |
+
parse_mode=parse_mode,
|
175 |
+
disable_web_page_preview=no_link_preview,
|
176 |
+
)
|
177 |
+
return await message.reply_text(
|
178 |
+
text, parse_mode=parse_mode, disable_web_page_preview=no_link_preview
|
179 |
+
)
|
180 |
+
return await message.edit_text(
|
181 |
+
text, parse_mode=parse_mode, disable_web_page_preview=no_link_preview
|
182 |
+
)
|
183 |
+
|
184 |
+
async def _delete(self, message: Message, delay: int = 0) -> None:
|
185 |
+
"""Delete a message after a certain period of time"""
|
186 |
+
await asyncio.sleep(delay)
|
187 |
+
await message.delete()
|
188 |
+
|
189 |
+
async def delete(
|
190 |
+
self, message: Message, text: str, delete: int = 10, in_background: bool = True
|
191 |
+
) -> None:
|
192 |
+
"""Edit a message and delete it after a certain period of time"""
|
193 |
+
to_del = await self.edit(message, text)
|
194 |
+
if in_background:
|
195 |
+
asyncio.create_task(self._delete(to_del, delete))
|
196 |
+
else:
|
197 |
+
await self._delete(to_del, delete)
|
198 |
+
|
199 |
+
async def error(self, message: Message, text: str, delete: int = 10) -> None:
|
200 |
+
"""Edit an error message and delete it after a certain period of time if mentioned"""
|
201 |
+
to_del = await self.edit(message, f"{Symbols.cross_mark} **Error:** \n\n{text}")
|
202 |
+
if delete:
|
203 |
+
asyncio.create_task(self._delete(to_del, delete))
|
204 |
+
|
205 |
+
async def _log(self, tag: str, text: str, file: str = None) -> None:
|
206 |
+
"""Log a message to the Logger Group"""
|
207 |
+
msg = f"**#{tag.upper()}**\n\n{text}"
|
208 |
+
try:
|
209 |
+
if file:
|
210 |
+
try:
|
211 |
+
await self.bot.send_document(Config.LOGGER_ID, file, caption=msg)
|
212 |
+
except:
|
213 |
+
await self.bot.send_message(
|
214 |
+
Config.LOGGER_ID, msg, disable_web_page_preview=True
|
215 |
+
)
|
216 |
+
else:
|
217 |
+
await self.bot.send_message(
|
218 |
+
Config.LOGGER_ID, msg, disable_web_page_preview=True
|
219 |
+
)
|
220 |
+
except Exception as e:
|
221 |
+
raise Exception(f"{Symbols.cross_mark} LogErr: {e}")
|
222 |
+
|
223 |
+
async def check_and_log(self, tag: str, text: str, file: str = None) -> None:
|
224 |
+
"""Check if :
|
225 |
+
\n-> the Logger Group is available
|
226 |
+
\n-> the logging is enabled"""
|
227 |
+
status = await db.get_env(ENV.is_logger)
|
228 |
+
if status and status.lower() == "true":
|
229 |
+
await self._log(tag, text, file)
|
230 |
+
|
231 |
+
|
232 |
+
hellbot = CustomMethods()
|
HellBot/core/config.py
ADDED
@@ -0,0 +1,155 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from os import getenv
|
2 |
+
|
3 |
+
from dotenv import load_dotenv
|
4 |
+
from pyrogram import filters
|
5 |
+
|
6 |
+
load_dotenv()
|
7 |
+
|
8 |
+
|
9 |
+
class Config:
|
10 |
+
# editable configs
|
11 |
+
API_HASH = getenv("API_HASH", None)
|
12 |
+
API_ID = int(getenv("API_ID", 0))
|
13 |
+
BOT_TOKEN = getenv("BOT_TOKEN", None)
|
14 |
+
DATABASE_URL = getenv("DATABASE_URL", None)
|
15 |
+
HANDLERS = getenv("HANDLERS", ". ! ?").strip().split()
|
16 |
+
LOGGER_ID = int(getenv("LOGGER_ID", 0))
|
17 |
+
OWNER_ID = int(getenv("OWNER_ID", 0))
|
18 |
+
|
19 |
+
# heroku related configs
|
20 |
+
HEROKU_APPNAME = getenv("HEROKU_APPNAME", None)
|
21 |
+
HEROKU_APIKEY = getenv("HEROKU_APIKEY", None)
|
22 |
+
|
23 |
+
# github related configs
|
24 |
+
PLUGINS_REPO = getenv("PLUGINS_REPO", "The-HellBot/Plugins")
|
25 |
+
DEPLOY_REPO = getenv("DEPLOY_REPO", "The-HellBot/Hellbot")
|
26 |
+
|
27 |
+
# storage dir: you may or may not edit
|
28 |
+
DWL_DIR = "./downloads/"
|
29 |
+
TEMP_DIR = "./temp/"
|
30 |
+
CHROME_BIN = getenv("CHROME_BIN", "/app/.chrome-for-testing/chrome-linux64/chrome")
|
31 |
+
CHROME_DRIVER = getenv(
|
32 |
+
"CHROME_DRIVER", "/app/.chrome-for-testing/chromedriver-linux64/chromedriver"
|
33 |
+
)
|
34 |
+
FONT_PATH = "./Hellbot/resources/fonts/Montserrat.ttf"
|
35 |
+
|
36 |
+
# users config: do not edit
|
37 |
+
AUTH_USERS = filters.user()
|
38 |
+
BANNED_USERS = filters.user()
|
39 |
+
GACHA_BOTS = filters.user()
|
40 |
+
MUTED_USERS = filters.user()
|
41 |
+
DEVS = filters.user([1432756163, 1874070588, 1533682758])
|
42 |
+
STAN_USERS = filters.user()
|
43 |
+
FORCESUBS = filters.chat()
|
44 |
+
|
45 |
+
# Global config: do not edit
|
46 |
+
AFK_CACHE = {}
|
47 |
+
BOT_CMD_INFO = {}
|
48 |
+
BOT_CMD_MENU = {}
|
49 |
+
BOT_HELP = {}
|
50 |
+
CMD_INFO = {}
|
51 |
+
CMD_MENU = {}
|
52 |
+
HELP_DICT = {}
|
53 |
+
TEMPLATES = {}
|
54 |
+
|
55 |
+
|
56 |
+
class ENV:
|
57 |
+
"""Database ENV Names"""
|
58 |
+
|
59 |
+
airing_template = "AIRING_TEMPLATE"
|
60 |
+
airpollution_template = "AIRPOLLUTION_TEMPLATE"
|
61 |
+
alive_pic = "ALIVE_PIC"
|
62 |
+
alive_template = "ALIVE_TEMPLATE"
|
63 |
+
anilist_user_template = "ANILIST_USER_TEMPLATE"
|
64 |
+
anime_template = "ANIME_TEMPLATE"
|
65 |
+
btn_in_help = "BUTTONS_IN_HELP"
|
66 |
+
character_template = "CHARACTER_TEMPLATE"
|
67 |
+
chat_info_template = "CHAT_INFO_TEMPLATE"
|
68 |
+
climate_api = "CLIMATE_API"
|
69 |
+
climate_template = "CLIMATE_TEMPLATE"
|
70 |
+
command_template = "COMMAND_TEMPLATE"
|
71 |
+
currency_api = "CURRENCY_API"
|
72 |
+
custom_pmpermit = "CUSTOM_PMPERMIT"
|
73 |
+
gban_template = "GBAN_TEMPLATE"
|
74 |
+
github_user_template = "GITHUB_USER_TEMPLATE"
|
75 |
+
help_emoji = "HELP_EMOJI"
|
76 |
+
help_template = "HELP_TEMPLATE"
|
77 |
+
is_logger = "IS_LOGGER"
|
78 |
+
lyrics_api = "LYRICS_API"
|
79 |
+
manga_template = "MANGA_TEMPLATE"
|
80 |
+
ocr_api = "OCR_API"
|
81 |
+
ping_pic = "PING_PIC"
|
82 |
+
ping_template = "PING_TEMPLATE"
|
83 |
+
pm_logger = "PM_LOGGER"
|
84 |
+
pm_max_spam = "PM_MAX_SPAM"
|
85 |
+
pmpermit = "PMPERMIT"
|
86 |
+
pmpermit_pic = "PMPERMIT_PIC"
|
87 |
+
remove_bg_api = "REMOVE_BG_API"
|
88 |
+
thumbnail_url = "THUMBNAIL_URL"
|
89 |
+
statistics_template = "STATISTICS_TEMPLATE"
|
90 |
+
sticker_packname = "STICKER_PACKNAME"
|
91 |
+
tag_logger = "TAG_LOGGER"
|
92 |
+
telegraph_account = "TELEGRAPH_ACCOUNT"
|
93 |
+
time_zone = "TIME_ZONE"
|
94 |
+
unload_plugins = "UNLOAD_PLUGINS"
|
95 |
+
unsplash_api = "UNSPLASH_API"
|
96 |
+
usage_template = "USAGE_TEMPLATE"
|
97 |
+
user_info_template = "USER_INFO_TEMPLATE"
|
98 |
+
|
99 |
+
|
100 |
+
class Limits:
|
101 |
+
AdminRoleLength = 16
|
102 |
+
AdminsLimit = 50
|
103 |
+
BioLength = 70
|
104 |
+
BotDescriptionLength = 512
|
105 |
+
BotInfoLength = 120
|
106 |
+
BotsLimit = 20
|
107 |
+
CaptionLength = 1024
|
108 |
+
ChannelGroupsLimit = 500
|
109 |
+
ChatTitleLength = 128
|
110 |
+
FileNameLength = 60
|
111 |
+
MessageLength = 4096
|
112 |
+
NameLength = 64
|
113 |
+
PremiumBioLength = 140
|
114 |
+
PremiumCaptionLength = 2048
|
115 |
+
PremiumChannelGroupsLimit = 1000
|
116 |
+
StickerAniamtedLimit = 50
|
117 |
+
StickerPackNameLength = 64
|
118 |
+
StickerStaticLimit = 120
|
119 |
+
|
120 |
+
|
121 |
+
class Symbols:
|
122 |
+
anchor = "⚘"
|
123 |
+
arrow_left = "«"
|
124 |
+
arrow_right = "»"
|
125 |
+
back = "🔙 back"
|
126 |
+
bullet = "•"
|
127 |
+
check_mark = "✔"
|
128 |
+
close = "🗑️"
|
129 |
+
cross_mark = "✘"
|
130 |
+
diamond_1 = "◇"
|
131 |
+
diamond_2 = "◈"
|
132 |
+
next = "⤚ next"
|
133 |
+
previous = "prev ⤙"
|
134 |
+
radio_select = "◉"
|
135 |
+
radio_unselect = "〇"
|
136 |
+
triangle_left = "◂"
|
137 |
+
triangle_right = "▸"
|
138 |
+
|
139 |
+
|
140 |
+
os_configs = [
|
141 |
+
"API_HASH",
|
142 |
+
"API_ID",
|
143 |
+
"BOT_TOKEN",
|
144 |
+
"DATABASE_URL",
|
145 |
+
"DEPLOY_REPO",
|
146 |
+
"HANDLERS",
|
147 |
+
"HEROKU_APIKEY",
|
148 |
+
"HEROKU_APPNAME",
|
149 |
+
"LOGGER_ID",
|
150 |
+
"OWNER_ID",
|
151 |
+
"PLUGINS_REPO",
|
152 |
+
]
|
153 |
+
all_env: list[str] = [
|
154 |
+
value for key, value in ENV.__dict__.items() if not key.startswith("__")
|
155 |
+
]
|
HellBot/core/database.py
ADDED
@@ -0,0 +1,584 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import datetime
|
2 |
+
import time
|
3 |
+
|
4 |
+
from motor import motor_asyncio
|
5 |
+
from motor.core import AgnosticClient
|
6 |
+
|
7 |
+
from .config import Config, Symbols
|
8 |
+
from .logger import LOGS
|
9 |
+
|
10 |
+
|
11 |
+
class Database:
|
12 |
+
def __init__(self, uri: str) -> None:
|
13 |
+
self.client: AgnosticClient = motor_asyncio.AsyncIOMotorClient(uri)
|
14 |
+
self.db = self.client["Hellbot"]
|
15 |
+
|
16 |
+
self.afk = self.db["afk"]
|
17 |
+
self.antiflood = self.db["antiflood"]
|
18 |
+
self.autopost = self.db["autopost"]
|
19 |
+
self.blacklist = self.db["blacklist"]
|
20 |
+
self.echo = self.db["echo"]
|
21 |
+
self.env = self.db["env"]
|
22 |
+
self.filter = self.db["filter"]
|
23 |
+
self.forcesub = self.db["forcesub"]
|
24 |
+
self.gachabots = self.db["gachabots"]
|
25 |
+
self.gban = self.db["gban"]
|
26 |
+
self.gmute = self.db["gmute"]
|
27 |
+
self.greetings = self.db["greetings"]
|
28 |
+
self.mute = self.db["mute"]
|
29 |
+
self.pmpermit = self.db["pmpermit"]
|
30 |
+
self.session = self.db["session"]
|
31 |
+
self.snips = self.db["snips"]
|
32 |
+
self.stan_users = self.db["stan_users"]
|
33 |
+
|
34 |
+
async def connect(self):
|
35 |
+
try:
|
36 |
+
await self.client.admin.command("ping")
|
37 |
+
LOGS.info(
|
38 |
+
f"{Symbols.bullet * 3} Database Connection Established! {Symbols.bullet * 3}"
|
39 |
+
)
|
40 |
+
except Exception as e:
|
41 |
+
LOGS.info(f"{Symbols.cross_mark} DatabaseErr: {e} ")
|
42 |
+
quit(1)
|
43 |
+
|
44 |
+
def get_datetime(self) -> str:
|
45 |
+
return datetime.datetime.now().strftime("%d/%m/%Y - %H:%M")
|
46 |
+
|
47 |
+
async def set_env(self, name: str, value: str) -> None:
|
48 |
+
await self.env.update_one(
|
49 |
+
{"name": name}, {"$set": {"value": value}}, upsert=True
|
50 |
+
)
|
51 |
+
|
52 |
+
async def get_env(self, name: str) -> str | None:
|
53 |
+
if await self.is_env(name):
|
54 |
+
data = await self.env.find_one({"name": name})
|
55 |
+
return data["value"]
|
56 |
+
return None
|
57 |
+
|
58 |
+
async def rm_env(self, name: str) -> None:
|
59 |
+
await self.env.delete_one({"name": name})
|
60 |
+
|
61 |
+
async def is_env(self, name: str) -> bool:
|
62 |
+
if await self.env.find_one({"name": name}):
|
63 |
+
return True
|
64 |
+
return False
|
65 |
+
|
66 |
+
async def get_all_env(self) -> list:
|
67 |
+
return [i async for i in self.env.find({})]
|
68 |
+
|
69 |
+
async def is_stan(self, client: int, user_id: int) -> bool:
|
70 |
+
if await self.stan_users.find_one({"client": client, "user_id": user_id}):
|
71 |
+
return True
|
72 |
+
return False
|
73 |
+
|
74 |
+
async def add_stan(self, client: int, user_id: int) -> bool:
|
75 |
+
if await self.is_stan(client, user_id):
|
76 |
+
return False
|
77 |
+
await self.stan_users.insert_one(
|
78 |
+
{"client": client, "user_id": user_id, "date": self.get_datetime()}
|
79 |
+
)
|
80 |
+
return True
|
81 |
+
|
82 |
+
async def rm_stan(self, client: int, user_id: int) -> bool:
|
83 |
+
if not await self.is_stan(client, user_id):
|
84 |
+
return False
|
85 |
+
await self.stan_users.delete_one({"client": client, "user_id": user_id})
|
86 |
+
return True
|
87 |
+
|
88 |
+
async def get_stans(self, client: int) -> list:
|
89 |
+
return [i async for i in self.stan_users.find({"client": client})]
|
90 |
+
|
91 |
+
async def get_all_stans(self) -> list:
|
92 |
+
return [i async for i in self.stan_users.find({})]
|
93 |
+
|
94 |
+
async def is_session(self, user_id: int) -> bool:
|
95 |
+
if await self.session.find_one({"user_id": user_id}):
|
96 |
+
return True
|
97 |
+
return False
|
98 |
+
|
99 |
+
async def update_session(self, user_id: int, session: str) -> None:
|
100 |
+
await self.session.update_one(
|
101 |
+
{"user_id": user_id},
|
102 |
+
{"$set": {"session": session, "date": self.get_datetime()}},
|
103 |
+
upsert=True,
|
104 |
+
)
|
105 |
+
|
106 |
+
async def rm_session(self, user_id: int) -> None:
|
107 |
+
await self.session.delete_one({"user_id": user_id})
|
108 |
+
|
109 |
+
async def get_session(self, user_id: int):
|
110 |
+
if not await self.is_session(user_id):
|
111 |
+
return False
|
112 |
+
data = await self.session.find_one({"user_id": user_id})
|
113 |
+
return data
|
114 |
+
|
115 |
+
async def get_all_sessions(self) -> list:
|
116 |
+
return [i async for i in self.session.find({})]
|
117 |
+
|
118 |
+
async def is_gbanned(self, user_id: int) -> bool:
|
119 |
+
if await self.gban.find_one({"user_id": user_id}):
|
120 |
+
return True
|
121 |
+
return False
|
122 |
+
|
123 |
+
async def add_gban(self, user_id: int, reason: str) -> bool:
|
124 |
+
if await self.is_gbanned(user_id):
|
125 |
+
return False
|
126 |
+
await self.gban.insert_one(
|
127 |
+
{"user_id": user_id, "reason": reason, "date": self.get_datetime()}
|
128 |
+
)
|
129 |
+
return True
|
130 |
+
|
131 |
+
async def rm_gban(self, user_id: int):
|
132 |
+
if not await self.is_gbanned(user_id):
|
133 |
+
return None
|
134 |
+
reason = (await self.gban.find_one({"user_id": user_id}))["reason"]
|
135 |
+
await self.gban.delete_one({"user_id": user_id})
|
136 |
+
return reason
|
137 |
+
|
138 |
+
async def get_gban(self) -> list:
|
139 |
+
return [i async for i in self.gban.find({})]
|
140 |
+
|
141 |
+
async def get_gban_user(self, user_id: int) -> dict | None:
|
142 |
+
if not await self.is_gbanned(user_id):
|
143 |
+
return None
|
144 |
+
return await self.gban.find_one({"user_id": user_id})
|
145 |
+
|
146 |
+
async def is_gmuted(self, user_id: int) -> bool:
|
147 |
+
if await self.gmute.find_one({"user_id": user_id}):
|
148 |
+
return True
|
149 |
+
return False
|
150 |
+
|
151 |
+
async def add_gmute(self, user_id: int, reason: str) -> bool:
|
152 |
+
if await self.is_gmuted(user_id):
|
153 |
+
return False
|
154 |
+
await self.gmute.insert_one(
|
155 |
+
{"user_id": user_id, "reason": reason, "date": self.get_datetime()}
|
156 |
+
)
|
157 |
+
return True
|
158 |
+
|
159 |
+
async def rm_gmute(self, user_id: int):
|
160 |
+
if not await self.is_gmuted(user_id):
|
161 |
+
return None
|
162 |
+
reason = (await self.gmute.find_one({"user_id": user_id}))["reason"]
|
163 |
+
await self.gmute.delete_one({"user_id": user_id})
|
164 |
+
return reason
|
165 |
+
|
166 |
+
async def get_gmute(self) -> list:
|
167 |
+
return [i async for i in self.gmute.find({})]
|
168 |
+
|
169 |
+
async def add_mute(self, client: int, user_id: int, chat_id: int, reason: str):
|
170 |
+
await self.mute.update_one(
|
171 |
+
{"client": client, "user_id": user_id, "chat_id": chat_id},
|
172 |
+
{"$set": {"reason": reason, "date": self.get_datetime()}},
|
173 |
+
upsert=True,
|
174 |
+
)
|
175 |
+
|
176 |
+
async def rm_mute(self, client: int, user_id: int, chat_id: int) -> str:
|
177 |
+
reason = (await self.get_mute(client, user_id, chat_id))["reason"]
|
178 |
+
await self.mute.delete_one({"client": client, "user_id": user_id, "chat_id": chat_id})
|
179 |
+
return reason
|
180 |
+
|
181 |
+
async def is_muted(self, client: int, user_id: int, chat_id: int) -> bool:
|
182 |
+
if await self.get_mute(client, user_id, chat_id):
|
183 |
+
return True
|
184 |
+
return False
|
185 |
+
|
186 |
+
async def get_mute(self, client: int, user_id: int, chat_id: int):
|
187 |
+
data = await self.mute.find_one({"client": client, "user_id": user_id, "chat_id": chat_id})
|
188 |
+
return data
|
189 |
+
|
190 |
+
async def set_afk(
|
191 |
+
self, user_id: int, reason: str, media: int, media_type: str
|
192 |
+
) -> None:
|
193 |
+
await self.afk.update_one(
|
194 |
+
{"user_id": user_id},
|
195 |
+
{
|
196 |
+
"$set": {
|
197 |
+
"reason": reason,
|
198 |
+
"time": time.time(),
|
199 |
+
"media": media,
|
200 |
+
"media_type": media_type,
|
201 |
+
}
|
202 |
+
},
|
203 |
+
upsert=True,
|
204 |
+
)
|
205 |
+
|
206 |
+
async def get_afk(self, user_id: int):
|
207 |
+
data = await self.afk.find_one({"user_id": user_id})
|
208 |
+
return data
|
209 |
+
|
210 |
+
async def is_afk(self, user_id: int) -> bool:
|
211 |
+
if await self.afk.find_one({"user_id": user_id}):
|
212 |
+
return True
|
213 |
+
return False
|
214 |
+
|
215 |
+
async def rm_afk(self, user_id: int) -> None:
|
216 |
+
await self.afk.delete_one({"user_id": user_id})
|
217 |
+
|
218 |
+
async def set_flood(self, client_chat: tuple[int, int], settings: dict):
|
219 |
+
await self.antiflood.update_one(
|
220 |
+
{"client": client_chat[0], "chat": client_chat[1]},
|
221 |
+
{"$set": settings},
|
222 |
+
upsert=True,
|
223 |
+
)
|
224 |
+
|
225 |
+
async def get_flood(self, client_chat: tuple[int, int]):
|
226 |
+
data = await self.antiflood.find_one(
|
227 |
+
{"client": client_chat[0], "chat": client_chat[1]}
|
228 |
+
)
|
229 |
+
return data or {}
|
230 |
+
|
231 |
+
async def is_flood(self, client_chat: tuple[int, int]) -> bool:
|
232 |
+
data = await self.get_flood(client_chat)
|
233 |
+
|
234 |
+
if not data:
|
235 |
+
return False
|
236 |
+
|
237 |
+
if data["limit"] == 0:
|
238 |
+
return False
|
239 |
+
|
240 |
+
return True
|
241 |
+
|
242 |
+
async def get_all_floods(self) -> list:
|
243 |
+
return [i async for i in self.antiflood.find({})]
|
244 |
+
|
245 |
+
async def set_autopost(self, client: int, from_channel: int, to_channel: int):
|
246 |
+
await self.autopost.update_one(
|
247 |
+
{"client": client},
|
248 |
+
{
|
249 |
+
"$push": {
|
250 |
+
"autopost": {
|
251 |
+
"from_channel": from_channel,
|
252 |
+
"to_channel": to_channel,
|
253 |
+
"date": self.get_datetime(),
|
254 |
+
}
|
255 |
+
}
|
256 |
+
},
|
257 |
+
upsert=True,
|
258 |
+
)
|
259 |
+
|
260 |
+
async def get_autopost(self, client: int, from_channel: int):
|
261 |
+
data = await self.autopost.find_one(
|
262 |
+
{
|
263 |
+
"client": client,
|
264 |
+
"autopost": {"$elemMatch": {"from_channel": from_channel}},
|
265 |
+
}
|
266 |
+
)
|
267 |
+
return data
|
268 |
+
|
269 |
+
async def is_autopost(
|
270 |
+
self, client: int, from_channel: int, to_channel: int = None
|
271 |
+
) -> bool:
|
272 |
+
if to_channel:
|
273 |
+
data = await self.autopost.find_one(
|
274 |
+
{
|
275 |
+
"client": client,
|
276 |
+
"autopost": {
|
277 |
+
"$elemMatch": {
|
278 |
+
"from_channel": from_channel,
|
279 |
+
"to_channel": to_channel,
|
280 |
+
}
|
281 |
+
},
|
282 |
+
}
|
283 |
+
)
|
284 |
+
else:
|
285 |
+
data = await self.autopost.find_one(
|
286 |
+
{
|
287 |
+
"client": client,
|
288 |
+
"autopost": {"$elemMatch": {"from_channel": from_channel}},
|
289 |
+
}
|
290 |
+
)
|
291 |
+
return True if data else False
|
292 |
+
|
293 |
+
async def rm_autopost(self, client: int, from_channel: int, to_channel: int):
|
294 |
+
await self.autopost.update_one(
|
295 |
+
{"client": client},
|
296 |
+
{
|
297 |
+
"$pull": {
|
298 |
+
"autopost": {
|
299 |
+
"from_channel": from_channel,
|
300 |
+
"to_channel": to_channel,
|
301 |
+
}
|
302 |
+
}
|
303 |
+
},
|
304 |
+
)
|
305 |
+
|
306 |
+
async def get_all_autoposts(self, client: int) -> list:
|
307 |
+
return [i async for i in self.autopost.find({"client": client})]
|
308 |
+
|
309 |
+
async def add_blacklist(self, client: int, chat: int, blacklist: str):
|
310 |
+
await self.blacklist.update_one(
|
311 |
+
{"client": client, "chat": chat},
|
312 |
+
{"$push": {"blacklist": blacklist}},
|
313 |
+
upsert=True,
|
314 |
+
)
|
315 |
+
|
316 |
+
async def rm_blacklist(self, client: int, chat: int, blacklist: str):
|
317 |
+
await self.blacklist.update_one(
|
318 |
+
{"client": client, "chat": chat},
|
319 |
+
{"$pull": {"blacklist": blacklist}},
|
320 |
+
)
|
321 |
+
|
322 |
+
async def is_blacklist(self, client: int, chat: int, blacklist: str) -> bool:
|
323 |
+
blacklists = await self.get_all_blacklists(client, chat)
|
324 |
+
if blacklist in blacklists:
|
325 |
+
return True
|
326 |
+
return False
|
327 |
+
|
328 |
+
async def get_all_blacklists(self, client: int, chat: int) -> list:
|
329 |
+
data = await self.blacklist.find_one({"client": client, "chat": chat})
|
330 |
+
|
331 |
+
if not data:
|
332 |
+
return []
|
333 |
+
|
334 |
+
return data["blacklist"]
|
335 |
+
|
336 |
+
async def get_blacklist_clients(self) -> list:
|
337 |
+
return [i async for i in self.blacklist.find({})]
|
338 |
+
|
339 |
+
async def set_echo(self, client: int, chat: int, user: int):
|
340 |
+
await self.echo.update_one(
|
341 |
+
{"client": client, "chat": chat},
|
342 |
+
{"$push": {"echo": user}},
|
343 |
+
upsert=True,
|
344 |
+
)
|
345 |
+
|
346 |
+
async def rm_echo(self, client: int, chat: int, user: int):
|
347 |
+
await self.echo.update_one(
|
348 |
+
{"client": client, "chat": chat},
|
349 |
+
{"$pull": {"echo": user}},
|
350 |
+
)
|
351 |
+
|
352 |
+
async def is_echo(self, client: int, chat: int, user: int) -> bool:
|
353 |
+
data = await self.get_all_echo(client, chat)
|
354 |
+
if user in data:
|
355 |
+
return True
|
356 |
+
return False
|
357 |
+
|
358 |
+
async def get_all_echo(self, client: int, chat: int) -> list:
|
359 |
+
data = await self.echo.find_one({"client": client, "chat": chat})
|
360 |
+
|
361 |
+
if not data:
|
362 |
+
return []
|
363 |
+
|
364 |
+
return data["echo"]
|
365 |
+
|
366 |
+
async def set_filter(self, client: int, chat: int, keyword: str, msgid: int):
|
367 |
+
await self.filter.update_one(
|
368 |
+
{"client": client, "chat": chat},
|
369 |
+
{"$push": {"filter": {"keyword": keyword, "msgid": msgid}}},
|
370 |
+
upsert=True,
|
371 |
+
)
|
372 |
+
|
373 |
+
async def rm_filter(self, client: int, chat: int, keyword: str):
|
374 |
+
await self.filter.update_one(
|
375 |
+
{"client": client, "chat": chat},
|
376 |
+
{"$pull": {"filter": {"keyword": keyword}}},
|
377 |
+
)
|
378 |
+
|
379 |
+
async def rm_all_filters(self, client: int, chat: int):
|
380 |
+
await self.filter.delete_one({"client": client, "chat": chat})
|
381 |
+
|
382 |
+
async def is_filter(self, client: int, chat: int, keyword: str) -> bool:
|
383 |
+
data = await self.get_filter(client, chat, keyword)
|
384 |
+
return True if data else False
|
385 |
+
|
386 |
+
async def get_filter(self, client: int, chat: int, keyword: str):
|
387 |
+
data = await self.filter.find_one(
|
388 |
+
{
|
389 |
+
"client": client,
|
390 |
+
"chat": chat,
|
391 |
+
"filter": {"$elemMatch": {"keyword": keyword}},
|
392 |
+
}
|
393 |
+
)
|
394 |
+
return data
|
395 |
+
|
396 |
+
async def get_all_filters(self, client: int, chat: int) -> list:
|
397 |
+
data = await self.filter.find_one({"client": client, "chat": chat})
|
398 |
+
|
399 |
+
if not data:
|
400 |
+
return []
|
401 |
+
|
402 |
+
return data["filter"]
|
403 |
+
|
404 |
+
async def set_snip(self, client: int, chat: int, keyword: str, msgid: int):
|
405 |
+
await self.snips.update_one(
|
406 |
+
{"client": client, "chat": chat},
|
407 |
+
{"$push": {"snips": {"keyword": keyword, "msgid": msgid}}},
|
408 |
+
upsert=True,
|
409 |
+
)
|
410 |
+
|
411 |
+
async def rm_snip(self, client: int, chat: int, keyword: str):
|
412 |
+
await self.snips.update_one(
|
413 |
+
{"client": client, "chat": chat},
|
414 |
+
{"$pull": {"snips": {"keyword": keyword}}},
|
415 |
+
)
|
416 |
+
|
417 |
+
async def rm_all_snips(self, client: int, chat: int):
|
418 |
+
await self.snips.delete_one({"client": client, "chat": chat})
|
419 |
+
|
420 |
+
async def is_snip(self, client: int, chat: int, keyword: str) -> bool:
|
421 |
+
data = await self.get_snip(client, chat, keyword)
|
422 |
+
return True if data else False
|
423 |
+
|
424 |
+
async def get_snip(self, client: int, chat: int, keyword: str):
|
425 |
+
data = await self.snips.find_one(
|
426 |
+
{
|
427 |
+
"client": client,
|
428 |
+
"chat": chat,
|
429 |
+
"snips": {"$elemMatch": {"keyword": keyword}},
|
430 |
+
}
|
431 |
+
)
|
432 |
+
return data
|
433 |
+
|
434 |
+
async def get_all_snips(self, client: int, chat: int) -> list:
|
435 |
+
data = await self.snips.find_one({"client": client, "chat": chat})
|
436 |
+
|
437 |
+
if not data:
|
438 |
+
return []
|
439 |
+
|
440 |
+
return data["snips"]
|
441 |
+
|
442 |
+
async def add_pmpermit(self, client: int, user: int):
|
443 |
+
await self.pmpermit.update_one(
|
444 |
+
{"client": client, "user": user},
|
445 |
+
{"$set": {"date": self.get_datetime()}},
|
446 |
+
upsert=True,
|
447 |
+
)
|
448 |
+
|
449 |
+
async def rm_pmpermit(self, client: int, user: int):
|
450 |
+
await self.pmpermit.delete_one({"client": client, "user": user})
|
451 |
+
|
452 |
+
async def is_pmpermit(self, client: int, user: int) -> bool:
|
453 |
+
data = await self.get_pmpermit(client, user)
|
454 |
+
return True if data else False
|
455 |
+
|
456 |
+
async def get_pmpermit(self, client: int, user: int):
|
457 |
+
data = await self.pmpermit.find_one({"client": client, "user": user})
|
458 |
+
return data
|
459 |
+
|
460 |
+
async def get_all_pmpermits(self, client: int) -> list:
|
461 |
+
return [i async for i in self.pmpermit.find({"client": client})]
|
462 |
+
|
463 |
+
async def set_welcome(self, client: int, chat: int, message: int):
|
464 |
+
await self.greetings.update_one(
|
465 |
+
{"client": client, "chat": chat, "welcome": True},
|
466 |
+
{"$set": {"message": message}},
|
467 |
+
upsert=True,
|
468 |
+
)
|
469 |
+
|
470 |
+
async def rm_welcome(self, client: int, chat: int):
|
471 |
+
await self.greetings.delete_one(
|
472 |
+
{"client": client, "chat": chat, "welcome": True}
|
473 |
+
)
|
474 |
+
|
475 |
+
async def is_welcome(self, client: int, chat: int) -> bool:
|
476 |
+
data = await self.get_welcome(client, chat)
|
477 |
+
return True if data else False
|
478 |
+
|
479 |
+
async def get_welcome(self, client: int, chat: int):
|
480 |
+
data = await self.greetings.find_one(
|
481 |
+
{"client": client, "chat": chat, "welcome": True}
|
482 |
+
)
|
483 |
+
return data
|
484 |
+
|
485 |
+
async def set_goodbye(self, client: int, chat: int, message: int):
|
486 |
+
await self.greetings.update_one(
|
487 |
+
{"client": client, "chat": chat, "welcome": False},
|
488 |
+
{"$set": {"message": message}},
|
489 |
+
upsert=True,
|
490 |
+
)
|
491 |
+
|
492 |
+
async def rm_goodbye(self, client: int, chat: int):
|
493 |
+
await self.greetings.delete_one(
|
494 |
+
{"client": client, "chat": chat, "welcome": False}
|
495 |
+
)
|
496 |
+
|
497 |
+
async def is_goodbye(self, client: int, chat: int) -> bool:
|
498 |
+
data = await self.get_goodbye(client, chat)
|
499 |
+
return True if data else False
|
500 |
+
|
501 |
+
async def get_goodbye(self, client: int, chat: int):
|
502 |
+
data = await self.greetings.find_one(
|
503 |
+
{"client": client, "chat": chat, "welcome": False}
|
504 |
+
)
|
505 |
+
return data
|
506 |
+
|
507 |
+
async def get_all_greetings(self, client: int) -> list:
|
508 |
+
return [i async for i in self.greetings.find({"client": client})]
|
509 |
+
|
510 |
+
async def add_forcesub(self, chat: int, must_join: int):
|
511 |
+
await self.forcesub.update_one(
|
512 |
+
{"chat": chat},
|
513 |
+
{"$push": {"must_join": must_join}},
|
514 |
+
upsert=True,
|
515 |
+
)
|
516 |
+
|
517 |
+
async def rm_forcesub(self, chat: int, must_join: int) -> int:
|
518 |
+
await self.forcesub.update_one(
|
519 |
+
{"chat": chat},
|
520 |
+
{"$pull": {"must_join": must_join}},
|
521 |
+
)
|
522 |
+
data = await self.forcesub.find_one({"chat": chat})
|
523 |
+
return len(data["must_join"])
|
524 |
+
|
525 |
+
async def rm_all_forcesub(self, in_chat: int):
|
526 |
+
await self.forcesub.delete_one({"chat": in_chat})
|
527 |
+
|
528 |
+
async def is_forcesub(self, chat: int, must_join: int) -> bool:
|
529 |
+
data = await self.get_forcesub(chat)
|
530 |
+
if must_join in data["must_join"]:
|
531 |
+
return True
|
532 |
+
return False
|
533 |
+
|
534 |
+
async def get_forcesub(self, in_chat: int):
|
535 |
+
data = await self.forcesub.find_one({"chat": in_chat})
|
536 |
+
return data
|
537 |
+
|
538 |
+
async def get_all_forcesubs(self) -> list:
|
539 |
+
return [i async for i in self.forcesub.find({})]
|
540 |
+
|
541 |
+
async def add_gachabot(
|
542 |
+
self, client: int, bot: tuple[int, str], catch_command: str, chat_id: int
|
543 |
+
):
|
544 |
+
await self.gachabots.update_one(
|
545 |
+
{"client": client, "bot": bot[0]},
|
546 |
+
{
|
547 |
+
"$set": {
|
548 |
+
"username": bot[1],
|
549 |
+
"catch_command": catch_command,
|
550 |
+
"chat_id": chat_id,
|
551 |
+
"date": self.get_datetime(),
|
552 |
+
}
|
553 |
+
},
|
554 |
+
upsert=True,
|
555 |
+
)
|
556 |
+
|
557 |
+
async def rm_gachabot(self, client: int, bot: int, chat_id: int = None):
|
558 |
+
if chat_id:
|
559 |
+
await self.gachabots.delete_one(
|
560 |
+
{"client": client, "bot": bot, "chat_id": chat_id}
|
561 |
+
)
|
562 |
+
else:
|
563 |
+
await self.gachabots.delete_one({"client": client, "bot": bot})
|
564 |
+
|
565 |
+
async def is_gachabot(self, client: int, bot: int, chat_id: int) -> bool:
|
566 |
+
data = await self.get_gachabot(client, bot, chat_id)
|
567 |
+
return True if data else False
|
568 |
+
|
569 |
+
async def get_gachabot(self, client: int, bot: int, chat_id: int):
|
570 |
+
data = await self.gachabots.find_one(
|
571 |
+
{"client": client, "bot": bot, "chat_id": chat_id}
|
572 |
+
)
|
573 |
+
|
574 |
+
return data
|
575 |
+
|
576 |
+
async def get_all_gachabots(self, client: int) -> list:
|
577 |
+
return [i async for i in self.gachabots.find({"client": client})]
|
578 |
+
|
579 |
+
async def get_all_gachabots_id(self) -> list:
|
580 |
+
data = await self.gachabots.distinct("bot")
|
581 |
+
return data
|
582 |
+
|
583 |
+
|
584 |
+
db = Database(Config.DATABASE_URL)
|
HellBot/core/initializer.py
ADDED
@@ -0,0 +1,94 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import sys
|
2 |
+
from .clients import hellbot
|
3 |
+
from .config import Config, Symbols
|
4 |
+
from .database import db
|
5 |
+
from .logger import LOGS
|
6 |
+
|
7 |
+
|
8 |
+
async def _AuthUsers() -> None:
|
9 |
+
temp_list = []
|
10 |
+
temp_list.append(Config.OWNER_ID)
|
11 |
+
temp_list.extend([(await client.get_me()).id for client in hellbot.users])
|
12 |
+
|
13 |
+
stan_users = await db.get_all_stans()
|
14 |
+
for user in stan_users:
|
15 |
+
temp_list.append(user["user_id"])
|
16 |
+
|
17 |
+
users = list(set(temp_list))
|
18 |
+
for user in users:
|
19 |
+
Config.AUTH_USERS.add(user)
|
20 |
+
|
21 |
+
temp_list = None
|
22 |
+
LOGS.info(
|
23 |
+
f"{Symbols.arrow_right * 2} Added Authorized Users {Symbols.arrow_left * 2}"
|
24 |
+
)
|
25 |
+
|
26 |
+
|
27 |
+
async def _StanUsers() -> None:
|
28 |
+
users = await db.get_all_stans()
|
29 |
+
for user in users:
|
30 |
+
Config.STAN_USERS.add(user["user_id"])
|
31 |
+
|
32 |
+
LOGS.info(f"{Symbols.arrow_right * 2} Added Stan Users {Symbols.arrow_left * 2}")
|
33 |
+
|
34 |
+
|
35 |
+
async def _GbanUsers() -> None:
|
36 |
+
users = await db.get_gban()
|
37 |
+
for user in users:
|
38 |
+
Config.BANNED_USERS.add(user["user_id"])
|
39 |
+
|
40 |
+
LOGS.info(
|
41 |
+
f"{Symbols.arrow_right * 2} Added {len(users)} Gbanned Users {Symbols.arrow_left * 2}"
|
42 |
+
)
|
43 |
+
|
44 |
+
musers = await db.get_gmute()
|
45 |
+
for user in musers:
|
46 |
+
Config.MUTED_USERS.add(user["user_id"])
|
47 |
+
|
48 |
+
LOGS.info(
|
49 |
+
f"{Symbols.arrow_right * 2} Added {len(musers)} Gmuted Users {Symbols.arrow_left * 2}"
|
50 |
+
)
|
51 |
+
|
52 |
+
|
53 |
+
async def UserSetup() -> None:
|
54 |
+
"""Initialize Users Config"""
|
55 |
+
LOGS.info(f"{Symbols.bullet * 3} Setting Up Users {Symbols.bullet * 3}")
|
56 |
+
await _AuthUsers()
|
57 |
+
await _StanUsers()
|
58 |
+
await _GbanUsers()
|
59 |
+
|
60 |
+
|
61 |
+
async def ForcesubSetup() -> None:
|
62 |
+
"""Initialize Forcesub Config"""
|
63 |
+
chats = await db.get_all_forcesubs()
|
64 |
+
for chat in chats:
|
65 |
+
if chat not in Config.FORCESUBS:
|
66 |
+
Config.FORCESUBS.add(chat["chat"])
|
67 |
+
|
68 |
+
|
69 |
+
async def GachaBotsSetup() -> None:
|
70 |
+
"""Initialize GachaBots Config"""
|
71 |
+
bots = await db.get_all_gachabots_id()
|
72 |
+
for bot in bots:
|
73 |
+
Config.GACHA_BOTS.add(bot)
|
74 |
+
|
75 |
+
|
76 |
+
async def TemplateSetup() -> None:
|
77 |
+
"""Initialize Templates Config"""
|
78 |
+
module_name = "temp_module"
|
79 |
+
module = sys.modules.get(module_name)
|
80 |
+
if module is None:
|
81 |
+
module = type(sys)(module_name)
|
82 |
+
|
83 |
+
with open("Hellbot/functions/templates.py", "r", encoding="utf-8") as file:
|
84 |
+
exec(file.read(), module.__dict__)
|
85 |
+
|
86 |
+
global_vars = module.__dict__
|
87 |
+
|
88 |
+
var_n_value: dict[str, str] = {
|
89 |
+
var_name: global_vars[var_name][0]
|
90 |
+
for var_name in global_vars
|
91 |
+
if var_name.isupper() and not callable(global_vars[var_name])
|
92 |
+
}
|
93 |
+
|
94 |
+
Config.TEMPLATES = var_n_value
|
HellBot/core/logger.py
ADDED
@@ -0,0 +1,19 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import logging
|
2 |
+
from logging.handlers import RotatingFileHandler
|
3 |
+
|
4 |
+
logging.basicConfig(
|
5 |
+
format="[%(asctime)s]:[%(name)s]:[%(levelname)s] - %(message)s",
|
6 |
+
level=logging.INFO,
|
7 |
+
datefmt="%H:%M:%S",
|
8 |
+
handlers=[
|
9 |
+
RotatingFileHandler(
|
10 |
+
"HellBot.log", maxBytes=(1024 * 1024 * 5), backupCount=10, encoding="utf-8"
|
11 |
+
),
|
12 |
+
logging.StreamHandler(),
|
13 |
+
],
|
14 |
+
)
|
15 |
+
|
16 |
+
|
17 |
+
logging.getLogger("pyrogram").setLevel(logging.ERROR)
|
18 |
+
|
19 |
+
LOGS = logging.getLogger("HellBot")
|
HellBot/functions/__init__.py
ADDED
File without changes
|
HellBot/functions/admins.py
ADDED
@@ -0,0 +1,24 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram.enums import ChatMembersFilter, ChatMemberStatus, ChatType
|
2 |
+
from pyrogram.types import Chat
|
3 |
+
|
4 |
+
from Hellbot.core import hellbot
|
5 |
+
|
6 |
+
|
7 |
+
async def get_admins(chat_id: int) -> list:
|
8 |
+
admins = []
|
9 |
+
async for x in hellbot.bot.get_chat_members(
|
10 |
+
chat_id, filter=ChatMembersFilter.ADMINISTRATORS
|
11 |
+
):
|
12 |
+
admins.append(x.user.id)
|
13 |
+
return admins
|
14 |
+
|
15 |
+
|
16 |
+
async def is_user_admin(chat: Chat, user_id: int) -> bool:
|
17 |
+
if chat.type in [ChatType.PRIVATE, ChatType.BOT]:
|
18 |
+
return True
|
19 |
+
|
20 |
+
status = (await chat.get_member(user_id)).status
|
21 |
+
if status in [ChatMemberStatus.OWNER, ChatMemberStatus.ADMINISTRATOR]:
|
22 |
+
return True
|
23 |
+
|
24 |
+
return False
|
HellBot/functions/convert.py
ADDED
@@ -0,0 +1,98 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import time
|
3 |
+
|
4 |
+
from pyrogram.types import Message
|
5 |
+
from PIL import Image
|
6 |
+
from Hellbot.core import Config
|
7 |
+
|
8 |
+
from .tools import runcmd
|
9 |
+
|
10 |
+
|
11 |
+
async def convert_to_gif(file: str, is_video: bool = False) -> str:
|
12 |
+
resultFileName = f"gif_{round(time.time())}.mp4"
|
13 |
+
|
14 |
+
if is_video:
|
15 |
+
cmd = f"ffmpeg -i '{file}' -c copy '{resultFileName}'"
|
16 |
+
else:
|
17 |
+
cmd = f"lottie_convert.py '{file}' '{resultFileName}'"
|
18 |
+
|
19 |
+
await runcmd(cmd)
|
20 |
+
|
21 |
+
return resultFileName
|
22 |
+
|
23 |
+
|
24 |
+
async def tgs_to_png(file: str) -> str:
|
25 |
+
resultFileName = f"png_{round(time.time())}.png"
|
26 |
+
|
27 |
+
cmd = f"lottie_convert.py '{file}' '{resultFileName}'"
|
28 |
+
|
29 |
+
await runcmd(cmd)
|
30 |
+
|
31 |
+
return resultFileName
|
32 |
+
|
33 |
+
|
34 |
+
async def image_to_sticker(file: str, max_size: tuple = (512, 512)) -> tuple[bool, str]:
|
35 |
+
try:
|
36 |
+
with Image.open(file) as img:
|
37 |
+
original_width, original_height = img.size
|
38 |
+
|
39 |
+
new_width = min(original_width, max_size[0])
|
40 |
+
new_height = min(original_height, max_size[1])
|
41 |
+
|
42 |
+
if original_width > max_size[0] or original_height > max_size[1]:
|
43 |
+
img = img.resize((new_width, new_height), Image.LANCZOS)
|
44 |
+
|
45 |
+
file_name = f"sticker_{int(time.time())}.png"
|
46 |
+
img.save(file_name, "PNG")
|
47 |
+
|
48 |
+
return True, file_name
|
49 |
+
|
50 |
+
except Exception as e:
|
51 |
+
return False, str(e)
|
52 |
+
|
53 |
+
|
54 |
+
async def video_to_png(
|
55 |
+
file: str, duration: float, output: str = None
|
56 |
+
) -> tuple[str, bool]:
|
57 |
+
resultFileName = output or f"{os.path.basename(file)}.png"
|
58 |
+
cut_at = duration // 2
|
59 |
+
|
60 |
+
cmd = f"ffmpeg -ss {cut_at} -i '{file}' -vframes 1 '{resultFileName}'"
|
61 |
+
|
62 |
+
_, err, _, _ = await runcmd(cmd)
|
63 |
+
if err:
|
64 |
+
return err, False
|
65 |
+
|
66 |
+
return resultFileName, True
|
67 |
+
|
68 |
+
|
69 |
+
async def video_to_sticker(file: Message) -> tuple[str, bool]:
|
70 |
+
try:
|
71 |
+
if file.animation:
|
72 |
+
width, height = file.animation.width, file.animation.height
|
73 |
+
elif file.video:
|
74 |
+
width, height = file.video.width, file.video.height
|
75 |
+
else:
|
76 |
+
return "Unsupported media type.", False
|
77 |
+
|
78 |
+
file_path = await file.download(Config.TEMP_DIR)
|
79 |
+
output_path = os.path.join(Config.TEMP_DIR, "videoSticker.webm")
|
80 |
+
|
81 |
+
if height > width:
|
82 |
+
scale_params = f"scale=-1:512"
|
83 |
+
else:
|
84 |
+
scale_params = f"scale=512:-1"
|
85 |
+
|
86 |
+
cmd = (
|
87 |
+
f"ffmpeg -i {file_path} "
|
88 |
+
f"-vf fps=30,{scale_params} -t 3 -c:v libvpx-vp9 -b:v 256k -an -pix_fmt yuv420p -auto-alt-ref 0 -loop 0 "
|
89 |
+
f"-f webm {output_path}"
|
90 |
+
)
|
91 |
+
|
92 |
+
await runcmd(cmd)
|
93 |
+
os.remove(file_path)
|
94 |
+
|
95 |
+
return output_path, True
|
96 |
+
|
97 |
+
except Exception as e:
|
98 |
+
return f"Error during conversion: {e}", False
|
HellBot/functions/driver.py
ADDED
@@ -0,0 +1,318 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import datetime
|
2 |
+
import json
|
3 |
+
import random
|
4 |
+
import re
|
5 |
+
import time
|
6 |
+
import urllib.parse
|
7 |
+
from urllib.parse import quote_plus
|
8 |
+
|
9 |
+
import httpx
|
10 |
+
import requests
|
11 |
+
from pytz import country_names, country_timezones, timezone
|
12 |
+
from selenium import webdriver
|
13 |
+
from selenium.webdriver.chrome.options import Options
|
14 |
+
from selenium.webdriver.chrome.service import Service
|
15 |
+
from selenium.webdriver.common.by import By
|
16 |
+
|
17 |
+
from Hellbot.core import ENV, Config, db
|
18 |
+
|
19 |
+
from .formatter import format_text
|
20 |
+
|
21 |
+
|
22 |
+
class ChromeDriver:
|
23 |
+
def __init__(self) -> None:
|
24 |
+
self.carbon_theme = [
|
25 |
+
"3024-night",
|
26 |
+
"a11y-dark",
|
27 |
+
"blackboard",
|
28 |
+
"base16-dark",
|
29 |
+
"base16-light",
|
30 |
+
"cobalt",
|
31 |
+
"duotone-dark",
|
32 |
+
"hopscotch",
|
33 |
+
"lucario",
|
34 |
+
"material",
|
35 |
+
"monokai",
|
36 |
+
"night-owl",
|
37 |
+
"nord",
|
38 |
+
"oceanic-next",
|
39 |
+
"one-light",
|
40 |
+
"one-dark",
|
41 |
+
"panda-syntax",
|
42 |
+
"paraiso-dark",
|
43 |
+
"seti",
|
44 |
+
"shades-of-purple",
|
45 |
+
"solarized+dark",
|
46 |
+
"solarized+light",
|
47 |
+
"synthwave-84",
|
48 |
+
"twilight",
|
49 |
+
"verminal",
|
50 |
+
"vscode",
|
51 |
+
"yeti",
|
52 |
+
"zenburn",
|
53 |
+
]
|
54 |
+
|
55 |
+
def get(self):
|
56 |
+
if not Config.CHROME_BIN:
|
57 |
+
return (
|
58 |
+
None,
|
59 |
+
"ChromeBinaryErr: No binary path found! Install Chromium or Google Chrome.",
|
60 |
+
)
|
61 |
+
|
62 |
+
try:
|
63 |
+
options = Options()
|
64 |
+
options.binary_location = Config.CHROME_BIN
|
65 |
+
options.add_argument("--disable-dev-shm-usage")
|
66 |
+
options.add_argument("--ignore-certificate-errors")
|
67 |
+
options.add_argument("--disable-gpu")
|
68 |
+
options.add_argument("--headless=new")
|
69 |
+
options.add_argument("--test-type")
|
70 |
+
options.add_argument("--no-sandbox")
|
71 |
+
options.add_argument("--window-size=1920x1080")
|
72 |
+
options.add_experimental_option(
|
73 |
+
"prefs", {"download.default_directory": "./"}
|
74 |
+
)
|
75 |
+
service = Service(Config.CHROME_DRIVER)
|
76 |
+
driver = webdriver.Chrome(options, service)
|
77 |
+
return driver, None
|
78 |
+
except Exception as e:
|
79 |
+
return None, f"ChromeDriverErr: {e}"
|
80 |
+
|
81 |
+
def close(self, driver: webdriver.Chrome):
|
82 |
+
driver.close()
|
83 |
+
driver.quit()
|
84 |
+
|
85 |
+
@property
|
86 |
+
def get_random_carbon(self) -> str:
|
87 |
+
url = "https://carbon.now.sh/?l=auto"
|
88 |
+
url += f"&t={random.choice(self.carbon_theme)}"
|
89 |
+
url += f"&bg=rgba%28{random.randint(1, 255)}%2C{random.randint(1, 255)}%2C{random.randint(1, 255)}%2C1%29"
|
90 |
+
url += "&code="
|
91 |
+
return url
|
92 |
+
|
93 |
+
async def generate_carbon(
|
94 |
+
self, driver: webdriver.Chrome, code: str, is_random: bool = False
|
95 |
+
) -> str:
|
96 |
+
filename = f"{round(time.time())}"
|
97 |
+
BASE_URL = (
|
98 |
+
self.get_random_carbon
|
99 |
+
if is_random
|
100 |
+
else "https://carbon.now.sh/?l=auto&code="
|
101 |
+
)
|
102 |
+
|
103 |
+
driver.get(BASE_URL + format_text(quote_plus(code)))
|
104 |
+
driver.command_executor._commands["send_command"] = (
|
105 |
+
"POST",
|
106 |
+
"/session/$sessionId/chromium/send_command",
|
107 |
+
)
|
108 |
+
params = {
|
109 |
+
"cmd": "Page.setDownloadBehavior",
|
110 |
+
"params": {"behavior": "allow", "downloadPath": Config.DWL_DIR},
|
111 |
+
}
|
112 |
+
driver.execute("send_command", params)
|
113 |
+
|
114 |
+
driver.find_element(By.XPATH, "//button[@id='export-menu']").click()
|
115 |
+
driver.find_element(By.XPATH, "//input[@title='filename']").send_keys(filename)
|
116 |
+
driver.find_element(By.XPATH, "//button[@id='export-png']").click()
|
117 |
+
|
118 |
+
return f"{Config.DWL_DIR}/{filename}.png"
|
119 |
+
|
120 |
+
|
121 |
+
class ClimateDriver:
|
122 |
+
def __init__(self) -> None:
|
123 |
+
self.weather_api = "https://api.openweathermap.org/data/2.5/weather?lat={0}&lon={1}&appid={2}&units=metric"
|
124 |
+
self.location_api = (
|
125 |
+
"https://api.openweathermap.org/geo/1.0/direct?q={0}&limit=1&appid={1}"
|
126 |
+
)
|
127 |
+
self.pollution_api = "http://api.openweathermap.org/data/2.5/air_pollution?lat={0}&lon={1}&appid={2}"
|
128 |
+
self.AQI_DICT = {
|
129 |
+
1: "Good",
|
130 |
+
2: "Fair",
|
131 |
+
3: "Moderate",
|
132 |
+
4: "Poor",
|
133 |
+
5: "Very Poor",
|
134 |
+
}
|
135 |
+
|
136 |
+
async def fetchLocation(self, city: str, apiKey: str):
|
137 |
+
response = httpx.get(self.location_api.format(city, apiKey))
|
138 |
+
if response.status_code == 200:
|
139 |
+
data = response.json()
|
140 |
+
if data:
|
141 |
+
return data[0]["lat"], data[0]["lon"]
|
142 |
+
return None, None
|
143 |
+
|
144 |
+
async def fetchWeather(self, city: str, apiKey: str):
|
145 |
+
lattitude, longitude = await self.fetchLocation(city, apiKey)
|
146 |
+
if not lattitude and not longitude:
|
147 |
+
return None
|
148 |
+
|
149 |
+
response = httpx.get(self.weather_api.format(lattitude, longitude, apiKey))
|
150 |
+
if response.status_code == 200:
|
151 |
+
return response.json()
|
152 |
+
return None
|
153 |
+
|
154 |
+
async def fetchAirPollution(self, city: str, apiKey: str):
|
155 |
+
lattitude, longitude = await self.fetchLocation(city, apiKey)
|
156 |
+
if not lattitude and not longitude:
|
157 |
+
return None
|
158 |
+
|
159 |
+
response = httpx.get(self.pollution_api.format(lattitude, longitude, apiKey))
|
160 |
+
if response.status_code == 200:
|
161 |
+
return response.json()
|
162 |
+
return None
|
163 |
+
|
164 |
+
async def getTime(self, timestamp: int) -> str:
|
165 |
+
tz = await db.get_env(ENV.time_zone) or "Asia/Kolkata"
|
166 |
+
tz = timezone(tz)
|
167 |
+
return datetime.datetime.fromtimestamp(timestamp, tz=tz).strftime("%I:%M %p")
|
168 |
+
|
169 |
+
def getCountry(self, country_code: str) -> str:
|
170 |
+
return country_names.get(country_code, "Unknown")
|
171 |
+
|
172 |
+
def getCountryTimezone(self, country_code: str) -> str:
|
173 |
+
timezones = country_timezones.get(country_code, [])
|
174 |
+
if timezones:
|
175 |
+
return ", ".join(timezones)
|
176 |
+
return "Unknown"
|
177 |
+
|
178 |
+
def getWindData(self, windSpeed: str, windDegree: str) -> str:
|
179 |
+
dirs = ["N", "NE", "E", "SE", "S", "SW", "W", "NW"]
|
180 |
+
ix = round(windDegree / (360.00 / len(dirs)))
|
181 |
+
kmph = str(float(windSpeed) * 3.6) + " km/h"
|
182 |
+
return f"[{dirs[ix % len(dirs)]}] {kmph}"
|
183 |
+
|
184 |
+
|
185 |
+
class YoutubeDriver:
|
186 |
+
def __init__(self, search_terms: str, max_results: int = 5):
|
187 |
+
self.base_url = "https://youtube.com/results?search_query={0}"
|
188 |
+
self.search_terms = search_terms
|
189 |
+
self.max_results = max_results
|
190 |
+
self.videos = self._search()
|
191 |
+
|
192 |
+
def _search(self):
|
193 |
+
encoded_search = urllib.parse.quote_plus(self.search_terms)
|
194 |
+
response = requests.get(self.base_url.format(encoded_search)).text
|
195 |
+
|
196 |
+
while "ytInitialData" not in response:
|
197 |
+
response = requests.get(self.base_url.format(encoded_search)).text
|
198 |
+
|
199 |
+
results = self._parse_html(response)
|
200 |
+
|
201 |
+
if self.max_results is not None and len(results) > self.max_results:
|
202 |
+
return results[: self.max_results]
|
203 |
+
|
204 |
+
return results
|
205 |
+
|
206 |
+
def _parse_html(self, response: str):
|
207 |
+
results = []
|
208 |
+
start = response.index("ytInitialData") + len("ytInitialData") + 3
|
209 |
+
end = response.index("};", start) + 1
|
210 |
+
json_str = response[start:end]
|
211 |
+
data = json.loads(json_str)
|
212 |
+
|
213 |
+
videos = data["contents"]["twoColumnSearchResultsRenderer"]["primaryContents"][
|
214 |
+
"sectionListRenderer"
|
215 |
+
]["contents"][0]["itemSectionRenderer"]["contents"]
|
216 |
+
|
217 |
+
for video in videos:
|
218 |
+
res = {}
|
219 |
+
if "videoRenderer" in video.keys():
|
220 |
+
video_data = video.get("videoRenderer", {})
|
221 |
+
_id = video_data.get("videoId", None)
|
222 |
+
|
223 |
+
res["id"] = _id
|
224 |
+
res["thumbnail"] = f"https://i.ytimg.com/vi/{_id}/hqdefault.jpg"
|
225 |
+
res["title"] = (
|
226 |
+
video_data.get("title", {}).get("runs", [[{}]])[0].get("text", None)
|
227 |
+
)
|
228 |
+
res["channel"] = (
|
229 |
+
video_data.get("longBylineText", {})
|
230 |
+
.get("runs", [[{}]])[0]
|
231 |
+
.get("text", None)
|
232 |
+
)
|
233 |
+
res["duration"] = video_data.get("lengthText", {}).get("simpleText", 0)
|
234 |
+
res["views"] = video_data.get("viewCountText", {}).get(
|
235 |
+
"simpleText", "Unknown"
|
236 |
+
)
|
237 |
+
res["publish_time"] = video_data.get("publishedTimeText", {}).get(
|
238 |
+
"simpleText", "Unknown"
|
239 |
+
)
|
240 |
+
res["url_suffix"] = (
|
241 |
+
video_data.get("navigationEndpoint", {})
|
242 |
+
.get("commandMetadata", {})
|
243 |
+
.get("webCommandMetadata", {})
|
244 |
+
.get("url", None)
|
245 |
+
)
|
246 |
+
|
247 |
+
results.append(res)
|
248 |
+
return results
|
249 |
+
|
250 |
+
def to_dict(self, clear_cache=True) -> list[dict]:
|
251 |
+
result = self.videos
|
252 |
+
if clear_cache:
|
253 |
+
self.videos = []
|
254 |
+
return result
|
255 |
+
|
256 |
+
@staticmethod
|
257 |
+
def check_url(url: str) -> tuple[bool, str]:
|
258 |
+
if "&" in url:
|
259 |
+
url = url[: url.index("&")]
|
260 |
+
|
261 |
+
if "?si=" in url:
|
262 |
+
url = url[: url.index("?si=")]
|
263 |
+
|
264 |
+
youtube_regex = (
|
265 |
+
r"(https?://)?(www\.)?"
|
266 |
+
r"(youtube|youtu|youtube-nocookie)\.(com|be)/"
|
267 |
+
r'(video|embed|shorts/|watch\?v=|v/|e/|u/\\w+/|\\w+/)?([^"&?\\s]{11})'
|
268 |
+
)
|
269 |
+
match = re.match(youtube_regex, url)
|
270 |
+
if match:
|
271 |
+
return True, match.group(6)
|
272 |
+
else:
|
273 |
+
return False, "Invalid YouTube URL!"
|
274 |
+
|
275 |
+
@staticmethod
|
276 |
+
def song_options() -> dict:
|
277 |
+
return {
|
278 |
+
"format": "bestaudio",
|
279 |
+
"addmetadata": True,
|
280 |
+
"key": "FFmpegMetadata",
|
281 |
+
"prefer_ffmpeg": True,
|
282 |
+
"geo_bypass": True,
|
283 |
+
"nocheckcertificate": True,
|
284 |
+
"postprocessors": [
|
285 |
+
{
|
286 |
+
"key": "FFmpegExtractAudio",
|
287 |
+
"preferredcodec": "mp3",
|
288 |
+
"preferredquality": "480",
|
289 |
+
}
|
290 |
+
],
|
291 |
+
"outtmpl": "%(id)s",
|
292 |
+
"quiet": True,
|
293 |
+
"logtostderr": False,
|
294 |
+
}
|
295 |
+
|
296 |
+
@staticmethod
|
297 |
+
def video_options() -> dict:
|
298 |
+
return {
|
299 |
+
"format": "best",
|
300 |
+
"addmetadata": True,
|
301 |
+
"key": "FFmpegMetadata",
|
302 |
+
"prefer_ffmpeg": True,
|
303 |
+
"geo_bypass": True,
|
304 |
+
"nocheckcertificate": True,
|
305 |
+
"postprocessors": [
|
306 |
+
{
|
307 |
+
"key": "FFmpegVideoConvertor",
|
308 |
+
"preferedformat": "mp4",
|
309 |
+
}
|
310 |
+
],
|
311 |
+
"outtmpl": "%(id)s.mp4",
|
312 |
+
"quiet": True,
|
313 |
+
"logtostderr": False,
|
314 |
+
}
|
315 |
+
|
316 |
+
|
317 |
+
Driver = ChromeDriver()
|
318 |
+
Climate = ClimateDriver()
|
HellBot/functions/formatter.py
ADDED
@@ -0,0 +1,94 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import math
|
2 |
+
import re
|
3 |
+
|
4 |
+
|
5 |
+
def format_text(text: str) -> str:
|
6 |
+
emoji_pattern = re.compile(
|
7 |
+
"["
|
8 |
+
"\U0001F600-\U0001F64F" # emoticons
|
9 |
+
"\U0001F300-\U0001F5FF" # symbols & pictographs
|
10 |
+
"\U0001F680-\U0001F6FF" # transport & map symbols
|
11 |
+
"\U0001F700-\U0001F77F" # alchemical symbols
|
12 |
+
"\U0001F780-\U0001F7FF" # Geometric Shapes Extended
|
13 |
+
"\U0001F800-\U0001F8FF" # Supplemental Arrows-C
|
14 |
+
"\U0001F900-\U0001F9FF" # Supplemental Symbols and Pictographs
|
15 |
+
"\U0001FA00-\U0001FA6F" # Chess Symbols
|
16 |
+
"\U0001FA70-\U0001FAFF" # Symbols and Pictographs Extended-A
|
17 |
+
"\U00002702-\U000027B0" # Dingbats
|
18 |
+
"\U000024C2-\U0001F251" # enclosed characters
|
19 |
+
"]+",
|
20 |
+
flags=re.UNICODE,
|
21 |
+
)
|
22 |
+
|
23 |
+
return re.sub(emoji_pattern, "", text)
|
24 |
+
|
25 |
+
|
26 |
+
def superscript(text: str) -> str:
|
27 |
+
superscript_digits = str.maketrans("0123456789", "⁰¹²³⁴⁵⁶⁷⁸⁹")
|
28 |
+
return text.translate(superscript_digits)
|
29 |
+
|
30 |
+
|
31 |
+
def subscript(text: str) -> str:
|
32 |
+
subscript_digits = str.maketrans("0123456789", "₀₁₂₃₄₅₆₇₈₉")
|
33 |
+
return text.translate(subscript_digits)
|
34 |
+
|
35 |
+
|
36 |
+
def readable_time(seconds: int) -> str:
|
37 |
+
count = 0
|
38 |
+
out_time = ""
|
39 |
+
time_list = []
|
40 |
+
time_suffix_list = ["secs", "mins", "hrs", "days"]
|
41 |
+
|
42 |
+
while count < 4:
|
43 |
+
count += 1
|
44 |
+
remainder, result = divmod(seconds, 60) if count < 3 else divmod(seconds, 24)
|
45 |
+
if seconds == 0 and remainder == 0:
|
46 |
+
break
|
47 |
+
time_list.append(int(result))
|
48 |
+
seconds = int(remainder)
|
49 |
+
|
50 |
+
for x in range(len(time_list)):
|
51 |
+
time_list[x] = str(time_list[x]) + time_suffix_list[x]
|
52 |
+
|
53 |
+
if len(time_list) == 4:
|
54 |
+
out_time += time_list.pop() + ", "
|
55 |
+
|
56 |
+
time_list.reverse()
|
57 |
+
out_time += " ".join(time_list)
|
58 |
+
|
59 |
+
return out_time or "0 secs"
|
60 |
+
|
61 |
+
|
62 |
+
def humanbytes(size: int):
|
63 |
+
if not size:
|
64 |
+
return ""
|
65 |
+
power = 2**10
|
66 |
+
number = 0
|
67 |
+
dict_power_n = {0: " ", 1: "Ki", 2: "Mi", 3: "Gi", 4: "Ti"}
|
68 |
+
while size > power:
|
69 |
+
size /= power
|
70 |
+
number += 1
|
71 |
+
return str(round(size, 2)) + " " + dict_power_n[number] + "B"
|
72 |
+
|
73 |
+
|
74 |
+
def add_to_dict(data: dict, keys: list, value: str | int | bool = None) -> None:
|
75 |
+
current_level = data
|
76 |
+
for key in keys[:-1]:
|
77 |
+
current_level = current_level.setdefault(key, {})
|
78 |
+
current_level[keys[-1]] = value
|
79 |
+
|
80 |
+
|
81 |
+
def get_from_dict(data: dict, key: list):
|
82 |
+
current_level = data
|
83 |
+
for k in key:
|
84 |
+
current_level = current_level[k]
|
85 |
+
return current_level
|
86 |
+
|
87 |
+
|
88 |
+
def limit_per_page(limit: int) -> int:
|
89 |
+
return math.ceil(limit / 10)
|
90 |
+
|
91 |
+
|
92 |
+
def secs_to_mins(secs: int) -> str:
|
93 |
+
mins, secs = divmod(secs, 60)
|
94 |
+
return f"{mins}:{secs}"
|
HellBot/functions/images.py
ADDED
@@ -0,0 +1,389 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import calendar
|
2 |
+
import logging
|
3 |
+
import os
|
4 |
+
import random
|
5 |
+
import textwrap
|
6 |
+
import time
|
7 |
+
|
8 |
+
import httpx
|
9 |
+
from icrawler.builtin import BingImageCrawler
|
10 |
+
from PIL import Image, ImageDraw, ImageEnhance, ImageFont, ImageOps
|
11 |
+
from unidecode import unidecode
|
12 |
+
|
13 |
+
from .formatter import format_text, limit_per_page
|
14 |
+
|
15 |
+
|
16 |
+
def convert_to_png(image: str) -> str:
|
17 |
+
output_img = f"png_{round(time.time())}.png"
|
18 |
+
|
19 |
+
img = Image.open(image)
|
20 |
+
img.save(output_img, "PNG")
|
21 |
+
img.close()
|
22 |
+
|
23 |
+
os.remove(image)
|
24 |
+
return output_img
|
25 |
+
|
26 |
+
|
27 |
+
def add_rounded_corners(img: Image.Image, radius: int = 80):
|
28 |
+
circle = Image.new("L", (radius * 2, radius * 2), 0)
|
29 |
+
|
30 |
+
draw = ImageDraw.Draw(circle)
|
31 |
+
draw.ellipse((0, 0, radius * 2, radius * 2), fill=255)
|
32 |
+
|
33 |
+
alpha = Image.new("L", img.size, 255)
|
34 |
+
w, h = img.size
|
35 |
+
|
36 |
+
alpha.paste(circle.crop((0, 0, radius, radius)), (0, 0))
|
37 |
+
alpha.paste(circle.crop((radius, 0, radius * 2, radius)), (w - radius, 0))
|
38 |
+
alpha.paste(circle.crop((0, radius, radius, radius * 2)), (0, h - radius))
|
39 |
+
alpha.paste(
|
40 |
+
circle.crop((radius, radius, radius * 2, radius * 2)), (w - radius, h - radius)
|
41 |
+
)
|
42 |
+
|
43 |
+
img.putalpha(alpha)
|
44 |
+
|
45 |
+
return img
|
46 |
+
|
47 |
+
|
48 |
+
def generate_alive_image(
|
49 |
+
username: str, profile_pic: str, del_img: bool, font_path: str
|
50 |
+
) -> str:
|
51 |
+
if not profile_pic.endswith(".png"):
|
52 |
+
profile_pic = convert_to_png(profile_pic)
|
53 |
+
|
54 |
+
img = Image.open(profile_pic).convert("RGBA")
|
55 |
+
img_rotated = img.rotate(45, expand=True)
|
56 |
+
|
57 |
+
width, height = img_rotated.size
|
58 |
+
left = width / 2 - 480 / 2
|
59 |
+
top = height / 2 - 480 / 2
|
60 |
+
right = width / 2 + 480 / 2
|
61 |
+
bottom = height / 2 + 480 / 2
|
62 |
+
|
63 |
+
cropped_img = img_rotated.crop((left, top, right, bottom))
|
64 |
+
|
65 |
+
img_rotated = ImageOps.fit(
|
66 |
+
cropped_img, (480, 480), method=0, bleed=0.0, centering=(0.5, 0.5)
|
67 |
+
)
|
68 |
+
|
69 |
+
img_rounded = add_rounded_corners(img_rotated)
|
70 |
+
|
71 |
+
img = img_rounded.rotate(-45, expand=True)
|
72 |
+
|
73 |
+
background = Image.open("./Hellbot/resources/images/hellbot_alive.png").convert(
|
74 |
+
"RGBA"
|
75 |
+
)
|
76 |
+
|
77 |
+
background.paste(img, (383, 445), img)
|
78 |
+
draw = ImageDraw.Draw(background)
|
79 |
+
|
80 |
+
text = format_text(username[:25] + ("..." if len(username) > 25 else ""))
|
81 |
+
|
82 |
+
font_size = width // 15
|
83 |
+
font = ImageFont.truetype(font_path, font_size, encoding="utf-8")
|
84 |
+
|
85 |
+
text_length = draw.textlength(text, font)
|
86 |
+
position = ((background.width - text_length) / 2, background.height - 145)
|
87 |
+
draw.text(
|
88 |
+
position,
|
89 |
+
unidecode(text),
|
90 |
+
(255, 255, 255),
|
91 |
+
font,
|
92 |
+
)
|
93 |
+
|
94 |
+
output_img = f"alive_{int(time.time())}.png"
|
95 |
+
background.save(output_img, "PNG")
|
96 |
+
background.close()
|
97 |
+
|
98 |
+
if del_img:
|
99 |
+
os.remove(profile_pic)
|
100 |
+
|
101 |
+
return output_img
|
102 |
+
|
103 |
+
|
104 |
+
async def get_wallpapers(
|
105 |
+
access: str,
|
106 |
+
limit: int,
|
107 |
+
query: str = "",
|
108 |
+
isRandom: bool = False,
|
109 |
+
) -> list[str]:
|
110 |
+
headers = {"Authorization": f"Client-ID {access}"}
|
111 |
+
|
112 |
+
if isRandom:
|
113 |
+
api = f"https://api.unsplash.com/photos/random?count={limit}"
|
114 |
+
response = httpx.get(api, headers=headers)
|
115 |
+
results = response.json()
|
116 |
+
urls = [i["urls"]["raw"] for i in results]
|
117 |
+
else:
|
118 |
+
api = f"https://api.unsplash.com/search/photos?query={query}&page={limit_per_page(limit)}"
|
119 |
+
response = httpx.get(api, headers=headers)
|
120 |
+
result = response.json()
|
121 |
+
urls = [i["urls"]["raw"] for i in result["results"]]
|
122 |
+
|
123 |
+
random.shuffle(urls)
|
124 |
+
|
125 |
+
return urls[:limit]
|
126 |
+
|
127 |
+
|
128 |
+
async def deep_fry(img: Image.Image) -> Image.Image:
|
129 |
+
colours = (
|
130 |
+
(random.randint(50, 200), random.randint(40, 170), random.randint(40, 190)),
|
131 |
+
(random.randint(190, 255), random.randint(170, 240), random.randint(180, 250)),
|
132 |
+
)
|
133 |
+
|
134 |
+
img = img.copy().convert("RGB")
|
135 |
+
img = img.convert("RGB")
|
136 |
+
|
137 |
+
width, height = img.width, img.height
|
138 |
+
|
139 |
+
img = img.resize(
|
140 |
+
(
|
141 |
+
int(width ** random.uniform(0.8, 0.9)),
|
142 |
+
int(height ** random.uniform(0.8, 0.9)),
|
143 |
+
),
|
144 |
+
resample=Image.LANCZOS,
|
145 |
+
)
|
146 |
+
|
147 |
+
img = img.resize(
|
148 |
+
(
|
149 |
+
int(width ** random.uniform(0.85, 0.95)),
|
150 |
+
int(height ** random.uniform(0.85, 0.95)),
|
151 |
+
),
|
152 |
+
resample=Image.BILINEAR,
|
153 |
+
)
|
154 |
+
|
155 |
+
img = img.resize(
|
156 |
+
(
|
157 |
+
int(width ** random.uniform(0.89, 0.98)),
|
158 |
+
int(height ** random.uniform(0.89, 0.98)),
|
159 |
+
),
|
160 |
+
resample=Image.BICUBIC,
|
161 |
+
)
|
162 |
+
|
163 |
+
img = img.resize((width, height), resample=Image.BICUBIC)
|
164 |
+
img = ImageOps.posterize(img, random.randint(3, 7))
|
165 |
+
|
166 |
+
overlay = img.split()[0]
|
167 |
+
overlay = ImageEnhance.Contrast(overlay).enhance(random.uniform(1.0, 2.0))
|
168 |
+
overlay = ImageEnhance.Brightness(overlay).enhance(random.uniform(1.0, 2.0))
|
169 |
+
overlay = ImageOps.colorize(overlay, colours[0], colours[1])
|
170 |
+
|
171 |
+
img = Image.blend(img, overlay, random.uniform(0.1, 0.4))
|
172 |
+
img = ImageEnhance.Sharpness(img).enhance(random.randint(5, 300))
|
173 |
+
|
174 |
+
return img
|
175 |
+
|
176 |
+
|
177 |
+
async def make_logo(background: str, text: str, font_path: str) -> str:
|
178 |
+
if not background.endswith(".png"):
|
179 |
+
background = convert_to_png(background)
|
180 |
+
|
181 |
+
bg = Image.open(background).convert("RGBA")
|
182 |
+
bgWidth, bgHeight = bg.size
|
183 |
+
|
184 |
+
text = format_text(text)
|
185 |
+
font_size = bgWidth // len(text)
|
186 |
+
font = ImageFont.truetype(font_path, font_size, encoding="utf-8")
|
187 |
+
|
188 |
+
draw = ImageDraw.Draw(bg)
|
189 |
+
text_length = draw.textlength(text, font)
|
190 |
+
|
191 |
+
x = (bgWidth - text_length) // 2
|
192 |
+
y = (bgHeight - font_size) // 2
|
193 |
+
|
194 |
+
draw.text(
|
195 |
+
(x, y),
|
196 |
+
unidecode(text),
|
197 |
+
(255, 255, 255),
|
198 |
+
font,
|
199 |
+
stroke_fill=(0, 0, 0),
|
200 |
+
stroke_width=2,
|
201 |
+
)
|
202 |
+
|
203 |
+
output_img = f"logo_{int(time.time())}.png"
|
204 |
+
bg.save(output_img, "PNG")
|
205 |
+
bg.close()
|
206 |
+
|
207 |
+
os.remove(background)
|
208 |
+
|
209 |
+
return output_img
|
210 |
+
|
211 |
+
|
212 |
+
async def draw_meme(
|
213 |
+
image_path: str, upper_text: str = "", lower_text: str = ""
|
214 |
+
) -> list[str]:
|
215 |
+
image = Image.open(image_path)
|
216 |
+
width, height = image.size
|
217 |
+
|
218 |
+
draw = ImageDraw.Draw(image)
|
219 |
+
font_size = int((30 / 500) * width)
|
220 |
+
font = ImageFont.truetype("./Hellbot/resources/fonts/Montserrat.ttf", font_size)
|
221 |
+
|
222 |
+
curr_height, padding = 20, 5
|
223 |
+
for utext in textwrap.wrap(upper_text, 25):
|
224 |
+
upper_width = draw.textlength(utext, font=font)
|
225 |
+
draw.text(
|
226 |
+
((width - upper_width) / 2, curr_height),
|
227 |
+
unidecode(utext),
|
228 |
+
(255, 255, 255),
|
229 |
+
font,
|
230 |
+
stroke_width=3,
|
231 |
+
stroke_fill=(0, 0, 0),
|
232 |
+
)
|
233 |
+
curr_height += font_size + padding
|
234 |
+
|
235 |
+
curr_height = height - font_size
|
236 |
+
for ltext in reversed(textwrap.wrap(lower_text, 25)):
|
237 |
+
lower_width = draw.textlength(ltext, font=font)
|
238 |
+
draw.text(
|
239 |
+
((width - lower_width) / 2, curr_height - font_size),
|
240 |
+
ltext,
|
241 |
+
(255, 255, 255),
|
242 |
+
font,
|
243 |
+
stroke_width=3,
|
244 |
+
stroke_fill=(0, 0, 0),
|
245 |
+
)
|
246 |
+
curr_height -= font_size + padding
|
247 |
+
|
248 |
+
filename = f"meme_{int(time.time())}"
|
249 |
+
image.save(f"{filename}.png", "PNG", optimize=True)
|
250 |
+
image.save(f"{filename}.webp", "WEBP", optimize=True)
|
251 |
+
image.close()
|
252 |
+
|
253 |
+
return [f"{filename}.png", f"{filename}.webp"]
|
254 |
+
|
255 |
+
|
256 |
+
async def remove_bg(api_key: str, image: str) -> str:
|
257 |
+
response = httpx.post(
|
258 |
+
"https://api.remove.bg/v1.0/removebg",
|
259 |
+
files={"image_file": open(image, "rb")},
|
260 |
+
data={"size": "auto"},
|
261 |
+
headers={"X-Api-Key": api_key},
|
262 |
+
)
|
263 |
+
filename = f"removedbg_{int(time.time())}.png"
|
264 |
+
|
265 |
+
if response.is_success:
|
266 |
+
with open(filename, "wb") as f:
|
267 |
+
f.write(response.content)
|
268 |
+
else:
|
269 |
+
raise Exception(
|
270 |
+
f"RemoveBGError: [{response.status_code}] {response.content.decode('utf-8')}"
|
271 |
+
)
|
272 |
+
|
273 |
+
return filename
|
274 |
+
|
275 |
+
|
276 |
+
def create_gradient(
|
277 |
+
size: tuple[int, int],
|
278 |
+
color_start: tuple[int, int, int],
|
279 |
+
color_end: tuple[int, int, int],
|
280 |
+
) -> Image.Image:
|
281 |
+
gradient = Image.new("RGB", (size))
|
282 |
+
draw = ImageDraw.Draw(gradient)
|
283 |
+
|
284 |
+
for x in range(size[0]):
|
285 |
+
r = int(color_start[0] + (color_end[0] - color_start[0]) * (x / size[0]))
|
286 |
+
g = int(color_start[1] + (color_end[1] - color_start[1]) * (x / size[0]))
|
287 |
+
b = int(color_start[2] + (color_end[2] - color_start[2]) * (x / size[0]))
|
288 |
+
|
289 |
+
draw.line([(x, 0), (x, size[1])], fill=(r, g, b))
|
290 |
+
|
291 |
+
return gradient
|
292 |
+
|
293 |
+
|
294 |
+
async def create_calendar(year: int, month: int) -> str:
|
295 |
+
cal = calendar.monthcalendar(year, month)
|
296 |
+
month_name = calendar.month_name[month]
|
297 |
+
|
298 |
+
calendar_image = create_gradient((500, 500), (140, 200, 250), (0, 150, 200))
|
299 |
+
draw = ImageDraw.Draw(calendar_image)
|
300 |
+
|
301 |
+
month_font = ImageFont.truetype("./Hellbot/resources/fonts/Montserrat.ttf", 40)
|
302 |
+
month_x = (
|
303 |
+
calendar_image.width - draw.textlength(f"{month_name} {year}", month_font)
|
304 |
+
) // 2
|
305 |
+
month_y = 30
|
306 |
+
draw.text(
|
307 |
+
(month_x, month_y),
|
308 |
+
f"{month_name} {year}",
|
309 |
+
(43, 255, 136),
|
310 |
+
month_font,
|
311 |
+
stroke_width=2,
|
312 |
+
stroke_fill=(255, 40, 40),
|
313 |
+
)
|
314 |
+
|
315 |
+
week_font = ImageFont.truetype("./Hellbot/resources/fonts/Montserrat.ttf", 23)
|
316 |
+
weekdays_text = " ".join([day[:3] for day in calendar.day_name])
|
317 |
+
textsize = draw.textlength(weekdays_text, week_font)
|
318 |
+
draw.text(
|
319 |
+
((calendar_image.width - textsize) // 2, month_y + 80),
|
320 |
+
weekdays_text,
|
321 |
+
(150, 190, 200),
|
322 |
+
week_font,
|
323 |
+
stroke_width=2,
|
324 |
+
stroke_fill=(200, 150, 250),
|
325 |
+
)
|
326 |
+
|
327 |
+
scale_factor = 1.5
|
328 |
+
cell_size = 30
|
329 |
+
padding = 15
|
330 |
+
|
331 |
+
font = ImageFont.truetype("./Hellbot/resources/fonts/Montserrat.ttf", 30)
|
332 |
+
|
333 |
+
for week_num, week in enumerate(cal):
|
334 |
+
for day_num, day in enumerate(week):
|
335 |
+
x = int(day_num * (cell_size + padding) * scale_factor)
|
336 |
+
y = int((week_num + 3) * (cell_size + padding) * scale_factor)
|
337 |
+
|
338 |
+
cell_width = int(cell_size * scale_factor)
|
339 |
+
cell_height = int(cell_size * scale_factor)
|
340 |
+
|
341 |
+
text_x = (
|
342 |
+
int(x + (cell_width - draw.textlength(str(day), font=font)) // 2)
|
343 |
+
+ cell_size
|
344 |
+
)
|
345 |
+
text_y = (
|
346 |
+
int(y + (cell_height - draw.textlength(str(day), font=font)) // 2) - 55
|
347 |
+
)
|
348 |
+
|
349 |
+
if day != 0:
|
350 |
+
draw.text(
|
351 |
+
(text_x, text_y),
|
352 |
+
str(day),
|
353 |
+
(240, 200, 100),
|
354 |
+
font,
|
355 |
+
stroke_width=1,
|
356 |
+
stroke_fill=(0, 0, 0),
|
357 |
+
)
|
358 |
+
|
359 |
+
filename = f"calendar_{int(time.time())}.png"
|
360 |
+
calendar_image.save(filename, "PNG")
|
361 |
+
calendar_image.close()
|
362 |
+
|
363 |
+
return filename
|
364 |
+
|
365 |
+
|
366 |
+
async def create_thumbnail(photo: str, xy: tuple[int, int], file_size: int):
|
367 |
+
img = Image.open(photo)
|
368 |
+
img.thumbnail(xy)
|
369 |
+
|
370 |
+
size_in_bytes = file_size * 1024
|
371 |
+
quality = 90
|
372 |
+
|
373 |
+
while True:
|
374 |
+
img.save(photo, "JPEG", quality=quality, optimize=True)
|
375 |
+
if os.path.getsize(photo) <= size_in_bytes:
|
376 |
+
break
|
377 |
+
|
378 |
+
quality -= 5
|
379 |
+
|
380 |
+
return photo
|
381 |
+
|
382 |
+
|
383 |
+
async def download_images(query: str, limit: int) -> list[str]:
|
384 |
+
offset = random.randint(0, 20)
|
385 |
+
|
386 |
+
crawler = BingImageCrawler(log_level=logging.ERROR)
|
387 |
+
crawler.crawl(query, offset=offset, max_num=limit)
|
388 |
+
|
389 |
+
return [os.path.join("images", image) for image in os.listdir("images")]
|
HellBot/functions/media.py
ADDED
@@ -0,0 +1,192 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
from typing import Union
|
3 |
+
|
4 |
+
import requests
|
5 |
+
from pyrogram import Client
|
6 |
+
from pyrogram.file_id import FileId
|
7 |
+
from pyrogram.raw.functions.messages import UploadMedia
|
8 |
+
from pyrogram.raw.types import (
|
9 |
+
DocumentAttributeFilename,
|
10 |
+
InputDocument,
|
11 |
+
InputMediaUploadedDocument,
|
12 |
+
)
|
13 |
+
from pyrogram.types import Animation, Audio, Document, Message, Photo, Sticker, Video
|
14 |
+
|
15 |
+
from Hellbot.core import Symbols
|
16 |
+
|
17 |
+
|
18 |
+
async def get_metedata(media: Union[Animation, Audio, Document, Photo, Sticker, Video]):
|
19 |
+
output = "📄 MetaData:\n\n"
|
20 |
+
if isinstance(media, Animation):
|
21 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
22 |
+
output += f"<b>{Symbols.diamond_2} Width:</b> <code>{media.width}</code>\n"
|
23 |
+
output += f"<b>{Symbols.diamond_2} Height:</b> <code>{media.height}</code>\n"
|
24 |
+
output += (
|
25 |
+
f"<b>{Symbols.diamond_2} Duration:</b> <code>{media.duration}</code>\n"
|
26 |
+
)
|
27 |
+
output += (
|
28 |
+
f"<b>{Symbols.diamond_2} File Name:</b> <code>{media.file_name}</code>\n"
|
29 |
+
)
|
30 |
+
output += (
|
31 |
+
f"<b>{Symbols.diamond_2} Mime Type:</b> <code>{media.mime_type}</code>\n"
|
32 |
+
)
|
33 |
+
output += (
|
34 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
35 |
+
)
|
36 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
37 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Animation</code>\n"
|
38 |
+
elif isinstance(media, Audio):
|
39 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
40 |
+
output += (
|
41 |
+
f"<b>{Symbols.diamond_2} Duration:</b> <code>{media.duration}</code>\n"
|
42 |
+
)
|
43 |
+
output += (
|
44 |
+
f"<b>{Symbols.diamond_2} Performer:</b> <code>{media.performer}</code>\n"
|
45 |
+
)
|
46 |
+
output += f"<b>{Symbols.diamond_2} Title:</b> <code>{media.title}</code>\n"
|
47 |
+
output += (
|
48 |
+
f"<b>{Symbols.diamond_2} File Name:</b> <code>{media.file_name}</code>\n"
|
49 |
+
)
|
50 |
+
output += (
|
51 |
+
f"<b>{Symbols.diamond_2} Mime Type:</b> <code>{media.mime_type}</code>\n"
|
52 |
+
)
|
53 |
+
output += (
|
54 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
55 |
+
)
|
56 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
57 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Audio</code>\n"
|
58 |
+
elif isinstance(media, Document):
|
59 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
60 |
+
output += (
|
61 |
+
f"<b>{Symbols.diamond_2} File Name:</b> <code>{media.file_name}</code>\n"
|
62 |
+
)
|
63 |
+
output += (
|
64 |
+
f"<b>{Symbols.diamond_2} Mime Type:</b> <code>{media.mime_type}</code>\n"
|
65 |
+
)
|
66 |
+
output += (
|
67 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
68 |
+
)
|
69 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
70 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Document</code>\n"
|
71 |
+
elif isinstance(media, Photo):
|
72 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
73 |
+
output += f"<b>{Symbols.diamond_2} Width:</b> <code>{media.width}</code>\n"
|
74 |
+
output += f"<b>{Symbols.diamond_2} Height:</b> <code>{media.height}</code>\n"
|
75 |
+
output += f"<b>{Symbols.diamond_2} File Name:</b> <code>photo.jpg</code>\n"
|
76 |
+
output += f"<b>{Symbols.diamond_2} Mime Type:</b> <code>image/jpeg</code>\n"
|
77 |
+
output += (
|
78 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
79 |
+
)
|
80 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
81 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Photo</code>\n"
|
82 |
+
elif isinstance(media, Sticker):
|
83 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
84 |
+
output += f"<b>{Symbols.diamond_2} Width:</b> <code>{media.width}</code>\n"
|
85 |
+
output += f"<b>{Symbols.diamond_2} Height:</b> <code>{media.height}</code>\n"
|
86 |
+
output += (
|
87 |
+
f"<b>{Symbols.diamond_2} File Name:</b> <code>{media.file_name}</code>\n"
|
88 |
+
)
|
89 |
+
output += (
|
90 |
+
f"<b>{Symbols.diamond_2} Mime Type:</b> <code>{media.mime_type}</code>\n"
|
91 |
+
)
|
92 |
+
output += (
|
93 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
94 |
+
)
|
95 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
96 |
+
output += f"<b>{Symbols.diamond_2} Emoji:</b> <code>{media.emoji}</code>\n"
|
97 |
+
output += (
|
98 |
+
f"<b>{Symbols.diamond_2} Set Name:</b> <code>{media.set_name}</code>\n"
|
99 |
+
)
|
100 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Sticker</code>\n"
|
101 |
+
elif isinstance(media, Video):
|
102 |
+
output += f"<b>{Symbols.diamond_2} File ID:</b> <code>{media.file_id}</code>\n"
|
103 |
+
output += f"<b>{Symbols.diamond_2} Width:</b> <code>{media.width}</code>\n"
|
104 |
+
output += f"<b>{Symbols.diamond_2} Height:</b> <code>{media.height}</code>\n"
|
105 |
+
output += (
|
106 |
+
f"<b>{Symbols.diamond_2} Duration:</b> <code>{media.duration}</code>\n"
|
107 |
+
)
|
108 |
+
output += (
|
109 |
+
f"<b>{Symbols.diamond_2} File Name:</b> <code>{media.file_name}</code>\n"
|
110 |
+
)
|
111 |
+
output += (
|
112 |
+
f"<b>{Symbols.diamond_2} Mime Type:</b> <code>{media.mime_type}</code>\n"
|
113 |
+
)
|
114 |
+
output += (
|
115 |
+
f"<b>{Symbols.diamond_2} File Size:</b> <code>{media.file_size}</code>\n"
|
116 |
+
)
|
117 |
+
output += f"<b>{Symbols.diamond_2} Date:</b> <code>{media.date}</code>\n"
|
118 |
+
output += f"<b>{Symbols.diamond_2} File Type:</b> <code>Video</code>\n"
|
119 |
+
else:
|
120 |
+
return None
|
121 |
+
|
122 |
+
return output
|
123 |
+
|
124 |
+
|
125 |
+
def get_media_text_ocr(filename: str, api_key: str, language: str = "eng") -> dict:
|
126 |
+
payload = {
|
127 |
+
"isOverlayRequired": False,
|
128 |
+
"apikey": api_key,
|
129 |
+
"language": language,
|
130 |
+
}
|
131 |
+
|
132 |
+
with open(filename, "rb") as f:
|
133 |
+
r = requests.post(
|
134 |
+
"https://api.ocr.space/parse/image",
|
135 |
+
files={filename: f},
|
136 |
+
data=payload,
|
137 |
+
)
|
138 |
+
|
139 |
+
return r.json()
|
140 |
+
|
141 |
+
|
142 |
+
async def upload_media(client: Client, chat_id: int, file: str) -> InputDocument:
|
143 |
+
media = await client.invoke(
|
144 |
+
UploadMedia(
|
145 |
+
peer=(await client.resolve_peer(chat_id)),
|
146 |
+
media=InputMediaUploadedDocument(
|
147 |
+
file=(await client.save_file(file)),
|
148 |
+
mime_type=client.guess_mime_type(file) or "application/zip",
|
149 |
+
attributes=[
|
150 |
+
DocumentAttributeFilename(file_name=os.path.basename(file))
|
151 |
+
],
|
152 |
+
force_file=True,
|
153 |
+
),
|
154 |
+
),
|
155 |
+
)
|
156 |
+
|
157 |
+
return InputDocument(
|
158 |
+
id=media.document.id,
|
159 |
+
access_hash=media.document.access_hash,
|
160 |
+
file_reference=media.document.file_reference,
|
161 |
+
)
|
162 |
+
|
163 |
+
|
164 |
+
async def get_media_from_id(file_id: str) -> InputDocument:
|
165 |
+
file = FileId.decode(file_id)
|
166 |
+
|
167 |
+
return InputDocument(
|
168 |
+
id=file.media_id,
|
169 |
+
access_hash=file.access_hash,
|
170 |
+
file_reference=file.file_reference,
|
171 |
+
)
|
172 |
+
|
173 |
+
|
174 |
+
async def get_media_fileid(message: Message) -> str | None:
|
175 |
+
file_id = None
|
176 |
+
if message.photo:
|
177 |
+
file_id = message.photo.file_id
|
178 |
+
elif message.animation:
|
179 |
+
file_id = message.animation.file_id
|
180 |
+
elif message.audio:
|
181 |
+
file_id = message.audio.file_id
|
182 |
+
elif message.document:
|
183 |
+
file_id = message.document.file_id
|
184 |
+
elif message.video:
|
185 |
+
file_id = message.video.file_id
|
186 |
+
elif message.sticker:
|
187 |
+
file_id = message.sticker.file_id
|
188 |
+
elif message.video_note:
|
189 |
+
file_id = message.video_note.file_id
|
190 |
+
elif message.voice:
|
191 |
+
file_id = message.voice.file_id
|
192 |
+
return file_id
|
HellBot/functions/paste.py
ADDED
@@ -0,0 +1,49 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import uuid
|
2 |
+
|
3 |
+
import requests
|
4 |
+
|
5 |
+
from .utility import TGraph
|
6 |
+
|
7 |
+
|
8 |
+
def post_to_telegraph(
|
9 |
+
title: str,
|
10 |
+
content: str,
|
11 |
+
author: str = "[ 𝖧𝖾𝗅𝗅𝖡𝗈𝗍 ]",
|
12 |
+
url: str = "https://t.me/Its_HellBot",
|
13 |
+
) -> str:
|
14 |
+
content = content.replace("\n", "<br>")
|
15 |
+
try:
|
16 |
+
response = TGraph.telegraph.create_page(
|
17 |
+
title=title,
|
18 |
+
html_content=content,
|
19 |
+
author_name=author,
|
20 |
+
author_url=url,
|
21 |
+
)
|
22 |
+
except Exception:
|
23 |
+
rnd_key = uuid.uuid4().hex[:8]
|
24 |
+
title = f"{title}_{rnd_key}"
|
25 |
+
response = TGraph.telegraph.create_page(
|
26 |
+
title=title,
|
27 |
+
html_content=content,
|
28 |
+
author_name=author,
|
29 |
+
author_url=url,
|
30 |
+
)
|
31 |
+
|
32 |
+
return f"https://te.legra.ph/{response['path']}"
|
33 |
+
|
34 |
+
|
35 |
+
def spaceBin(data: str, extension: str = "none") -> str:
|
36 |
+
data = {
|
37 |
+
"content": data,
|
38 |
+
"extension": extension,
|
39 |
+
}
|
40 |
+
|
41 |
+
resp = requests.post("https://spaceb.in/api/v1/documents/", data)
|
42 |
+
|
43 |
+
try:
|
44 |
+
result = resp.json()
|
45 |
+
url = f"https://spaceb.in/{result['payload']['id']}"
|
46 |
+
except Exception:
|
47 |
+
url = ""
|
48 |
+
|
49 |
+
return url
|
HellBot/functions/scraping.py
ADDED
@@ -0,0 +1,528 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import calendar
|
2 |
+
import time
|
3 |
+
|
4 |
+
import httpx
|
5 |
+
from bs4 import BeautifulSoup
|
6 |
+
from urllib.parse import quote, urlparse
|
7 |
+
from Hellbot.core import Symbols
|
8 |
+
|
9 |
+
from .paste import post_to_telegraph
|
10 |
+
from .templates import (
|
11 |
+
airing_templates,
|
12 |
+
anilist_user_templates,
|
13 |
+
anime_template,
|
14 |
+
character_templates,
|
15 |
+
manga_templates,
|
16 |
+
)
|
17 |
+
|
18 |
+
anime_query = """query ($id: Int,$search: String) {
|
19 |
+
Page (perPage: 10) {
|
20 |
+
media (id: $id, type: ANIME,search: $search) {
|
21 |
+
id
|
22 |
+
title {
|
23 |
+
romaji
|
24 |
+
english
|
25 |
+
native
|
26 |
+
}
|
27 |
+
type
|
28 |
+
format
|
29 |
+
status
|
30 |
+
description (asHtml: false)
|
31 |
+
episodes
|
32 |
+
duration
|
33 |
+
countryOfOrigin
|
34 |
+
source
|
35 |
+
trailer{
|
36 |
+
id
|
37 |
+
site
|
38 |
+
}
|
39 |
+
genres
|
40 |
+
tags {
|
41 |
+
name
|
42 |
+
}
|
43 |
+
isAdult
|
44 |
+
averageScore
|
45 |
+
studios (isMain: true){
|
46 |
+
nodes{
|
47 |
+
name
|
48 |
+
}
|
49 |
+
}
|
50 |
+
nextAiringEpisode{
|
51 |
+
episode
|
52 |
+
}
|
53 |
+
siteUrl
|
54 |
+
}
|
55 |
+
}
|
56 |
+
}"""
|
57 |
+
|
58 |
+
|
59 |
+
manga_query = """query ($id: Int,$search: String) {
|
60 |
+
Page (perPage: 10) {
|
61 |
+
media (id: $id, type: MANGA,search: $search) {
|
62 |
+
id
|
63 |
+
title {
|
64 |
+
romaji
|
65 |
+
english
|
66 |
+
native
|
67 |
+
}
|
68 |
+
type
|
69 |
+
format
|
70 |
+
status
|
71 |
+
description (asHtml: false)
|
72 |
+
chapters
|
73 |
+
volumes
|
74 |
+
countryOfOrigin
|
75 |
+
source
|
76 |
+
genres
|
77 |
+
isAdult
|
78 |
+
averageScore
|
79 |
+
siteUrl
|
80 |
+
}
|
81 |
+
}
|
82 |
+
}"""
|
83 |
+
|
84 |
+
|
85 |
+
character_query = """query ($id: Int, $search: String) {
|
86 |
+
Page {
|
87 |
+
characters (id: $id, search: $search) {
|
88 |
+
id
|
89 |
+
name {
|
90 |
+
full
|
91 |
+
native
|
92 |
+
}
|
93 |
+
image {
|
94 |
+
large
|
95 |
+
}
|
96 |
+
description
|
97 |
+
gender
|
98 |
+
dateOfBirth {
|
99 |
+
year
|
100 |
+
month
|
101 |
+
day
|
102 |
+
}
|
103 |
+
age
|
104 |
+
bloodType
|
105 |
+
siteUrl
|
106 |
+
favourites
|
107 |
+
media {
|
108 |
+
nodes {
|
109 |
+
title {
|
110 |
+
romaji
|
111 |
+
english
|
112 |
+
native
|
113 |
+
}
|
114 |
+
type
|
115 |
+
format
|
116 |
+
siteUrl
|
117 |
+
}
|
118 |
+
}
|
119 |
+
}
|
120 |
+
}
|
121 |
+
}"""
|
122 |
+
|
123 |
+
|
124 |
+
airing_query = """query ($id: Int, $idMal:Int, $search: String) {
|
125 |
+
Media (id: $id, idMal: $idMal, search: $search, type: ANIME) {
|
126 |
+
id
|
127 |
+
title {
|
128 |
+
romaji
|
129 |
+
english
|
130 |
+
native
|
131 |
+
}
|
132 |
+
status
|
133 |
+
episodes
|
134 |
+
countryOfOrigin
|
135 |
+
nextAiringEpisode {
|
136 |
+
airingAt
|
137 |
+
timeUntilAiring
|
138 |
+
episode
|
139 |
+
}
|
140 |
+
}
|
141 |
+
}"""
|
142 |
+
|
143 |
+
|
144 |
+
anilist_user_query = """query($id: Int, $search: String) {
|
145 |
+
User(id: $id, name: $search) {
|
146 |
+
id
|
147 |
+
name
|
148 |
+
siteUrl
|
149 |
+
statistics {
|
150 |
+
anime {
|
151 |
+
count
|
152 |
+
meanScore
|
153 |
+
minutesWatched
|
154 |
+
episodesWatched
|
155 |
+
}
|
156 |
+
manga {
|
157 |
+
count
|
158 |
+
meanScore
|
159 |
+
chaptersRead
|
160 |
+
volumesRead
|
161 |
+
}
|
162 |
+
}
|
163 |
+
}
|
164 |
+
}"""
|
165 |
+
|
166 |
+
|
167 |
+
def is_valid_url(text: str) -> bool:
|
168 |
+
try:
|
169 |
+
result = urlparse(text)
|
170 |
+
return all([result.scheme, result.netloc])
|
171 |
+
except ValueError:
|
172 |
+
return False
|
173 |
+
|
174 |
+
|
175 |
+
def post_request(query: str, search_term: str):
|
176 |
+
url = "https://graphql.anilist.co"
|
177 |
+
variables = {"search": search_term}
|
178 |
+
r = httpx.post(url, json={"query": query, "variables": variables})
|
179 |
+
if r.status_code != 200:
|
180 |
+
return None
|
181 |
+
return r.json()
|
182 |
+
|
183 |
+
|
184 |
+
def get_country_flag(country: str) -> str:
|
185 |
+
base = ord('🇦')
|
186 |
+
emoji_flag = "".join(chr(base + ord(letter) - ord("A")) for letter in country)
|
187 |
+
return emoji_flag
|
188 |
+
|
189 |
+
|
190 |
+
def get_date(data: dict) -> str:
|
191 |
+
try:
|
192 |
+
year = data["year"] or ""
|
193 |
+
if not year and not data["month"]:
|
194 |
+
return "N/A"
|
195 |
+
day = data["day"]
|
196 |
+
if 10 <= day % 100 <= 20:
|
197 |
+
day = f"{day}th"
|
198 |
+
else:
|
199 |
+
day_dict = {1: "st", 2: "nd", 3: "rd"}
|
200 |
+
day = f"{day}{day_dict.get(day % 10, 'th')}"
|
201 |
+
month_name = calendar.month_name[int(data["month"])]
|
202 |
+
return f"{day} {month_name} {year}"
|
203 |
+
except:
|
204 |
+
return "N/A"
|
205 |
+
|
206 |
+
|
207 |
+
def search_anime_filler(search_term: str):
|
208 |
+
BASE = "https://www.animefillerlist.com/shows/"
|
209 |
+
|
210 |
+
response = httpx.get(BASE).text
|
211 |
+
soup = BeautifulSoup(response, "html.parser")
|
212 |
+
div = soup.findAll("div", {"class": "Group"})
|
213 |
+
|
214 |
+
index = {}
|
215 |
+
for i in div:
|
216 |
+
li = i.findAll("li")
|
217 |
+
for j in li:
|
218 |
+
index[j.text] = j.a["href"].split("/")[-1]
|
219 |
+
|
220 |
+
results = {}
|
221 |
+
keys = list(index.keys())
|
222 |
+
for i in range(len(keys)):
|
223 |
+
if search_term.lower() in keys[i].lower():
|
224 |
+
results[keys[i]] = index[keys[i]]
|
225 |
+
|
226 |
+
for result in results.keys():
|
227 |
+
data = []
|
228 |
+
response = httpx.get(BASE + results[result]).text
|
229 |
+
soup = BeautifulSoup(response, "html.parser")
|
230 |
+
base_div = soup.find("div", {"id": "Condensed"})
|
231 |
+
|
232 |
+
if not base_div:
|
233 |
+
continue
|
234 |
+
|
235 |
+
divs = base_div.findAll("div")
|
236 |
+
for div in divs:
|
237 |
+
heading = div.find("span", {"class": "Label"}).text
|
238 |
+
episodes = div.find("span", {"class": "Episodes"}).text
|
239 |
+
data.append((heading, episodes))
|
240 |
+
|
241 |
+
yield result, data
|
242 |
+
|
243 |
+
|
244 |
+
async def get_filler_info(search_term: str) -> str:
|
245 |
+
animes = search_anime_filler(search_term)
|
246 |
+
message = ""
|
247 |
+
|
248 |
+
for anime in animes:
|
249 |
+
html_message = f"<strong>{Symbols.check_mark} {anime[0]} Filler Guide:</strong>\n\n"
|
250 |
+
|
251 |
+
for data in anime[1]:
|
252 |
+
html_message += (
|
253 |
+
f"<h3>{Symbols.bullet} {data[0]}:</h3> \n<code>{data[1]}</code>\n\n"
|
254 |
+
)
|
255 |
+
|
256 |
+
paste = post_to_telegraph(anime[0] + " Filler Guide", html_message)
|
257 |
+
message += f"{Symbols.anchor} [{anime[0]}]({paste})\n"
|
258 |
+
|
259 |
+
return message
|
260 |
+
|
261 |
+
|
262 |
+
def search_watch_order(anime: str):
|
263 |
+
response = httpx.get(f"https://www.animechrono.com/search?q={quote(anime)}")
|
264 |
+
soup = BeautifulSoup(response.text, "html.parser")
|
265 |
+
|
266 |
+
item_div = soup.find("div", {"class": "search-result-items"})
|
267 |
+
animes = item_div.find_all("a", {"class": "list-item search w-inline-block"})
|
268 |
+
|
269 |
+
for anime in animes:
|
270 |
+
name = anime.find("h1", "h1 search").text
|
271 |
+
yield name, anime["href"]
|
272 |
+
|
273 |
+
|
274 |
+
async def get_watch_order(search_term: str) -> str:
|
275 |
+
animes = search_watch_order(search_term)
|
276 |
+
message = ""
|
277 |
+
|
278 |
+
for anime in animes:
|
279 |
+
message += f"**{Symbols.anchor} {anime[0]}:** \n"
|
280 |
+
response = httpx.get("https://www.animechrono.com" + anime[1]).text
|
281 |
+
soup = BeautifulSoup(response, "html.parser")
|
282 |
+
|
283 |
+
elements = soup.find_all("h2", {"class": "heading-5"})
|
284 |
+
for element in elements:
|
285 |
+
message += f" {Symbols.bullet} `{element.text}`\n"
|
286 |
+
message += "\n"
|
287 |
+
|
288 |
+
return message
|
289 |
+
|
290 |
+
|
291 |
+
async def get_anime_info(search_term: str) -> tuple[str, str]:
|
292 |
+
data = post_request(anime_query, search_term)
|
293 |
+
if not data:
|
294 |
+
return "", ""
|
295 |
+
|
296 |
+
data = data["data"]["Page"]["media"][0]
|
297 |
+
english_title = data["title"]["english"]
|
298 |
+
native_title = data["title"]["native"]
|
299 |
+
if not english_title:
|
300 |
+
english_title = data["title"]["romaji"]
|
301 |
+
flag = get_country_flag(data["countryOfOrigin"])
|
302 |
+
name = f"**[{flag}] {english_title} ({native_title})**"
|
303 |
+
|
304 |
+
anime_id = data["id"]
|
305 |
+
score = data["averageScore"] if data["averageScore"] else "N/A"
|
306 |
+
source = str(data["source"]).title() if data["source"] else "N/A"
|
307 |
+
mtype = str(data["type"]).title() if data["type"] else "N/A"
|
308 |
+
synopsis = data["description"]
|
309 |
+
|
310 |
+
episodes = data["episodes"]
|
311 |
+
if not episodes:
|
312 |
+
try:
|
313 |
+
episodes = data["nextAiringEpisode"]["episode"] - 1
|
314 |
+
except:
|
315 |
+
episodes = "N/A"
|
316 |
+
|
317 |
+
duration = data["duration"] if data["duration"] else "N/A"
|
318 |
+
status = str(data["status"]).title() if data["status"] else "N/A"
|
319 |
+
format = str(data["format"]).title() if data["format"] else "N/A"
|
320 |
+
genre = ", ".join(data["genres"]) if data["genres"] else "N/A"
|
321 |
+
tags = ", ".join([i["name"] for i in data["tags"][:5]]) if data["tags"] else "N/A"
|
322 |
+
studio = data["studios"]["nodes"][0]["name"] if data["studios"]["nodes"] else "N/A"
|
323 |
+
siteurl = f"[Anilist Website]({data['siteUrl']})" if data["siteUrl"] else "N/A"
|
324 |
+
isAdult = data["isAdult"]
|
325 |
+
|
326 |
+
trailer = "N/A"
|
327 |
+
if data["trailer"] and data["trailer"]["site"] == "youtube":
|
328 |
+
trailer = f"[Youtube](https://youtu.be/{data['trailer']['id']})"
|
329 |
+
|
330 |
+
response = httpx.get(f"https://img.anili.st/media/{anime_id}").content
|
331 |
+
banner = f"anime_{anime_id}.jpg"
|
332 |
+
with open(banner, "wb") as f:
|
333 |
+
f.write(response)
|
334 |
+
|
335 |
+
description = post_to_telegraph(name, synopsis)
|
336 |
+
|
337 |
+
message = await anime_template(
|
338 |
+
name=name,
|
339 |
+
score=score,
|
340 |
+
source=source,
|
341 |
+
mtype=mtype,
|
342 |
+
episodes=episodes,
|
343 |
+
duration=duration,
|
344 |
+
status=status,
|
345 |
+
format=format,
|
346 |
+
genre=genre,
|
347 |
+
studio=studio,
|
348 |
+
trailer=trailer,
|
349 |
+
siteurl=siteurl,
|
350 |
+
description=description,
|
351 |
+
tags=tags,
|
352 |
+
isAdult=isAdult,
|
353 |
+
)
|
354 |
+
|
355 |
+
return message, banner
|
356 |
+
|
357 |
+
|
358 |
+
async def get_manga_info(search_term: str) -> tuple[str, str]:
|
359 |
+
data = post_request(manga_query, search_term)
|
360 |
+
if not data:
|
361 |
+
return "", ""
|
362 |
+
|
363 |
+
data = data["data"]["Page"]["media"][0]
|
364 |
+
english_title = data["title"]["english"]
|
365 |
+
native_title = data["title"]["native"]
|
366 |
+
if not english_title:
|
367 |
+
english_title = data["title"]["romaji"]
|
368 |
+
flag = get_country_flag(data["countryOfOrigin"])
|
369 |
+
name = f"**[{flag}] {english_title} ({native_title})**"
|
370 |
+
|
371 |
+
manga_id = data["id"]
|
372 |
+
score = data["averageScore"] if data["averageScore"] else "N/A"
|
373 |
+
source = str(data["source"]).title() if data["source"] else "N/A"
|
374 |
+
mtype = str(data["type"]).title() if data["type"] else "N/A"
|
375 |
+
synopsis = data["description"]
|
376 |
+
|
377 |
+
chapters = data["chapters"] if data["chapters"] else "N/A"
|
378 |
+
volumes = data["volumes"] if data["volumes"] else "N/A"
|
379 |
+
status = str(data["status"]).title() if data["status"] else "N/A"
|
380 |
+
format = str(data["format"]).title() if data["format"] else "N/A"
|
381 |
+
genre = ", ".join(data["genres"]) if data["genres"] else "N/A"
|
382 |
+
siteurl = f"[Anilist Website]({data['siteUrl']})" if data["siteUrl"] else "N/A"
|
383 |
+
isAdult = data["isAdult"]
|
384 |
+
|
385 |
+
response = httpx.get(f"https://img.anili.st/media/{manga_id}").content
|
386 |
+
banner = f"manga_{manga_id}.jpg"
|
387 |
+
with open(banner, "wb") as f:
|
388 |
+
f.write(response)
|
389 |
+
|
390 |
+
description = post_to_telegraph(name, synopsis)
|
391 |
+
|
392 |
+
message = await manga_templates(
|
393 |
+
name=name,
|
394 |
+
score=score,
|
395 |
+
source=source,
|
396 |
+
mtype=mtype,
|
397 |
+
chapters=chapters,
|
398 |
+
volumes=volumes,
|
399 |
+
status=status,
|
400 |
+
format=format,
|
401 |
+
genre=genre,
|
402 |
+
siteurl=siteurl,
|
403 |
+
description=description,
|
404 |
+
isAdult=isAdult,
|
405 |
+
)
|
406 |
+
|
407 |
+
return message, banner
|
408 |
+
|
409 |
+
|
410 |
+
async def get_character_info(search_term: str) -> tuple[str, str]:
|
411 |
+
data = post_request(character_query, search_term)
|
412 |
+
if not data:
|
413 |
+
return "", ""
|
414 |
+
|
415 |
+
data = data["data"]["Page"]["characters"][0]
|
416 |
+
name = f"**{data['name']['full']} ({data['name']['native']})**"
|
417 |
+
char_id = data["id"]
|
418 |
+
|
419 |
+
description = data["description"].split("\n\n", 1)[0]
|
420 |
+
gender = data["gender"] if data["gender"] else "N/A"
|
421 |
+
date_of_birth = get_date(data["dateOfBirth"])
|
422 |
+
age = data["age"] if data["age"] else "N/A"
|
423 |
+
blood_type = data["bloodType"] if data["bloodType"] else "N/A"
|
424 |
+
siteurl = f"[Anilist Website]({data['siteUrl']})" if data["siteUrl"] else "N/A"
|
425 |
+
favorites = data["favourites"] if data["favourites"] else "N/A"
|
426 |
+
|
427 |
+
cameo = data["media"]["nodes"][0] if data["media"]["nodes"] else {}
|
428 |
+
if cameo:
|
429 |
+
role_in = f"\n╰➢ **𝖱𝗈𝗅𝖾 𝖨𝗇:** [{cameo['title']['romaji']}]({cameo['siteUrl']})"
|
430 |
+
else:
|
431 |
+
role_in = ""
|
432 |
+
|
433 |
+
response = httpx.get(data["image"]["large"]).content
|
434 |
+
banner = f"character_{char_id}.jpg"
|
435 |
+
with open(banner, "wb") as f:
|
436 |
+
f.write(response)
|
437 |
+
|
438 |
+
message = await character_templates(
|
439 |
+
name=name,
|
440 |
+
gender=gender,
|
441 |
+
date_of_birth=date_of_birth,
|
442 |
+
age=age,
|
443 |
+
blood_type=blood_type,
|
444 |
+
favorites=favorites,
|
445 |
+
siteurl=siteurl,
|
446 |
+
role_in=role_in,
|
447 |
+
description=description,
|
448 |
+
)
|
449 |
+
|
450 |
+
return message, banner
|
451 |
+
|
452 |
+
|
453 |
+
async def get_airing_info(search_term: str) -> tuple[str, str]:
|
454 |
+
data = post_request(airing_query, search_term)
|
455 |
+
if not data:
|
456 |
+
return "", ""
|
457 |
+
|
458 |
+
data = data["data"]["Media"]
|
459 |
+
english_title = data["title"]["english"]
|
460 |
+
native_title = data["title"]["native"]
|
461 |
+
if not english_title:
|
462 |
+
english_title = data["title"]["romaji"]
|
463 |
+
flag = get_country_flag(data["countryOfOrigin"])
|
464 |
+
name = f"**[{flag}] {english_title} ({native_title})**"
|
465 |
+
|
466 |
+
episode = data["episodes"]
|
467 |
+
if not episode:
|
468 |
+
try:
|
469 |
+
episode = data["nextAiringEpisode"]["episode"]
|
470 |
+
except:
|
471 |
+
episode = "N/A"
|
472 |
+
|
473 |
+
response = httpx.get(f"https://img.anili.st/media/{data['id']}").content
|
474 |
+
banner = f"airing_{data['id']}.jpg"
|
475 |
+
with open(banner, "wb") as f:
|
476 |
+
f.write(response)
|
477 |
+
|
478 |
+
status = str(data["status"]).title() if data["status"] else "N/A"
|
479 |
+
next_date = data["nextAiringEpisode"]["airingAt"] if data["nextAiringEpisode"] else ""
|
480 |
+
|
481 |
+
airing_info = ""
|
482 |
+
if next_date:
|
483 |
+
airing_info = f"\n**🗓️ {time.ctime(next_date)}**"
|
484 |
+
|
485 |
+
message = await airing_templates(
|
486 |
+
name=name,
|
487 |
+
status=status,
|
488 |
+
episode=episode,
|
489 |
+
airing_info=airing_info,
|
490 |
+
)
|
491 |
+
|
492 |
+
return message, banner
|
493 |
+
|
494 |
+
|
495 |
+
async def get_anilist_user_info(search_term: str) -> tuple[str, str]:
|
496 |
+
data = post_request(anilist_user_query, search_term)
|
497 |
+
if not data:
|
498 |
+
return "", ""
|
499 |
+
|
500 |
+
data = data["data"]["User"]
|
501 |
+
user_id = data["id"]
|
502 |
+
name = data['name']
|
503 |
+
siteurl = f"[Anilist Website]({data['siteUrl']})" if data["siteUrl"] else "N/A"
|
504 |
+
|
505 |
+
response = httpx.get(f"https://img.anili.st/user/{user_id}").content
|
506 |
+
banner = f"aniuser_{user_id}.jpg"
|
507 |
+
with open(banner, "wb") as f:
|
508 |
+
f.write(response)
|
509 |
+
|
510 |
+
anime_stats = data["statistics"]["anime"]
|
511 |
+
manga_stats = data["statistics"]["manga"]
|
512 |
+
|
513 |
+
anime = (
|
514 |
+
anime_stats["count"] or 0,
|
515 |
+
anime_stats["meanScore"] or 0,
|
516 |
+
anime_stats["minutesWatched"] or 0,
|
517 |
+
anime_stats["episodesWatched"] or 0,
|
518 |
+
)
|
519 |
+
manga = (
|
520 |
+
manga_stats["count"] or 0,
|
521 |
+
manga_stats["meanScore"] or 0,
|
522 |
+
manga_stats["chaptersRead"] or 0,
|
523 |
+
manga_stats["volumesRead"] or 0,
|
524 |
+
)
|
525 |
+
|
526 |
+
message = await anilist_user_templates(name, anime, manga, siteurl)
|
527 |
+
|
528 |
+
return message, banner
|
HellBot/functions/sticker.py
ADDED
@@ -0,0 +1,135 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from typing import Tuple
|
2 |
+
|
3 |
+
from emoji import EMOJI_DATA
|
4 |
+
from pyrogram import Client
|
5 |
+
from pyrogram.raw.functions.messages import GetStickerSet
|
6 |
+
from pyrogram.raw.functions.stickers import AddStickerToSet, CreateStickerSet, RemoveStickerFromSet
|
7 |
+
from pyrogram.raw import base, types
|
8 |
+
from pyrogram.types import Message
|
9 |
+
|
10 |
+
from .media import get_media_from_id, upload_media
|
11 |
+
|
12 |
+
|
13 |
+
def is_emoji(text: str) -> bool:
|
14 |
+
return any(c in EMOJI_DATA for c in text)
|
15 |
+
|
16 |
+
|
17 |
+
def get_emoji_and_id(message: Message) -> Tuple[int, str]:
|
18 |
+
pack_id = None
|
19 |
+
pack_emoji = None
|
20 |
+
|
21 |
+
for command in message.command:
|
22 |
+
if command.isdigit():
|
23 |
+
pack_id = int(command)
|
24 |
+
elif is_emoji(command):
|
25 |
+
pack_emoji = command
|
26 |
+
|
27 |
+
if pack_id is None:
|
28 |
+
pack_id = 1
|
29 |
+
|
30 |
+
if pack_emoji is None:
|
31 |
+
sticker = message.reply_to_message.sticker
|
32 |
+
try:
|
33 |
+
pack_emoji = sticker.emoji if sticker and sticker.emoji else "🍀"
|
34 |
+
except:
|
35 |
+
pack_emoji = "🍀"
|
36 |
+
|
37 |
+
return pack_id, pack_emoji
|
38 |
+
|
39 |
+
|
40 |
+
def check_sticker_data(replied: Message) -> Tuple[str | None, bool, bool, bool, int]:
|
41 |
+
pack_type = None
|
42 |
+
is_animated = False
|
43 |
+
is_video = False
|
44 |
+
is_static = False
|
45 |
+
pack_limit = 50
|
46 |
+
|
47 |
+
if replied.sticker:
|
48 |
+
if replied.sticker.is_animated:
|
49 |
+
pack_type, is_animated = "animated", True
|
50 |
+
elif replied.sticker.is_video:
|
51 |
+
pack_type, is_video = "video", True
|
52 |
+
else:
|
53 |
+
pack_type, is_static, pack_limit = "static", True, 120
|
54 |
+
|
55 |
+
elif replied.photo:
|
56 |
+
pack_type, is_static, pack_limit = "static", True, 120
|
57 |
+
|
58 |
+
elif replied.video or replied.animation:
|
59 |
+
pack_type, is_video = "video", True
|
60 |
+
|
61 |
+
elif replied.document:
|
62 |
+
mime_type = replied.document.mime_type.lower()
|
63 |
+
if mime_type.startswith("video/"):
|
64 |
+
pack_type, is_video = "video", True
|
65 |
+
elif mime_type.startswith("image/"):
|
66 |
+
pack_type, is_static, pack_limit = "static", True, 120
|
67 |
+
elif mime_type in ["application/x-tgsticker", "application/x-bad-tgsticker"]:
|
68 |
+
pack_type, is_animated = "animated", True
|
69 |
+
|
70 |
+
return pack_type, is_animated, is_video, is_static, pack_limit
|
71 |
+
|
72 |
+
|
73 |
+
async def create_sticker(
|
74 |
+
client: Client,
|
75 |
+
chat_id: int,
|
76 |
+
file: str,
|
77 |
+
emoji: str,
|
78 |
+
) -> types.InputStickerSetItem:
|
79 |
+
sticker = await upload_media(client, chat_id, file)
|
80 |
+
|
81 |
+
return types.InputStickerSetItem(
|
82 |
+
document=sticker,
|
83 |
+
emoji=emoji,
|
84 |
+
)
|
85 |
+
|
86 |
+
|
87 |
+
async def remove_sticker(client: Client, stickerid: str) -> base.messages.StickerSet:
|
88 |
+
sticker = await get_media_from_id(stickerid)
|
89 |
+
return await client.invoke(RemoveStickerFromSet(sticker=sticker))
|
90 |
+
|
91 |
+
|
92 |
+
async def get_sticker_set(client: Client, name: str) -> base.messages.StickerSet | None:
|
93 |
+
try:
|
94 |
+
return await client.invoke(
|
95 |
+
GetStickerSet(
|
96 |
+
stickerset=types.InputStickerSetShortName(short_name=name),
|
97 |
+
hash=0,
|
98 |
+
)
|
99 |
+
)
|
100 |
+
except:
|
101 |
+
return None
|
102 |
+
|
103 |
+
|
104 |
+
async def add_sticker(
|
105 |
+
client: Client,
|
106 |
+
stickerset: base.messages.StickerSet,
|
107 |
+
sticker: base.InputStickerSetItem,
|
108 |
+
) -> base.messages.StickerSet:
|
109 |
+
return await client.invoke(
|
110 |
+
AddStickerToSet(
|
111 |
+
stickerset=types.InputStickerSetShortName(short_name=stickerset.set.short_name),
|
112 |
+
sticker=sticker,
|
113 |
+
)
|
114 |
+
)
|
115 |
+
|
116 |
+
|
117 |
+
async def new_sticker_set(
|
118 |
+
client: Client,
|
119 |
+
user_id: int,
|
120 |
+
title: str,
|
121 |
+
short_name: str,
|
122 |
+
stickers: list[base.InputStickerSetItem],
|
123 |
+
animated: bool,
|
124 |
+
video: bool,
|
125 |
+
) -> base.messages.StickerSet:
|
126 |
+
return await client.invoke(
|
127 |
+
CreateStickerSet(
|
128 |
+
user_id=(await client.resolve_peer(user_id)),
|
129 |
+
title=title,
|
130 |
+
short_name=short_name,
|
131 |
+
stickers=stickers,
|
132 |
+
animated=animated,
|
133 |
+
videos=video,
|
134 |
+
)
|
135 |
+
)
|
HellBot/functions/templates.py
ADDED
@@ -0,0 +1,466 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import random
|
2 |
+
|
3 |
+
from Hellbot import __version__
|
4 |
+
from Hellbot.core import ENV, db
|
5 |
+
|
6 |
+
ALIVE_TEMPLATES = [
|
7 |
+
(
|
8 |
+
"•────────────────•\n"
|
9 |
+
"• 𝐇ᴇʟʟ𝐁ᴏᴛ 𝐈s 𝐀ʟɪᴠᴇ •\n"
|
10 |
+
"╭────────────────•\n"
|
11 |
+
"╰➢ ᴏᴡɴᴇʀ » {owner}\n"
|
12 |
+
"╰➢ ᴘʏʀᴏɢʀᴀᴍ » {pyrogram}\n"
|
13 |
+
"╰➢ ʜᴇʟʟʙᴏᴛ » {hellbot}\n"
|
14 |
+
"╰➢ ᴘʏᴛʜᴏɴ » {python}\n"
|
15 |
+
"╰➢ ᴜᴘᴛɪᴍᴇ » {uptime}\n"
|
16 |
+
"╰────────────────•\n"
|
17 |
+
"𝖡𝗒 © @HellBot_Networks\n"
|
18 |
+
"•────────────────•\n"
|
19 |
+
),
|
20 |
+
]
|
21 |
+
|
22 |
+
PING_TEMPLATES = [
|
23 |
+
"""**🍀 𝖯𝗂𝗇𝗀!**
|
24 |
+
|
25 |
+
⚘ **ѕρєє∂:** {speed} m/s
|
26 |
+
⚘ **υρтιмє:** {uptime}
|
27 |
+
⚘ **σωηєя:** {owner}""",
|
28 |
+
]
|
29 |
+
|
30 |
+
HELP_MENU_TEMPLATES = [
|
31 |
+
"""**🍀 𝖧𝖾𝗅𝗉 𝖬𝖾𝗇𝗎 𝖿𝗈𝗋:** {owner}
|
32 |
+
|
33 |
+
__📃 𝖫𝗈𝖺𝖽𝖾𝖽__ **{plugins} 𝗉𝗅𝗎𝗀𝗂𝗇𝗌** __𝗐𝗂𝗍𝗁 𝖺 𝗍𝗈𝗍𝖺𝗅 𝗈𝖿__ **{commands} 𝖼𝗈𝗆𝗆𝖺𝗇𝖽𝗌.**
|
34 |
+
|
35 |
+
**📑 Page:** __{current}/{last}__""",
|
36 |
+
]
|
37 |
+
|
38 |
+
COMMAND_MENU_TEMPLATES = [
|
39 |
+
"""**𝖯𝗅𝗎𝗀𝗂𝗇 𝖥𝗂𝗅𝖾:** `{file}`
|
40 |
+
**𝖯𝗅𝗎𝗀𝗂𝗇 𝖨𝗇𝖿𝗈:** __{info} 🍀__
|
41 |
+
|
42 |
+
**📃 𝖫𝗈𝖺𝖽𝖾𝖽 𝖢𝗈𝗆𝗆𝖺𝗇𝖽𝗌:** `{commands}`""",
|
43 |
+
]
|
44 |
+
|
45 |
+
ANIME_TEMPLATES = [
|
46 |
+
"""
|
47 |
+
{name}
|
48 |
+
|
49 |
+
╭────────────────•
|
50 |
+
╰➢ **𝖲𝖼𝗈𝗋𝖾:** `{score}`
|
51 |
+
╰➢ **𝖲𝗈𝗎𝗋𝖼𝖾:** `{source}`
|
52 |
+
╰➢ **𝖳𝗒𝗉𝖾:** `{mtype}`
|
53 |
+
╰➢ **𝖤𝗉𝗂𝗌𝗈𝖽𝖾𝗌:** `{episodes}`
|
54 |
+
╰➢ **𝖣𝗎𝗋𝖺𝗍𝗂𝗈𝗇:** `{duration} minutes`
|
55 |
+
╰➢ **𝖲𝗍𝖺𝗍𝗎𝗌:** `{status}`
|
56 |
+
╰➢ **𝖥𝗈𝗋𝗆𝖺𝗍:** `{format}`
|
57 |
+
╰➢ **𝖦𝖾𝗇𝗋𝖾:** `{genre}`
|
58 |
+
╰➢ **𝖳𝖺𝗀𝗌:** `{tags}`
|
59 |
+
╰➢ **𝖠𝖽𝗎𝗅𝗍 𝖱𝖺𝗍𝖾𝖽:** `{isAdult}`
|
60 |
+
╰➢ **𝖲𝗍𝗎𝖽𝗂𝗈:** `{studio}`
|
61 |
+
╰➢ **𝖳𝗋𝖺𝗂𝗅𝖾𝗋:** {trailer}
|
62 |
+
╰➢ **𝖶𝖾𝖻𝗌𝗂𝗍𝖾:** {siteurl}
|
63 |
+
╰➢ **𝖲𝗒𝗇𝗈𝗉𝗌𝗂𝗌:** [𝖢𝗅𝗂𝖼𝗄 𝖧𝖾𝗋𝖾]({description})
|
64 |
+
╰────────────────•
|
65 |
+
"""
|
66 |
+
]
|
67 |
+
|
68 |
+
MANGA_TEMPLATES = [
|
69 |
+
"""
|
70 |
+
{name}
|
71 |
+
|
72 |
+
╭────────────────•
|
73 |
+
╰➢ **𝖲𝖼𝗈𝗋𝖾:** `{score}`
|
74 |
+
╰➢ **𝖲𝗈𝗎𝗋𝖼𝖾:** `{source}`
|
75 |
+
╰➢ **𝖳𝗒𝗉𝖾:** `{mtype}`
|
76 |
+
╰➢ **𝖢𝗁𝖺𝗉𝗍𝖾𝗋𝗌:** `{chapters}`
|
77 |
+
╰➢ **𝖵𝗈𝗅𝗎𝗆𝖾𝗌:** `{volumes}`
|
78 |
+
╰➢ **𝖲𝗍𝖺𝗍𝗎𝗌:** `{status}`
|
79 |
+
╰➢ **𝖥𝗈𝗋𝗆𝖺𝗍:** `{format}`
|
80 |
+
╰➢ **𝖦𝖾𝗇𝗋𝖾:** `{genre}`
|
81 |
+
╰➢ **𝖠𝖽𝗎𝗅𝗍 𝖱𝖺𝗍𝖾𝖽:** `{isAdult}`
|
82 |
+
╰➢ **𝖶𝖾𝖻𝗌𝗂𝗍𝖾:** {siteurl}
|
83 |
+
╰➢ **𝖲𝗒𝗇𝗈𝗉𝗌𝗂𝗌:** [𝖢𝗅𝗂𝖼𝗄 𝖧𝖾𝗋𝖾]({description})
|
84 |
+
╰────────────────•
|
85 |
+
"""
|
86 |
+
]
|
87 |
+
|
88 |
+
CHARACTER_TEMPLATES = [
|
89 |
+
"""
|
90 |
+
{name}
|
91 |
+
|
92 |
+
╭────────────────•
|
93 |
+
╰➢ **𝖦𝖾𝗇𝖽𝖾𝗋:** `{gender}`
|
94 |
+
╰➢ **𝖣𝖺𝗍𝖾 𝗈𝖿 𝖡𝗂𝗋𝗍𝗁:** `{date_of_birth}`
|
95 |
+
╰➢ **𝖠𝗀𝖾:** `{age}`
|
96 |
+
╰➢ **𝖡𝗅𝗈𝗈𝖽 𝖳𝗒𝗉𝖾:** `{blood_type}`
|
97 |
+
╰➢ **𝖥𝖺𝗏𝗈𝗎𝗋𝗂𝗍𝖾𝗌:** `{favorites}`
|
98 |
+
╰➢ **𝖶𝖾𝖻𝗌𝗂𝗍𝖾:** {siteurl}{role_in}
|
99 |
+
╰────────────────•
|
100 |
+
{description}
|
101 |
+
"""
|
102 |
+
]
|
103 |
+
|
104 |
+
AIRING_TEMPLATES = [
|
105 |
+
"""
|
106 |
+
{name}
|
107 |
+
|
108 |
+
╭────────────────•
|
109 |
+
╰➢ **𝖲𝗍𝖺𝗍𝗎𝗌:** `{status}`
|
110 |
+
╰➢ **𝖤𝗉𝗂𝗌𝗈𝖽𝖾:** `{episode}`
|
111 |
+
╰────────────────•{airing_info}
|
112 |
+
"""
|
113 |
+
]
|
114 |
+
|
115 |
+
ANILIST_USER_TEMPLATES = [
|
116 |
+
"""
|
117 |
+
**💫 {name}**
|
118 |
+
|
119 |
+
╭──── 𝖠𝗇𝗂𝗆𝖾 ─────•
|
120 |
+
╰➢ **𝖢𝗈𝗎𝗇𝗍:** `{anime_count}`
|
121 |
+
╰➢ **𝖲𝖼𝗈𝗋𝖾:** `{anime_score}`
|
122 |
+
╰➢ **𝖬𝗂𝗇𝗎𝗍𝖾𝗌 𝖲𝗉𝖾𝗇𝗍:** `{minutes}`
|
123 |
+
╰➢ **𝖤𝗉𝗂𝗌𝗈𝖽𝖾𝗌 𝖶𝖺𝗍𝖼𝗁𝖾𝖽:** `{episodes}`
|
124 |
+
╰────────────────•
|
125 |
+
╭──── 𝖬𝖺𝗇𝗀𝖺 ─────•
|
126 |
+
╰➢ **𝖢𝗈𝗎𝗇𝗍:** `{manga_count}`
|
127 |
+
╰➢ **𝖲𝖼𝗈𝗋𝖾:** `{manga_score}`
|
128 |
+
╰➢ **𝖢𝗁𝖺𝗉𝗍𝖾𝗋𝗌:** `{chapters}`
|
129 |
+
╰➢ **𝖵𝗈𝗅𝗎𝗆𝖾𝗌:** `{volumes}`
|
130 |
+
╰────────────────•
|
131 |
+
|
132 |
+
𝖶𝖾𝖻𝗌𝗂𝗍𝖾: {siteurl}
|
133 |
+
"""
|
134 |
+
]
|
135 |
+
|
136 |
+
CLIMATE_TEMPLATES = [
|
137 |
+
"""
|
138 |
+
🌆 {city_name}, {country}
|
139 |
+
|
140 |
+
╭────────────────•
|
141 |
+
╰➢ **𝖶𝖾𝖺𝗍𝗁𝖾𝗋:** {weather}
|
142 |
+
╰➢ **𝖳𝗂𝗆𝖾𝗓𝗈𝗇𝖾:** {timezone}
|
143 |
+
╰➢ **𝖲𝗎𝗇𝗋𝗂𝗌𝖾:** {sunrise}
|
144 |
+
╰➢ **𝖲𝗎𝗇𝗌𝖾𝗍:** {sunset}
|
145 |
+
╰➢ **𝖶𝗂𝗇𝖽:** {wind}
|
146 |
+
╰➢ **𝖳𝖾𝗆𝗉𝖾𝗋𝖺𝗍𝗎𝗋𝖾:** {temperature}°C
|
147 |
+
╰➢ **𝖥𝖾𝖾𝗅𝗌 𝗅𝗂𝗄𝖾:** {feels_like}°C
|
148 |
+
╰➢ **𝖬𝗂𝗇𝗂𝗆𝗎𝗆:** {temp_min}°C
|
149 |
+
╰➢ **𝖬𝖺𝗑𝗂𝗆𝗎𝗆:** {temp_max}°C
|
150 |
+
╰➢ **𝖯𝗋𝖾𝗌𝗌𝗎𝗋𝖾:** {pressure} hPa
|
151 |
+
╰➢ **𝖧𝗎𝗆𝗂𝖽𝗂𝗍𝗒:** {humidity}%
|
152 |
+
╰➢ **𝖵𝗂𝗌𝗂𝖻𝗂𝗅𝗂𝗍𝗒:** {visibility} m
|
153 |
+
╰➢ **𝖢𝗅𝗈𝗎𝖽𝗌:** {clouds}%
|
154 |
+
╰────────────────•
|
155 |
+
"""
|
156 |
+
]
|
157 |
+
|
158 |
+
AIR_POLLUTION_TEMPLATES = [
|
159 |
+
"""
|
160 |
+
🌆 {city_name}
|
161 |
+
|
162 |
+
╭────────────────•
|
163 |
+
╰➢ **𝖠𝖰𝖨:** {aqi}
|
164 |
+
╰➢ **𝖢𝖺𝗋𝖻𝗈𝗇 𝖬𝗈𝗇𝗈𝗑𝗂𝖽𝖾:** {co}
|
165 |
+
╰➢ **𝖭𝗈𝗂𝗍𝗋𝗈𝗀𝖾𝗇 𝖬𝗈𝗇𝗈𝗑𝗂𝖽𝖾:** {no}
|
166 |
+
╰➢ **𝖭𝗂𝗍𝗋𝗈𝗀𝖾𝗇 𝖣𝗂𝗈𝗑𝗂𝖽𝖾:** {no2}
|
167 |
+
╰➢ **𝖮𝗓𝗈𝗇𝖾:** {o3}
|
168 |
+
╰➢ **𝖲𝗎𝗅𝗉𝗁𝗎𝗋 𝖣𝗂𝗈𝗑𝗂𝖽𝖾:** {so2}
|
169 |
+
╰➢ **𝖠𝗆𝗆𝗈𝗇𝗂𝖺:** {nh3}
|
170 |
+
╰➢ **𝖥𝗂𝗇𝖾 𝖯𝖺𝗋𝗍𝗂𝖼𝗅𝖾𝗌 (PM{sub2_5}):** {pm2_5}
|
171 |
+
╰➢ **𝖢𝗈𝖺𝗋𝗌𝖾 𝖯𝖺𝗋𝗍𝗂𝖼𝗅𝖾𝗌 (PM{sub10}):** {pm10}
|
172 |
+
╰────────────────•
|
173 |
+
"""
|
174 |
+
]
|
175 |
+
|
176 |
+
GITHUB_USER_TEMPLATES = [
|
177 |
+
"""
|
178 |
+
🍀 {username} ({git_id})
|
179 |
+
|
180 |
+
╭──────── {id_type} ────────•
|
181 |
+
╰➢ **𝖭𝖺𝗆𝖾:** [{name}]({profile_url})
|
182 |
+
╰➢ **𝖡𝗅𝗈𝗀:** {blog}
|
183 |
+
╰➢ **𝖢𝗈𝗆𝗉𝖺𝗇𝗒:** {company}
|
184 |
+
╰➢ **𝖤𝗆𝖺𝗂𝗅:** {email}
|
185 |
+
╰➢ **𝖫𝗈𝖼𝖺𝗍𝗂𝗈𝗇:** {location}
|
186 |
+
╰➢ **𝖱𝖾𝗉𝗈:** {public_repos}
|
187 |
+
╰➢ **𝖦𝗂𝗌𝗍𝗌:** {public_gists}
|
188 |
+
╰➢ **𝖥𝗈𝗅𝗅𝗈𝗐𝖾𝗋𝗌:** {followers}
|
189 |
+
╰➢ **𝖥𝗈𝗅𝗅𝗈𝗐𝗂𝗇𝗀:** {following}
|
190 |
+
╰➢ **𝖠𝖼𝖼𝗈𝗎𝗇𝗍 𝖼𝗋𝖾𝖺𝗍𝖾𝖽:** {created_at}
|
191 |
+
╰────────────────•
|
192 |
+
|
193 |
+
**💫 𝖡𝗂𝗈:** {bio}
|
194 |
+
"""
|
195 |
+
]
|
196 |
+
|
197 |
+
STATISTICS_TEMPLATES = [
|
198 |
+
"""
|
199 |
+
🍀 {name}
|
200 |
+
|
201 |
+
╭──────── 𝖢𝗁𝖺𝗇𝗇𝖾𝗅𝗌 ────────•
|
202 |
+
╰➢ **𝖳𝗈𝗍𝖺𝗅:** `{channels}`
|
203 |
+
╰➢ **𝖠𝖽𝗆𝗂𝗇:** `{ch_admin}`
|
204 |
+
╰➢ **𝖮𝗐𝗇𝖾𝗋:** `{ch_owner}`
|
205 |
+
|
206 |
+
╭──────── 𝖦𝗋𝗈𝗎𝗉𝗌 ────────•
|
207 |
+
╰➢ **𝖳𝗈𝗍𝖺𝗅:** `{groups}`
|
208 |
+
╰➢ **𝖠𝖽𝗆𝗂𝗇:** `{gc_admin}`
|
209 |
+
╰➢ **𝖮𝗐𝗇𝖾𝗋:** `{gc_owner}`
|
210 |
+
|
211 |
+
╭──────── 𝖮𝗍𝗁𝖾𝗋𝗌 ────────•
|
212 |
+
╰➢ **𝖯𝗋𝗂𝗏𝖺𝗍𝖾:** `{users}`
|
213 |
+
╰➢ **𝖡𝗈𝗍𝗌:** `{bots}`
|
214 |
+
╰➢ **𝖴𝗇𝗋𝖾𝖺𝖽 𝖬𝖾𝗌𝗌𝖺𝗀𝖾𝗌:** `{unread_msg}`
|
215 |
+
╰➢ **𝖴𝗇𝗋𝖾𝖺𝖽 𝖬𝖾𝗇𝗍𝗂𝗈𝗇𝗌:** `{unread_mention}`
|
216 |
+
|
217 |
+
⌛ **𝖳𝗂𝗆𝖾 𝖳𝖺𝗄𝖾𝗇:** `{time_taken}`
|
218 |
+
"""
|
219 |
+
]
|
220 |
+
|
221 |
+
GBAN_TEMPLATES = [
|
222 |
+
"""
|
223 |
+
╭──────── {gtype} ────────•
|
224 |
+
╰➢ **𝖵𝗂𝖼𝗍𝗂𝗆:** {name}
|
225 |
+
╰➢ **𝖲𝗎𝖼𝖼𝖾𝗌𝗌:** {success}
|
226 |
+
╰➢ **𝖥𝖺𝗂𝗅𝖾𝖽:** {failed}
|
227 |
+
╰➢ **𝖱𝖾𝖺𝗌𝗈𝗇:** {reason}
|
228 |
+
╰────────────────•
|
229 |
+
"""
|
230 |
+
]
|
231 |
+
|
232 |
+
USAGE_TEMPLATES = [
|
233 |
+
"""
|
234 |
+
**📝 𝖣𝗂𝗌𝗄 & 𝖣𝗒𝗇𝗈 𝖴𝗌𝖺𝗀𝖾:**
|
235 |
+
|
236 |
+
**➢ 𝖣𝗒𝗇𝗈 𝖴𝗌𝖺𝗀𝖾 𝖿𝗈𝗋** `{appName}`
|
237 |
+
◈ __{appHours}hrs {appMinutes}mins__ | __{appPercentage}%__
|
238 |
+
|
239 |
+
**➢ 𝖣𝗒𝗇𝗈 𝗋𝖾𝗆𝖺𝗂𝗇𝗂𝗇𝗀 𝗍𝗁𝗂𝗌 𝗆𝗈𝗇𝗍𝗁:**
|
240 |
+
◈ __{hours}hrs {minutes}mins__ | __{percentage}%__
|
241 |
+
|
242 |
+
**➢ 𝖣𝗂𝗌𝗄 𝖴𝗌𝖺𝗀𝖾:**
|
243 |
+
◈ __{diskUsed}GB__ / __{diskTotal}GB__ | __{diskPercent}%__
|
244 |
+
|
245 |
+
**➢ 𝖬𝖾𝗆𝗈𝗋𝗒 𝖴𝗌𝖺𝗀𝖾:**
|
246 |
+
◈ __{memoryUsed}GB__ / __{memoryTotal}GB__ | __{memoryPercent}%__
|
247 |
+
"""
|
248 |
+
]
|
249 |
+
|
250 |
+
USER_INFO_TEMPLATES = [
|
251 |
+
"""
|
252 |
+
**🍀 𝖴𝗌𝖾𝗋 𝖨𝗇𝖿𝗈 𝗈𝖿 {mention}:**
|
253 |
+
|
254 |
+
**➢ 𝖥𝗂𝗋𝗌𝗍 𝖭𝖺𝗆𝖾:** `{firstName}`
|
255 |
+
**➢ 𝖫𝖺𝗌𝗍 𝖭𝖺𝗆𝖾:** `{lastName}`
|
256 |
+
**➢ 𝖴𝗌𝖾𝗋𝖨𝖣:** `{userId}`
|
257 |
+
|
258 |
+
**➢ 𝖢𝗈𝗆𝗆𝗈𝗇 𝖦𝗋𝗈𝗎𝗉𝗌:** `{commonGroups}`
|
259 |
+
**➢ 𝖣𝖢-𝖨𝖣:** `{dcId}`
|
260 |
+
**➢ 𝖯𝗂𝖼𝗍𝗎𝗋𝖾𝗌:** `{totalPictures}`
|
261 |
+
**➢ 𝖱𝖾𝗌𝗍𝗋𝗂𝖼𝗍𝖾𝖽:** `{isRestricted}`
|
262 |
+
**➢ 𝖵𝖾𝗋𝗂𝖿𝗂𝖾𝖽:** `{isVerified}`
|
263 |
+
**➢ 𝖡𝗈𝗍:** `{isBot}`
|
264 |
+
**➢ 𝖡𝗂𝗈:** `{bio}`
|
265 |
+
|
266 |
+
**</> @HellBot_Networks**
|
267 |
+
"""
|
268 |
+
]
|
269 |
+
|
270 |
+
CHAT_INFO_TEMPLATES = [
|
271 |
+
"""
|
272 |
+
**🍀 𝖢𝗁𝖺𝗍 𝖨𝗇𝖿𝗈:**
|
273 |
+
|
274 |
+
**➢ 𝖢𝗁𝖺𝗍 𝖭𝖺𝗆𝖾:** `{chatName}`
|
275 |
+
**➢ 𝖢𝗁𝖺𝗍 𝖨𝖣:** `{chatId}`
|
276 |
+
**➢ 𝖢𝗁𝖺𝗍 𝖫𝗂𝗇𝗄:** {chatLink}
|
277 |
+
**➢ 𝖮𝗐𝗇𝖾𝗋:** {chatOwner}
|
278 |
+
**➢ 𝖣𝖢-𝖨𝖣:** `{dcId}`
|
279 |
+
**➢ 𝖬𝖾𝗆𝖻𝖾𝗋𝗌:** `{membersCount}`
|
280 |
+
**➢ 𝖠𝖽𝗆𝗂𝗇𝗌:** `{adminsCount}`
|
281 |
+
**➢ 𝖡𝗈𝗍𝗌:** `{botsCount}`
|
282 |
+
**➢ 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** `{description}`
|
283 |
+
|
284 |
+
**</> @HellBot_Networks**
|
285 |
+
"""
|
286 |
+
]
|
287 |
+
|
288 |
+
|
289 |
+
async def alive_template(owner: str, uptime: str) -> str:
|
290 |
+
template = await db.get_env(ENV.alive_template)
|
291 |
+
if template:
|
292 |
+
message = template
|
293 |
+
else:
|
294 |
+
message = random.choice(ALIVE_TEMPLATES)
|
295 |
+
return message.format(
|
296 |
+
owner=owner,
|
297 |
+
pyrogram=__version__["pyrogram"],
|
298 |
+
hellbot=__version__["hellbot"],
|
299 |
+
python=__version__["python"],
|
300 |
+
uptime=uptime,
|
301 |
+
)
|
302 |
+
|
303 |
+
|
304 |
+
async def ping_template(speed: float, uptime: str, owner: str) -> str:
|
305 |
+
template = await db.get_env(ENV.ping_template)
|
306 |
+
if template:
|
307 |
+
message = template
|
308 |
+
else:
|
309 |
+
message = random.choice(PING_TEMPLATES)
|
310 |
+
return message.format(speed=speed, uptime=uptime, owner=owner)
|
311 |
+
|
312 |
+
|
313 |
+
async def help_template(
|
314 |
+
owner: str, cmd_n_plgn: tuple[int, int], page: tuple[int, int]
|
315 |
+
) -> str:
|
316 |
+
template = await db.get_env(ENV.help_template)
|
317 |
+
if template:
|
318 |
+
message = template
|
319 |
+
else:
|
320 |
+
message = random.choice(HELP_MENU_TEMPLATES)
|
321 |
+
return message.format(
|
322 |
+
owner=owner,
|
323 |
+
commands=cmd_n_plgn[0],
|
324 |
+
plugins=cmd_n_plgn[1],
|
325 |
+
current=page[0],
|
326 |
+
last=page[1],
|
327 |
+
)
|
328 |
+
|
329 |
+
|
330 |
+
async def command_template(file: str, info: str, commands: str) -> str:
|
331 |
+
template = await db.get_env(ENV.command_template)
|
332 |
+
if template:
|
333 |
+
message = template
|
334 |
+
else:
|
335 |
+
message = random.choice(COMMAND_MENU_TEMPLATES)
|
336 |
+
return message.format(file=file, info=info, commands=commands)
|
337 |
+
|
338 |
+
|
339 |
+
async def anime_template(**kwargs) -> str:
|
340 |
+
template = await db.get_env(ENV.anime_template)
|
341 |
+
if template:
|
342 |
+
message = template
|
343 |
+
else:
|
344 |
+
message = random.choice(ANIME_TEMPLATES)
|
345 |
+
return message.format(**kwargs)
|
346 |
+
|
347 |
+
|
348 |
+
async def manga_templates(**kwargs) -> str:
|
349 |
+
template = await db.get_env(ENV.manga_template)
|
350 |
+
if template:
|
351 |
+
message = template
|
352 |
+
else:
|
353 |
+
message = random.choice(MANGA_TEMPLATES)
|
354 |
+
return message.format(**kwargs)
|
355 |
+
|
356 |
+
|
357 |
+
async def character_templates(**kwargs) -> str:
|
358 |
+
template = await db.get_env(ENV.character_template)
|
359 |
+
if template:
|
360 |
+
message = template
|
361 |
+
else:
|
362 |
+
message = random.choice(CHARACTER_TEMPLATES)
|
363 |
+
return message.format(**kwargs)
|
364 |
+
|
365 |
+
|
366 |
+
async def airing_templates(**kwargs) -> str:
|
367 |
+
template = await db.get_env(ENV.airing_template)
|
368 |
+
if template:
|
369 |
+
message = template
|
370 |
+
else:
|
371 |
+
message = random.choice(AIRING_TEMPLATES)
|
372 |
+
return message.format(**kwargs)
|
373 |
+
|
374 |
+
|
375 |
+
async def anilist_user_templates(
|
376 |
+
name: str, anime: tuple, manga: tuple, siteurl: str
|
377 |
+
) -> str:
|
378 |
+
template = await db.get_env(ENV.anilist_user_template)
|
379 |
+
if template:
|
380 |
+
message = template
|
381 |
+
else:
|
382 |
+
message = random.choice(ANILIST_USER_TEMPLATES)
|
383 |
+
return message.format(
|
384 |
+
name=name,
|
385 |
+
anime_count=anime[0],
|
386 |
+
anime_score=anime[1],
|
387 |
+
minutes=anime[2],
|
388 |
+
episodes=anime[3],
|
389 |
+
manga_count=manga[0],
|
390 |
+
manga_score=manga[1],
|
391 |
+
chapters=manga[2],
|
392 |
+
volumes=manga[3],
|
393 |
+
siteurl=siteurl,
|
394 |
+
)
|
395 |
+
|
396 |
+
|
397 |
+
async def climate_templates(**kwargs) -> str:
|
398 |
+
template = await db.get_env(ENV.climate_template)
|
399 |
+
if template:
|
400 |
+
message = template
|
401 |
+
else:
|
402 |
+
message = random.choice(CLIMATE_TEMPLATES)
|
403 |
+
return message.format(**kwargs)
|
404 |
+
|
405 |
+
|
406 |
+
async def airpollution_templates(**kwargs) -> str:
|
407 |
+
template = await db.get_env(ENV.airpollution_template)
|
408 |
+
if template:
|
409 |
+
message = template
|
410 |
+
else:
|
411 |
+
message = random.choice(AIR_POLLUTION_TEMPLATES)
|
412 |
+
return message.format(**kwargs)
|
413 |
+
|
414 |
+
|
415 |
+
async def statistics_templates(**kwargs) -> str:
|
416 |
+
template = await db.get_env(ENV.statistics_template)
|
417 |
+
if template:
|
418 |
+
message = template
|
419 |
+
else:
|
420 |
+
message = random.choice(STATISTICS_TEMPLATES)
|
421 |
+
return message.format(**kwargs)
|
422 |
+
|
423 |
+
|
424 |
+
async def github_user_templates(**kwargs) -> str:
|
425 |
+
template = await db.get_env(ENV.github_user_template)
|
426 |
+
if template:
|
427 |
+
message = template
|
428 |
+
else:
|
429 |
+
message = random.choice(GITHUB_USER_TEMPLATES)
|
430 |
+
return message.format(**kwargs)
|
431 |
+
|
432 |
+
|
433 |
+
async def gban_templates(**kwargs) -> str:
|
434 |
+
template = await db.get_env(ENV.gban_template)
|
435 |
+
if template:
|
436 |
+
message = template
|
437 |
+
else:
|
438 |
+
message = random.choice(GBAN_TEMPLATES)
|
439 |
+
return message.format(**kwargs)
|
440 |
+
|
441 |
+
|
442 |
+
async def usage_templates(**kwargs) -> str:
|
443 |
+
template = await db.get_env(ENV.usage_template)
|
444 |
+
if template:
|
445 |
+
message = template
|
446 |
+
else:
|
447 |
+
message = random.choice(USAGE_TEMPLATES)
|
448 |
+
return message.format(**kwargs)
|
449 |
+
|
450 |
+
|
451 |
+
async def user_info_templates(**kwargs) -> str:
|
452 |
+
template = await db.get_env(ENV.user_info_template)
|
453 |
+
if template:
|
454 |
+
message = template
|
455 |
+
else:
|
456 |
+
message = random.choice(USER_INFO_TEMPLATES)
|
457 |
+
return message.format(**kwargs)
|
458 |
+
|
459 |
+
|
460 |
+
async def chat_info_templates(**kwargs) -> str:
|
461 |
+
template = await db.get_env(ENV.chat_info_template)
|
462 |
+
if template:
|
463 |
+
message = template
|
464 |
+
else:
|
465 |
+
message = random.choice(CHAT_INFO_TEMPLATES)
|
466 |
+
return message.format(**kwargs)
|
HellBot/functions/tools.py
ADDED
@@ -0,0 +1,139 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import contextlib
|
3 |
+
import math
|
4 |
+
import os
|
5 |
+
import shlex
|
6 |
+
import shutil
|
7 |
+
import time
|
8 |
+
|
9 |
+
from git import Repo
|
10 |
+
from git.exc import GitCommandError, InvalidGitRepositoryError, NoSuchPathError
|
11 |
+
from pyrogram.types import Message
|
12 |
+
|
13 |
+
from Hellbot.core import Config, Symbols
|
14 |
+
|
15 |
+
from .formatter import humanbytes, readable_time
|
16 |
+
|
17 |
+
|
18 |
+
async def progress(
|
19 |
+
current: int, total: int, message: Message, start: float, process: str
|
20 |
+
):
|
21 |
+
now = time.time()
|
22 |
+
diff = now - start
|
23 |
+
if round(diff % 10.00) == 0 or current == total:
|
24 |
+
percentage = current * 100 / total
|
25 |
+
speed = current / diff
|
26 |
+
elapsed_time = round(diff) * 1000
|
27 |
+
complete_time = round((total - current) / speed) * 1000
|
28 |
+
estimated_total_time = elapsed_time + complete_time
|
29 |
+
progress_str = "**[{0}{1}] : {2}%\n**".format(
|
30 |
+
"".join(["●" for i in range(math.floor(percentage / 10))]),
|
31 |
+
"".join(["○" for i in range(10 - math.floor(percentage / 10))]),
|
32 |
+
round(percentage, 2),
|
33 |
+
)
|
34 |
+
msg = (
|
35 |
+
progress_str
|
36 |
+
+ "__{0}__ **𝗈𝖿** __{1}__\n**𝖲𝗉𝖾𝖾𝖽:** __{2}/s__\n**𝖤𝖳𝖠:** __{3}__".format(
|
37 |
+
humanbytes(current),
|
38 |
+
humanbytes(total),
|
39 |
+
humanbytes(speed),
|
40 |
+
readable_time(estimated_total_time / 1000),
|
41 |
+
)
|
42 |
+
)
|
43 |
+
await message.edit_text(f"**{process} ...**\n\n{msg}")
|
44 |
+
|
45 |
+
|
46 |
+
async def get_files_from_directory(directory: str) -> list:
|
47 |
+
all_files = []
|
48 |
+
for path, _, files in os.walk(directory):
|
49 |
+
for file in files:
|
50 |
+
all_files.append(os.path.join(path, file))
|
51 |
+
return all_files
|
52 |
+
|
53 |
+
|
54 |
+
async def runcmd(cmd: str) -> tuple[str, str, int, int]:
|
55 |
+
args = shlex.split(cmd)
|
56 |
+
process = await asyncio.create_subprocess_exec(
|
57 |
+
*args, stdout=asyncio.subprocess.PIPE, stderr=asyncio.subprocess.PIPE
|
58 |
+
)
|
59 |
+
stdout, stderr = await process.communicate()
|
60 |
+
return (
|
61 |
+
stdout.decode("utf-8", "replace").strip(),
|
62 |
+
stderr.decode("utf-8", "replace").strip(),
|
63 |
+
process.returncode,
|
64 |
+
process.pid,
|
65 |
+
)
|
66 |
+
|
67 |
+
|
68 |
+
async def update_dotenv(key: str, value: str) -> None:
|
69 |
+
with open(".env", "r") as file:
|
70 |
+
data = file.readlines()
|
71 |
+
|
72 |
+
for index, line in enumerate(data):
|
73 |
+
if line.startswith(f"{key}="):
|
74 |
+
data[index] = f"{key}={value}\n"
|
75 |
+
break
|
76 |
+
|
77 |
+
with open(".env", "w") as file:
|
78 |
+
file.writelines(data)
|
79 |
+
|
80 |
+
|
81 |
+
async def restart(
|
82 |
+
update: bool = False,
|
83 |
+
clean_up: bool = False,
|
84 |
+
shutdown: bool = False,
|
85 |
+
):
|
86 |
+
try:
|
87 |
+
shutil.rmtree(Config.DWL_DIR)
|
88 |
+
shutil.rmtree(Config.TEMP_DIR)
|
89 |
+
except BaseException:
|
90 |
+
pass
|
91 |
+
|
92 |
+
if clean_up:
|
93 |
+
os.system(f"mkdir {Config.DWL_DIR}")
|
94 |
+
os.system(f"mkdir {Config.TEMP_DIR}")
|
95 |
+
return
|
96 |
+
|
97 |
+
if shutdown:
|
98 |
+
return os.system(f"kill -9 {os.getpid()}")
|
99 |
+
|
100 |
+
cmd = (
|
101 |
+
"git pull && pip3 install -U -r requirements.txt && bash start.sh"
|
102 |
+
if update
|
103 |
+
else "bash start.sh"
|
104 |
+
)
|
105 |
+
|
106 |
+
os.system(f"kill -9 {os.getpid()} && {cmd}")
|
107 |
+
|
108 |
+
|
109 |
+
async def gen_changelogs(repo: Repo, branch: str) -> str:
|
110 |
+
changelogs = ""
|
111 |
+
commits = list(repo.iter_commits(branch))[:5]
|
112 |
+
for index, commit in enumerate(commits):
|
113 |
+
changelogs += f"**{Symbols.triangle_right} {index + 1}.** `{commit.summary}`\n"
|
114 |
+
|
115 |
+
return changelogs
|
116 |
+
|
117 |
+
|
118 |
+
async def initialize_git(git_repo: str):
|
119 |
+
force = False
|
120 |
+
try:
|
121 |
+
repo = Repo()
|
122 |
+
except NoSuchPathError as pathErr:
|
123 |
+
repo.__del__()
|
124 |
+
return False, pathErr, force
|
125 |
+
except GitCommandError as gitErr:
|
126 |
+
repo.__del__()
|
127 |
+
return False, gitErr, force
|
128 |
+
except InvalidGitRepositoryError:
|
129 |
+
repo = Repo.init()
|
130 |
+
origin = repo.create_remote("upstream", f"https://github.com/{git_repo}")
|
131 |
+
origin.fetch()
|
132 |
+
repo.create_head("master", origin.refs.master)
|
133 |
+
repo.heads.master.set_tracking_branch(origin.refs.master)
|
134 |
+
repo.heads.master.checkout(True)
|
135 |
+
force = True
|
136 |
+
with contextlib.suppress(BaseException):
|
137 |
+
repo.create_remote("upstream", f"https://github.com/{git_repo}")
|
138 |
+
|
139 |
+
return True, repo, force
|
HellBot/functions/utility.py
ADDED
@@ -0,0 +1,241 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import os
|
3 |
+
import time
|
4 |
+
|
5 |
+
from pyrogram import Client
|
6 |
+
from pyrogram.enums import ChatType
|
7 |
+
from pyrogram.errors import FloodWait
|
8 |
+
from pyrogram.types import Message
|
9 |
+
from telegraph import Telegraph
|
10 |
+
|
11 |
+
from Hellbot.core import ENV, LOGS, db
|
12 |
+
|
13 |
+
from .formatter import readable_time
|
14 |
+
|
15 |
+
|
16 |
+
class TelegraphAPI:
|
17 |
+
def __init__(self) -> None:
|
18 |
+
self.shortname: str = "TheHellbot"
|
19 |
+
self.telegraph: Telegraph = None
|
20 |
+
|
21 |
+
async def setup(self):
|
22 |
+
shortname = await db.get_env(ENV.telegraph_account) or self.shortname
|
23 |
+
|
24 |
+
try:
|
25 |
+
self.telegraph = Telegraph(domain="telegra.ph")
|
26 |
+
self.telegraph.create_account(shortname)
|
27 |
+
except:
|
28 |
+
LOGS.warning("Failed to setup Telegraph API")
|
29 |
+
|
30 |
+
|
31 |
+
class Gcast:
|
32 |
+
def __init__(self) -> None:
|
33 |
+
self.file_name = "gcast_{0}.txt"
|
34 |
+
self.complete_msg = "**🍀 𝖦𝖼𝖺𝗌𝗍 𝖢𝗈𝗆𝗉𝗅𝖾𝗍𝖾𝖽!** \n\n**𝖬𝖾𝗌𝗌𝖺𝗀𝖾:** [click here]({0})\n**𝖢𝗈𝗎𝗇𝗍:** `{1} {2}`\n**𝖥𝗈𝗋𝗐𝖺𝗋𝖽 𝗍𝖺𝗀:** `{3}`\n**𝖳𝗂𝗆𝖾 𝗍𝖺𝗄𝖾𝗇:** `{4}`"
|
35 |
+
|
36 |
+
async def _send_msg(self, chat_id: int, msg: Message, tag: bool):
|
37 |
+
await msg.forward(chat_id) if tag else await msg.copy(chat_id)
|
38 |
+
|
39 |
+
async def start(self, message: Message, client: Client, mode: str, tag: bool):
|
40 |
+
link = message.link
|
41 |
+
status = "Enabled" if tag else "Removed"
|
42 |
+
start = time.time()
|
43 |
+
|
44 |
+
if mode == "all":
|
45 |
+
uCount, uFileName = await self.users(message, client, tag)
|
46 |
+
gCount, gFileName = await self.groups(message, client, tag)
|
47 |
+
count = uCount + gCount
|
48 |
+
with open(uFileName, "a", encoding="utf-8") as file1, open(
|
49 |
+
gFileName, "r", encoding="utf-8"
|
50 |
+
) as file2:
|
51 |
+
file1.write(file2.read())
|
52 |
+
file2.close()
|
53 |
+
file1.close()
|
54 |
+
os.remove(gFileName)
|
55 |
+
fileName = uFileName
|
56 |
+
elif mode == "groups":
|
57 |
+
count, fileName = await self.groups(message, client, tag)
|
58 |
+
elif mode == "users":
|
59 |
+
count, fileName = await self.users(message, client, tag)
|
60 |
+
else:
|
61 |
+
return None
|
62 |
+
|
63 |
+
end = time.time()
|
64 |
+
outStr = self.complete_msg.format(
|
65 |
+
link, count, mode, status, readable_time(int(end - start))
|
66 |
+
)
|
67 |
+
|
68 |
+
return fileName, outStr
|
69 |
+
|
70 |
+
async def groups(self, message: Message, client: Client, tag: bool):
|
71 |
+
filename = self.file_name.format(round(time.time()))
|
72 |
+
count = 0
|
73 |
+
|
74 |
+
with open(filename, "w", encoding="utf-8") as f:
|
75 |
+
f.write("Group ID | Error\n\n")
|
76 |
+
async for dialog in client.get_dialogs():
|
77 |
+
if dialog.chat.type == ChatType.SUPERGROUP:
|
78 |
+
try:
|
79 |
+
await self._send_msg(dialog.chat.id, message, tag)
|
80 |
+
count += 1
|
81 |
+
except FloodWait as fw:
|
82 |
+
await asyncio.sleep(fw.value)
|
83 |
+
await self._send_msg(dialog.chat.id, message, tag)
|
84 |
+
count += 1
|
85 |
+
except Exception as e:
|
86 |
+
f.write(f"{dialog.chat.id} | {e}\n")
|
87 |
+
|
88 |
+
f.close()
|
89 |
+
|
90 |
+
return count, filename
|
91 |
+
|
92 |
+
async def users(self, message: Message, client: Client, tag: bool):
|
93 |
+
filename = self.file_name.format(round(time.time()))
|
94 |
+
count = 0
|
95 |
+
|
96 |
+
with open(filename, "w", encoding="utf-8") as f:
|
97 |
+
f.write("User ID | Error\n\n")
|
98 |
+
async for dialog in client.get_dialogs():
|
99 |
+
if dialog.chat.type == ChatType.PRIVATE:
|
100 |
+
try:
|
101 |
+
await self._send_msg(dialog.chat.id, message, tag)
|
102 |
+
count += 1
|
103 |
+
except FloodWait as fw:
|
104 |
+
await asyncio.sleep(fw.value)
|
105 |
+
await self._send_msg(dialog.chat.id, message, tag)
|
106 |
+
count += 1
|
107 |
+
except Exception as e:
|
108 |
+
f.write(f"{dialog.chat.id} | {e}\n")
|
109 |
+
|
110 |
+
f.close()
|
111 |
+
|
112 |
+
return count, filename
|
113 |
+
|
114 |
+
|
115 |
+
class AntiFlood:
|
116 |
+
def __init__(self) -> None:
|
117 |
+
self.FloodCount = {}
|
118 |
+
self.settings = {}
|
119 |
+
self.client_chats = {}
|
120 |
+
|
121 |
+
def updateSettings(self, client: int, chat: int, data: dict):
|
122 |
+
mode = data.get("mode", "mute")
|
123 |
+
mtime = data.get("time", 0)
|
124 |
+
limit = data.get("limit", 5)
|
125 |
+
|
126 |
+
self.settings[client] = {chat: {"mode": mode, "time": mtime, "limit": limit}}
|
127 |
+
|
128 |
+
def getSettings(self, client: int, chat: int) -> tuple[str, int, int]:
|
129 |
+
mode = "mute"
|
130 |
+
mtime = 0
|
131 |
+
limit = 5
|
132 |
+
|
133 |
+
cli_settings: dict = self.settings.get(client, None)
|
134 |
+
if cli_settings:
|
135 |
+
chat_settings: dict = cli_settings.get(chat, None)
|
136 |
+
if chat_settings:
|
137 |
+
mode = chat_settings.get("mode", "mute")
|
138 |
+
mtime = chat_settings.get("time", 0)
|
139 |
+
limit = chat_settings.get("limit", 5)
|
140 |
+
|
141 |
+
return mode, int(mtime), limit
|
142 |
+
|
143 |
+
def updateFlood(self, client: int, chat: int, user: int, count: int):
|
144 |
+
self.FloodCount[client] = {chat: {"last_user": user, "count": count}}
|
145 |
+
|
146 |
+
def getLastUser(self, client: int, chat: int) -> tuple[int, int]:
|
147 |
+
try:
|
148 |
+
cli_dict: dict = self.FloodCount[client]
|
149 |
+
except KeyError:
|
150 |
+
self.FloodCount[client] = {}
|
151 |
+
cli_dict: dict = self.FloodCount[client]
|
152 |
+
|
153 |
+
try:
|
154 |
+
chat_dict: dict = cli_dict[chat]
|
155 |
+
except KeyError:
|
156 |
+
cli_dict[chat] = {}
|
157 |
+
chat_dict: dict = cli_dict[chat]
|
158 |
+
|
159 |
+
last_user: int = chat_dict.get("last_user", 0)
|
160 |
+
count: int = chat_dict.get("count", 0)
|
161 |
+
|
162 |
+
return last_user, count
|
163 |
+
|
164 |
+
async def updateFromDB(self):
|
165 |
+
floods = await db.get_all_floods()
|
166 |
+
for flood in floods:
|
167 |
+
client = flood["client"]
|
168 |
+
chat = flood["chat"]
|
169 |
+
mode = flood.get("mode", "mute")
|
170 |
+
mtime = flood.get("time", 0)
|
171 |
+
limit = flood.get("limit", 5)
|
172 |
+
settings = {"mode": mode, "time": mtime, "limit": limit}
|
173 |
+
|
174 |
+
self.updateSettings(client, chat, settings)
|
175 |
+
try:
|
176 |
+
self.client_chats[client].append(chat)
|
177 |
+
except KeyError:
|
178 |
+
self.client_chats[client] = [chat]
|
179 |
+
|
180 |
+
def check_client_chat(self, client: int, chat: int) -> bool:
|
181 |
+
try:
|
182 |
+
chats = self.client_chats[client]
|
183 |
+
except KeyError:
|
184 |
+
return False
|
185 |
+
|
186 |
+
if chat in chats:
|
187 |
+
return True
|
188 |
+
|
189 |
+
return False
|
190 |
+
|
191 |
+
|
192 |
+
class Blacklists:
|
193 |
+
def __init__(self) -> None:
|
194 |
+
self.blacklists = {}
|
195 |
+
|
196 |
+
async def updateBlacklists(self):
|
197 |
+
datas = await db.get_blacklist_clients()
|
198 |
+
for data in datas:
|
199 |
+
client = data["client"]
|
200 |
+
chats = data.get("chats", [])
|
201 |
+
for chat in chats:
|
202 |
+
blacklists = data["blacklist"]
|
203 |
+
self.blacklists[client] = {chat: blacklists}
|
204 |
+
|
205 |
+
async def addBlacklist(self, client: int, chat: int, text: str):
|
206 |
+
try:
|
207 |
+
self.blacklists[client][chat].append(text)
|
208 |
+
except KeyError:
|
209 |
+
self.blacklists[client] = {chat: [text]}
|
210 |
+
|
211 |
+
await db.add_blacklist(client, chat, text)
|
212 |
+
|
213 |
+
async def rmBlacklist(self, client: int, chat: int, text: str):
|
214 |
+
try:
|
215 |
+
self.blacklists[client][chat].remove(text)
|
216 |
+
except KeyError:
|
217 |
+
return
|
218 |
+
|
219 |
+
await db.rm_blacklist(client, chat, text)
|
220 |
+
|
221 |
+
def getBlacklists(self, client: int, chat: int) -> list:
|
222 |
+
try:
|
223 |
+
return self.blacklists[client][chat]
|
224 |
+
except KeyError:
|
225 |
+
return []
|
226 |
+
|
227 |
+
def check_client_chat(self, client: int, chat: int) -> bool:
|
228 |
+
try:
|
229 |
+
chats = self.blacklists[client]
|
230 |
+
except KeyError:
|
231 |
+
return False
|
232 |
+
|
233 |
+
if chat in chats:
|
234 |
+
return True
|
235 |
+
|
236 |
+
return False
|
237 |
+
|
238 |
+
|
239 |
+
Flood = AntiFlood()
|
240 |
+
BList = Blacklists()
|
241 |
+
TGraph = TelegraphAPI()
|
HellBot/plugins/__init__.py
ADDED
File without changes
|
HellBot/plugins/bot/__init__.py
ADDED
@@ -0,0 +1,29 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from Hellbot.core.clients import hellbot
|
2 |
+
from Hellbot.core.config import Config, Symbols
|
3 |
+
from Hellbot.core.database import db
|
4 |
+
from Hellbot.plugins.help import BotHelp
|
5 |
+
|
6 |
+
|
7 |
+
START_MSG = """
|
8 |
+
👋 **𝖦𝗋𝖾𝖾𝗍𝗂𝗇𝗀𝗌, {0} - 𝗐𝖺𝗋𝗋𝗂𝗈𝗋𝗌 𝗈𝖿 𝖧𝖾𝗅𝗅𝖻𝗈𝗍!** 👹 𝖨 𝖺𝗆 𝗒𝗈𝗎𝗋 𝗍𝗋𝗎𝗌𝗍𝗒 𝖼𝗈𝗆𝗉𝖺𝗇𝗂𝗈𝗇, 𝗍𝗁𝖾 **𝖧𝖾𝗅𝗅𝖻𝗈𝗍 𝖠𝗌𝗌𝗂𝗌𝗍𝖺𝗇𝗍!** 🚀
|
9 |
+
|
10 |
+
𝖧𝖾𝗋𝖾 𝗍𝗈 𝗌𝖾𝗋𝗏𝖾, 𝗀𝗎𝗂𝖽𝖾, 𝖺𝗇𝖽 💪 𝗎𝗇𝗅𝖾𝖺𝗌𝗁 𝗍𝗁𝖾 𝗉𝗈𝗐𝖾𝗋 𝗈𝖿 𝖧𝖾𝗅𝗅𝖡𝗈𝗍 𝖺𝗍 𝗒𝗈𝗎𝗋 𝖼𝗈𝗆𝗆𝖺𝗇𝖽! 🎉
|
11 |
+
𝖶𝗁𝖾𝗍𝗁𝖾𝗋 𝗂𝗍'𝗌 𝖼𝗋𝖾𝖺𝗍𝗂𝗇𝗀, 𝖽𝖾𝗅𝖾𝗍𝗂𝗇𝗀, 𝗈𝗋 𝗎𝗉𝖽𝖺𝗍𝗂𝗇𝗀 𝗒𝗈𝗎𝗋 𝗎𝗌𝖾𝗋𝖻𝗈𝗍, 𝖨'𝗏𝖾 𝗀𝗈𝗍 𝗒𝗈𝗎𝗋 𝖻𝖺𝖼𝗄.
|
12 |
+
𝖢𝗈𝗇𝗌𝗂𝖽𝖾𝗋 𝗆𝖾 𝗒𝗈𝗎𝗋 𝗉𝖾𝗋𝗌𝗈𝗇𝖺𝗅 𝗌𝗂𝖽𝖾𝗄𝗂𝖼𝗄 🤭 𝗂𝗇 𝗍𝗁𝖾 𝗋𝖾𝖺𝗅𝗆 𝗈𝖿 𝗎𝗅𝗍𝗂𝗆𝖺𝗍𝖾 𝗎𝗌𝖾𝗋𝖻𝗈𝗍 𝗆𝖺𝗌𝗍𝖾𝗋𝗒.
|
13 |
+
|
14 |
+
🍀 𝖫𝖾𝗍'𝗌 𝖾𝗆𝖻𝖺𝗋𝗄 𝗈𝗇 𝗍𝗁𝗂𝗌 𝖾𝗉𝗂𝖼 𝗃𝗈𝗎𝗋𝗇𝖾𝗒 𝗍𝗈𝗀𝖾𝗍𝗁𝖾𝗋!
|
15 |
+
𝖨𝖿 𝗒𝗈𝗎 𝖾𝗏𝖾𝗋 𝗇𝖾𝖾𝖽 𝖺𝗌𝗌𝗂𝗌𝗍𝖺𝗇𝖼𝖾 𝗈𝗋 𝖼𝗋𝖺𝗏𝖾 ✨ 𝗍𝗁𝖾 𝗍𝗁𝗋𝗂𝗅𝗅 𝗈𝖿 𝗎𝗇𝗅𝖾𝖺𝗌𝗁𝗂𝗇𝗀 𝖧𝖾𝗅𝗅𝖻𝗈𝗍'𝗌 𝗆𝗂𝗀𝗁𝗍, 𝗃𝗎𝗌𝗍 𝗌𝗎𝗆𝗆𝗈𝗇 𝗆𝖾.
|
16 |
+
𝖶𝖾'𝗋𝖾 𝖺𝖻𝗈𝗎𝗍 𝗍𝗈 𝖼𝗈𝗇𝗊𝗎𝖾𝗋 𝗇𝖾𝗐 𝗁𝖾𝗂𝗀𝗁𝗍𝗌 🚀 𝗂𝗇 𝗍𝗁𝖾 𝗎𝗌𝖾𝗋𝖻𝗈𝗍 𝗎𝗇𝗂𝗏𝖾𝗋𝗌𝖾!
|
17 |
+
|
18 |
+
💫 𝖬𝖺𝗒 𝗒𝗈𝗎𝗋 𝖼𝗈𝗆𝗆𝖺𝗇𝖽𝗌 𝖻𝖾 𝗌𝗐𝗂𝖿𝗍 𝖺𝗇𝖽 𝗒𝗈𝗎𝗋 𝗌𝖾𝗌𝗌𝗂𝗈𝗇𝗌 𝗅𝖾𝗀𝖾𝗇𝖽𝖺𝗋𝗒.
|
19 |
+
**𝖶𝖾𝗅𝖼𝗈𝗆𝖾 𝗍𝗈 𝖧𝖾𝗅𝗅𝖻𝗈𝗍 𝖠𝗌𝗌𝗂𝗌𝗍𝖺𝗇𝗍 – 𝗐𝗁𝖾𝗋𝖾 𝖧𝖾𝗅𝗅𝖻𝗈𝗍'𝗌 𝗅𝖾𝗀𝖺𝖼𝗒 𝗅𝗂𝗏𝖾𝗌 𝗈𝗇 🤖!**
|
20 |
+
"""
|
21 |
+
|
22 |
+
HELP_MSG = """
|
23 |
+
**⚙️ 𝖧𝖾𝗅𝗉:**
|
24 |
+
|
25 |
+
__» All commands are categorized and you can use these buttons below to navigate each category and get respective commands.__
|
26 |
+
__» Feel free to contact us if you need any help regarding the bot.__
|
27 |
+
|
28 |
+
**❤️ @HellBot_Networks 🇮🇳**
|
29 |
+
"""
|
HellBot/plugins/bot/bot.py
ADDED
@@ -0,0 +1,60 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import heroku3
|
2 |
+
from pyrogram import filters
|
3 |
+
from pyrogram.types import InlineKeyboardMarkup, Message
|
4 |
+
|
5 |
+
from Hellbot import HEROKU_APP
|
6 |
+
from Hellbot.core import LOGS
|
7 |
+
from Hellbot.functions.tools import restart
|
8 |
+
|
9 |
+
from ..btnsG import gen_bot_help_buttons, start_button
|
10 |
+
from . import HELP_MSG, START_MSG, BotHelp, Config, hellbot
|
11 |
+
|
12 |
+
|
13 |
+
@hellbot.bot.on_message(filters.command("start") & Config.AUTH_USERS)
|
14 |
+
async def start_pm(_, message: Message):
|
15 |
+
btns = start_button()
|
16 |
+
|
17 |
+
await message.reply_text(
|
18 |
+
START_MSG.format(message.from_user.mention),
|
19 |
+
disable_web_page_preview=True,
|
20 |
+
reply_markup=InlineKeyboardMarkup(btns),
|
21 |
+
)
|
22 |
+
|
23 |
+
|
24 |
+
@hellbot.bot.on_message(filters.command("help") & Config.AUTH_USERS)
|
25 |
+
async def help_pm(_, message: Message):
|
26 |
+
btns = await gen_bot_help_buttons()
|
27 |
+
|
28 |
+
await message.reply_text(
|
29 |
+
HELP_MSG,
|
30 |
+
disable_web_page_preview=True,
|
31 |
+
reply_markup=InlineKeyboardMarkup(btns),
|
32 |
+
)
|
33 |
+
|
34 |
+
|
35 |
+
@hellbot.bot.on_message(filters.command("restart") & Config.AUTH_USERS)
|
36 |
+
async def restart_clients(_, message: Message):
|
37 |
+
await message.reply_text("Restarted Bot Successfully ✅")
|
38 |
+
try:
|
39 |
+
if HEROKU_APP:
|
40 |
+
try:
|
41 |
+
heroku = heroku3.from_key(Config.HEROKU_APIKEY)
|
42 |
+
app = heroku.apps()[Config.HEROKU_APPNAME]
|
43 |
+
app.restart()
|
44 |
+
except:
|
45 |
+
await restart()
|
46 |
+
else:
|
47 |
+
await restart()
|
48 |
+
except Exception as e:
|
49 |
+
LOGS.error(e)
|
50 |
+
|
51 |
+
|
52 |
+
BotHelp("Others").add(
|
53 |
+
"start", "To start the bot and get the main menu."
|
54 |
+
).add(
|
55 |
+
"help", "To get the help menu with all the command for this assistant bot."
|
56 |
+
).add(
|
57 |
+
"restart", "To restart the bot."
|
58 |
+
).info(
|
59 |
+
"Some basic commands of the bot."
|
60 |
+
).done()
|
HellBot/plugins/bot/callbacks.py
ADDED
@@ -0,0 +1,279 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import filters
|
2 |
+
from pyrogram.enums import ParseMode
|
3 |
+
from pyrogram.types import CallbackQuery, InlineKeyboardButton, InlineKeyboardMarkup
|
4 |
+
|
5 |
+
from Hellbot.functions.templates import command_template, help_template
|
6 |
+
|
7 |
+
from ..btnsG import gen_bot_help_buttons, gen_inline_help_buttons, start_button
|
8 |
+
from . import HELP_MSG, START_MSG, Config, Symbols, hellbot
|
9 |
+
|
10 |
+
|
11 |
+
async def check_auth_click(cb: CallbackQuery) -> bool:
|
12 |
+
if cb.from_user.id not in Config.AUTH_USERS:
|
13 |
+
await cb.answer(
|
14 |
+
"You are not authorized to use this bot. \n\n</> @Its_HellBot",
|
15 |
+
show_alert=True,
|
16 |
+
)
|
17 |
+
return False
|
18 |
+
return True
|
19 |
+
|
20 |
+
|
21 |
+
@hellbot.bot.on_callback_query(filters.regex(r"auth_close"))
|
22 |
+
async def auth_close_cb(_, cb: CallbackQuery):
|
23 |
+
if await check_auth_click(cb):
|
24 |
+
await cb.message.delete()
|
25 |
+
|
26 |
+
|
27 |
+
@hellbot.bot.on_callback_query(filters.regex(r"close"))
|
28 |
+
async def close_cb(_, cb: CallbackQuery):
|
29 |
+
await cb.message.delete()
|
30 |
+
|
31 |
+
|
32 |
+
@hellbot.bot.on_callback_query(filters.regex(r"bot_help_menu"))
|
33 |
+
async def bot_help_menu_cb(_, cb: CallbackQuery):
|
34 |
+
if not await check_auth_click(cb):
|
35 |
+
return
|
36 |
+
|
37 |
+
plugin = str(cb.data.split(":")[1])
|
38 |
+
|
39 |
+
try:
|
40 |
+
buttons = [
|
41 |
+
InlineKeyboardButton(f"{Symbols.bullet} {i}", f"bot_help_cmd:{plugin}:{i}")
|
42 |
+
for i in sorted(Config.BOT_HELP[plugin]["commands"])
|
43 |
+
]
|
44 |
+
except KeyError:
|
45 |
+
await cb.answer("No description provided for this plugin!", show_alert=True)
|
46 |
+
return
|
47 |
+
|
48 |
+
buttons = [buttons[i : i + 2] for i in range(0, len(buttons), 2)]
|
49 |
+
buttons.append([InlineKeyboardButton(Symbols.back, "help_data:bothelp")])
|
50 |
+
|
51 |
+
caption = (
|
52 |
+
f"**𝖯𝗅𝗎𝗀𝗂𝗇 𝖥𝗂𝗅𝖾:** `{plugin}`\n"
|
53 |
+
f"**𝖯𝗅𝗎𝗀𝗂𝗇 𝖨𝗇𝖿𝗈:** __{Config.BOT_HELP[plugin]['info']} 🍀__\n\n"
|
54 |
+
f"**📃 𝖫𝗈𝖺𝖽𝖾𝖽 𝖢𝗈𝗆𝗆𝖺𝗇𝖽𝗌:** `{len(sorted(Config.BOT_HELP[plugin]['commands']))}`"
|
55 |
+
)
|
56 |
+
|
57 |
+
try:
|
58 |
+
await cb.edit_message_text(
|
59 |
+
caption,
|
60 |
+
disable_web_page_preview=True,
|
61 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
62 |
+
)
|
63 |
+
except Exception:
|
64 |
+
# handles MessageNotModified error
|
65 |
+
pass
|
66 |
+
|
67 |
+
|
68 |
+
@hellbot.bot.on_callback_query(filters.regex(r"bot_help_cmd"))
|
69 |
+
async def bot_help_cmd_cb(_, cb: CallbackQuery):
|
70 |
+
if not await check_auth_click(cb):
|
71 |
+
return
|
72 |
+
|
73 |
+
result = ""
|
74 |
+
plugin = str(cb.data.split(":")[1])
|
75 |
+
command = str(cb.data.split(":")[2])
|
76 |
+
cmd_dict = Config.BOT_HELP[plugin]["commands"][command]
|
77 |
+
|
78 |
+
result += f"**{Symbols.radio_select} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `/{cmd_dict['command']}`"
|
79 |
+
result += (
|
80 |
+
f"\n\n**{Symbols.arrow_right} 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** __{cmd_dict['description']}__"
|
81 |
+
)
|
82 |
+
result += f"\n\n**<\\> @Its_HellBot 🍀**"
|
83 |
+
|
84 |
+
buttons = [
|
85 |
+
[
|
86 |
+
InlineKeyboardButton(Symbols.back, f"bot_help_menu:{plugin}"),
|
87 |
+
InlineKeyboardButton(Symbols.close, "help_data:botclose"),
|
88 |
+
]
|
89 |
+
]
|
90 |
+
|
91 |
+
try:
|
92 |
+
await cb.edit_message_text(
|
93 |
+
result,
|
94 |
+
ParseMode.MARKDOWN,
|
95 |
+
True,
|
96 |
+
InlineKeyboardMarkup(buttons),
|
97 |
+
)
|
98 |
+
except Exception:
|
99 |
+
# handles MessageNotModified error
|
100 |
+
pass
|
101 |
+
|
102 |
+
|
103 |
+
@hellbot.bot.on_callback_query(filters.regex(r"help_page"))
|
104 |
+
async def help_page_cb(_, cb: CallbackQuery):
|
105 |
+
if not await check_auth_click(cb):
|
106 |
+
return
|
107 |
+
|
108 |
+
page = int(cb.data.split(":")[1])
|
109 |
+
buttons, max_page = await gen_inline_help_buttons(page, sorted(Config.CMD_MENU))
|
110 |
+
|
111 |
+
caption = await help_template(
|
112 |
+
cb.from_user.mention,
|
113 |
+
(len(Config.CMD_INFO), len(Config.CMD_MENU)),
|
114 |
+
(page + 1, max_page),
|
115 |
+
)
|
116 |
+
|
117 |
+
try:
|
118 |
+
await cb.edit_message_text(
|
119 |
+
caption,
|
120 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
121 |
+
)
|
122 |
+
except Exception:
|
123 |
+
# handles MessageNotModified error
|
124 |
+
pass
|
125 |
+
|
126 |
+
|
127 |
+
@hellbot.bot.on_callback_query(filters.regex(r"help_menu"))
|
128 |
+
async def help_menu_cb(_, cb: CallbackQuery):
|
129 |
+
if not await check_auth_click(cb):
|
130 |
+
return
|
131 |
+
|
132 |
+
page = int(cb.data.split(":")[1])
|
133 |
+
plugin = str(cb.data.split(":")[2])
|
134 |
+
|
135 |
+
try:
|
136 |
+
buttons = [
|
137 |
+
InlineKeyboardButton(
|
138 |
+
f"{Symbols.bullet} {i}", f"help_cmd:{page}:{plugin}:{i}"
|
139 |
+
)
|
140 |
+
for i in sorted(Config.HELP_DICT[plugin]["commands"])
|
141 |
+
]
|
142 |
+
except KeyError:
|
143 |
+
await cb.answer("No description provided for this plugin!", show_alert=True)
|
144 |
+
return
|
145 |
+
|
146 |
+
buttons = [buttons[i : i + 2] for i in range(0, len(buttons), 2)]
|
147 |
+
buttons.append([InlineKeyboardButton(Symbols.back, f"help_page:{page}")])
|
148 |
+
|
149 |
+
caption = await command_template(
|
150 |
+
plugin,
|
151 |
+
Config.HELP_DICT[plugin]["info"],
|
152 |
+
len(sorted(Config.HELP_DICT[plugin]["commands"])),
|
153 |
+
)
|
154 |
+
|
155 |
+
try:
|
156 |
+
await cb.edit_message_text(
|
157 |
+
caption,
|
158 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
159 |
+
)
|
160 |
+
except Exception:
|
161 |
+
# handles MessageNotModified error
|
162 |
+
pass
|
163 |
+
|
164 |
+
|
165 |
+
@hellbot.bot.on_callback_query(filters.regex(r"help_cmd"))
|
166 |
+
async def help_cmd_cb(_, cb: CallbackQuery):
|
167 |
+
if not await check_auth_click(cb):
|
168 |
+
return
|
169 |
+
|
170 |
+
page = int(cb.data.split(":")[1])
|
171 |
+
plugin = str(cb.data.split(":")[2])
|
172 |
+
command = str(cb.data.split(":")[3])
|
173 |
+
result = ""
|
174 |
+
cmd_dict = Config.HELP_DICT[plugin]["commands"][command]
|
175 |
+
|
176 |
+
if cmd_dict["parameters"] is None:
|
177 |
+
result += f"**{Symbols.radio_select} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `{Config.HANDLERS[0]}{cmd_dict['command']}`"
|
178 |
+
else:
|
179 |
+
result += f"**{Symbols.radio_select} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `{Config.HANDLERS[0]}{cmd_dict['command']} {cmd_dict['parameters']}`"
|
180 |
+
|
181 |
+
if cmd_dict["description"]:
|
182 |
+
result += (
|
183 |
+
f"\n\n**{Symbols.arrow_right} 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** __{cmd_dict['description']}__"
|
184 |
+
)
|
185 |
+
|
186 |
+
if cmd_dict["example"]:
|
187 |
+
result += f"\n\n**{Symbols.arrow_right} 𝖤𝗑𝖺𝗆𝗉𝗅𝖾:** `{Config.HANDLERS[0]}{cmd_dict['example']}`"
|
188 |
+
|
189 |
+
if cmd_dict["note"]:
|
190 |
+
result += f"\n\n**{Symbols.arrow_right} 𝖭𝗈𝗍𝖾:** __{cmd_dict['note']}__"
|
191 |
+
|
192 |
+
result += f"\n\n**<\\> @Its_HellBot 🍀**"
|
193 |
+
|
194 |
+
buttons = [
|
195 |
+
[
|
196 |
+
InlineKeyboardButton(Symbols.back, f"help_menu:{page}:{plugin}"),
|
197 |
+
InlineKeyboardButton(Symbols.close, "help_data:c"),
|
198 |
+
]
|
199 |
+
]
|
200 |
+
|
201 |
+
try:
|
202 |
+
await cb.edit_message_text(
|
203 |
+
result,
|
204 |
+
ParseMode.MARKDOWN,
|
205 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
206 |
+
)
|
207 |
+
except Exception:
|
208 |
+
# handles MessageNotModified error
|
209 |
+
pass
|
210 |
+
|
211 |
+
|
212 |
+
@hellbot.bot.on_callback_query(filters.regex(r"help_data"))
|
213 |
+
async def help_close_cb(_, cb: CallbackQuery):
|
214 |
+
if not await check_auth_click(cb):
|
215 |
+
return
|
216 |
+
|
217 |
+
action = str(cb.data.split(":")[1])
|
218 |
+
if action == "c":
|
219 |
+
await cb.edit_message_text(
|
220 |
+
"**𝖧𝖾𝗅𝗉 𝖬𝖾𝗇𝗎 𝖢𝗅𝗈𝗌𝖾𝖽!**",
|
221 |
+
reply_markup=InlineKeyboardMarkup(
|
222 |
+
[[InlineKeyboardButton("Reopen", "help_data:reopen")]]
|
223 |
+
),
|
224 |
+
)
|
225 |
+
elif action == "reopen":
|
226 |
+
buttons, pages = await gen_inline_help_buttons(0, sorted(Config.CMD_MENU))
|
227 |
+
caption = await help_template(
|
228 |
+
cb.from_user.mention,
|
229 |
+
(len(Config.CMD_INFO), len(Config.CMD_MENU)),
|
230 |
+
(1, pages),
|
231 |
+
)
|
232 |
+
await cb.edit_message_text(
|
233 |
+
caption,
|
234 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
235 |
+
)
|
236 |
+
elif action == "botclose":
|
237 |
+
await cb.message.delete()
|
238 |
+
elif action == "bothelp":
|
239 |
+
buttons = await gen_bot_help_buttons()
|
240 |
+
await cb.edit_message_text(
|
241 |
+
HELP_MSG,
|
242 |
+
disable_web_page_preview=True,
|
243 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
244 |
+
)
|
245 |
+
elif action == "source":
|
246 |
+
buttons = [
|
247 |
+
[
|
248 |
+
InlineKeyboardButton("🚀 Deploy", url="https://github.com/The-HellBot/HellBot"),
|
249 |
+
InlineKeyboardButton("Plugins 📂", url="https://github.com/The-HellBot/Plugins"),
|
250 |
+
],
|
251 |
+
[
|
252 |
+
InlineKeyboardButton("нєℓℓвσт ηєтωσяк 🇮🇳", url="https://t.me/HellBot_Networks"),
|
253 |
+
],
|
254 |
+
[
|
255 |
+
InlineKeyboardButton("🎙️ Support", url="https://t.me/HellBot_Chats"),
|
256 |
+
InlineKeyboardButton("Updates 📣", url="https://t.me/Its_HellBot"),
|
257 |
+
],
|
258 |
+
[
|
259 |
+
InlineKeyboardButton("🔙", "help_data:start"),
|
260 |
+
InlineKeyboardButton(Symbols.close, "help_data:botclose"),
|
261 |
+
],
|
262 |
+
]
|
263 |
+
await cb.edit_message_text(
|
264 |
+
"__» The source code is available on GitHub. You can find the link below.__\n"
|
265 |
+
"__» Every project available under The-HellBot are open-source and free to use and modify to your needs.__\n"
|
266 |
+
"__» Anyone pretending to be the developer of this bot and selling the code, is a scammer.__\n\n"
|
267 |
+
"__» Please consider giving a star to the repository if you liked the project.__\n"
|
268 |
+
"__» Feel free to contact us if you need any help regarding the source code.__\n\n"
|
269 |
+
"**❤️ @HellBot_Networks 🇮🇳**",
|
270 |
+
disable_web_page_preview=True,
|
271 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
272 |
+
)
|
273 |
+
elif action == "start":
|
274 |
+
buttons = start_button()
|
275 |
+
await cb.edit_message_text(
|
276 |
+
START_MSG.format(cb.from_user.mention),
|
277 |
+
disable_web_page_preview=True,
|
278 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
279 |
+
)
|
HellBot/plugins/bot/forcesub.py
ADDED
@@ -0,0 +1,226 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import Client, filters
|
2 |
+
from pyrogram.errors import ChatAdminRequired, UserNotParticipant
|
3 |
+
from pyrogram.types import (
|
4 |
+
CallbackQuery,
|
5 |
+
ChatPermissions,
|
6 |
+
InlineKeyboardButton,
|
7 |
+
InlineKeyboardMarkup,
|
8 |
+
Message,
|
9 |
+
)
|
10 |
+
|
11 |
+
from Hellbot.core import LOGS
|
12 |
+
from Hellbot.functions.admins import is_user_admin
|
13 |
+
|
14 |
+
from ..btnsG import gen_inline_keyboard
|
15 |
+
from . import BotHelp, Config, Symbols, db, hellbot
|
16 |
+
|
17 |
+
|
18 |
+
@hellbot.bot.on_message(filters.command("forcesub") & Config.AUTH_USERS & filters.group)
|
19 |
+
async def force_sub(client: Client, message: Message):
|
20 |
+
if len(message.command) < 2:
|
21 |
+
return await message.reply_text("Give a channel username with command!")
|
22 |
+
|
23 |
+
try:
|
24 |
+
is_admin = await is_user_admin(message.chat, client.me.id)
|
25 |
+
if not is_admin:
|
26 |
+
return await message.reply_text(f"To use forcesub i must be an admin in {must_join}!")
|
27 |
+
except UserNotParticipant:
|
28 |
+
return await message.reply_text(f"To use forcesub i must be an admin in {must_join}!")
|
29 |
+
|
30 |
+
must_join = message.command[1]
|
31 |
+
try:
|
32 |
+
chat = await client.get_chat(must_join)
|
33 |
+
except Exception as e:
|
34 |
+
return await message.reply_text(f"**Error:**\n`{e}`")
|
35 |
+
|
36 |
+
if not await is_user_admin(chat, client.me.id):
|
37 |
+
return await message.reply_text("Make me admin in that channel first!")
|
38 |
+
|
39 |
+
await db.add_forcesub(message.chat.id, chat.id)
|
40 |
+
await message.reply_text(
|
41 |
+
f"**📌 𝖢𝗁𝖺𝗍 𝖥𝗈𝗋𝖼𝖾𝗌𝗎𝖻 𝖤𝗇𝖺𝖻𝗅𝖾𝖽!** \n\n"
|
42 |
+
f"__Users must join__ {chat.title} (`{chat.id}`) __to chat here!__"
|
43 |
+
)
|
44 |
+
|
45 |
+
if message.chat.id not in Config.FORCESUBS:
|
46 |
+
Config.FORCESUBS.add(message.chat.id)
|
47 |
+
|
48 |
+
|
49 |
+
@hellbot.bot.on_message(filters.command("unforcesub") & Config.AUTH_USERS)
|
50 |
+
async def unforce_sub(client: Client, message: Message):
|
51 |
+
if len(message.command) < 2:
|
52 |
+
return await message.reply_text(
|
53 |
+
"Give a channel username with command or give 'all' to remove all forcesubs from this chat!"
|
54 |
+
)
|
55 |
+
|
56 |
+
if not await is_user_admin(message.chat, client.me.id):
|
57 |
+
return await message.reply_text("To use forcesub i must be an admin!")
|
58 |
+
|
59 |
+
if "all" == message.command[1].lower():
|
60 |
+
await db.rm_all_forcesub(message.chat.id)
|
61 |
+
Config.FORCESUBS.remove(message.chat.id)
|
62 |
+
return await message.reply_text(f"**📌 Forcesub disabled!**")
|
63 |
+
|
64 |
+
try:
|
65 |
+
if await db.is_forcesub(message.chat.id, int(message.command[1])):
|
66 |
+
remaining = await db.rm_forcesub(message.chat.id, int(message.command[1]))
|
67 |
+
if remaining:
|
68 |
+
return await message.reply_text(
|
69 |
+
f"**📌 Removed Forcesub `{message.command[1]}`!**\n\n**Remaining Forcesub(s) in this chat:** `{remaining}`"
|
70 |
+
)
|
71 |
+
else:
|
72 |
+
Config.FORCESUBS.remove(message.chat.id)
|
73 |
+
return await message.reply_text(
|
74 |
+
f"**📌 Removed Forcesub `{message.command[1]}`!**"
|
75 |
+
)
|
76 |
+
else:
|
77 |
+
return await message.reply_text(f"**📌 This chat is not forcesub enabled!**")
|
78 |
+
except Exception as e:
|
79 |
+
return await message.reply_text(f"**Error:**\n`{e}`")
|
80 |
+
|
81 |
+
|
82 |
+
@hellbot.bot.on_message(filters.command("listforcesub") & Config.AUTH_USERS)
|
83 |
+
async def list_force_subs(client: Client, message: Message):
|
84 |
+
if not await is_user_admin(message.chat, client.me.id):
|
85 |
+
return await message.reply_text("To use forcesub i must be an admin!")
|
86 |
+
|
87 |
+
all_forcesubs = Config.FORCESUBS
|
88 |
+
|
89 |
+
text = ""
|
90 |
+
if len(all_forcesubs) > 0:
|
91 |
+
for forcesub in all_forcesubs:
|
92 |
+
try:
|
93 |
+
chat = await client.get_chat(forcesub["chat"])
|
94 |
+
text += f"**📌 {chat.title}** (`{chat.id}`)\n"
|
95 |
+
except:
|
96 |
+
text += f"**📌 {forcesub['chat']}** - `Invalid Chat!`\n"
|
97 |
+
else:
|
98 |
+
text = "**📌 No Forcesub Enabled in Bot!**"
|
99 |
+
|
100 |
+
await message.reply_text(text)
|
101 |
+
|
102 |
+
|
103 |
+
@hellbot.bot.on_message(filters.command("getforcesub") & Config.AUTH_USERS)
|
104 |
+
async def getforcesub(client: Client, message: Message):
|
105 |
+
if len(message.command) < 2:
|
106 |
+
chat = message.chat
|
107 |
+
else:
|
108 |
+
try:
|
109 |
+
chat = await client.get_chat(message.command[1])
|
110 |
+
except:
|
111 |
+
return await message.reply_text(f"**Invalid Channel Username/ID!**")
|
112 |
+
|
113 |
+
mustjoins = await db.get_forcesub(chat.id)
|
114 |
+
if mustjoins:
|
115 |
+
text = f"**This chat has {len(mustjoins['must_join'])} forcesub(s):**\n"
|
116 |
+
for must_join in mustjoins["must_join"]:
|
117 |
+
try:
|
118 |
+
chat = await client.get_chat(must_join)
|
119 |
+
text += f"**📌 {chat.title}** (`{chat.id}`)\n"
|
120 |
+
except:
|
121 |
+
text += f"**📌 {must_join}** - `Invalid Chat!`\n"
|
122 |
+
else:
|
123 |
+
text = "**📌 No Forcesub Enabled in This Chat!**"
|
124 |
+
|
125 |
+
await message.reply_text(text)
|
126 |
+
|
127 |
+
|
128 |
+
@hellbot.bot.on_message(
|
129 |
+
filters.group
|
130 |
+
& filters.incoming
|
131 |
+
& filters.new_chat_members
|
132 |
+
& ~filters.bot
|
133 |
+
& ~filters.service
|
134 |
+
& ~Config.AUTH_USERS
|
135 |
+
& ~filters.me
|
136 |
+
)
|
137 |
+
async def handle_force_sub(client: Client, message: Message):
|
138 |
+
if message.chat.id not in Config.FORCESUBS:
|
139 |
+
return
|
140 |
+
|
141 |
+
if not is_user_admin(message.chat, client.me.id):
|
142 |
+
return
|
143 |
+
|
144 |
+
btns_list = []
|
145 |
+
mustjoins = await db.get_forcesub(message.chat.id)
|
146 |
+
|
147 |
+
for i, must_join in enumerate(mustjoins["must_join"]):
|
148 |
+
try:
|
149 |
+
await client.get_chat_member(must_join, message.from_user.id)
|
150 |
+
except UserNotParticipant:
|
151 |
+
invite_link = await client.export_chat_invite_link(must_join)
|
152 |
+
btns_list.append((f"Join {i}", invite_link, "url"))
|
153 |
+
continue
|
154 |
+
except ChatAdminRequired:
|
155 |
+
continue
|
156 |
+
except Exception as e:
|
157 |
+
LOGS.warning(e)
|
158 |
+
continue
|
159 |
+
|
160 |
+
if len(btns_list) == 0:
|
161 |
+
return
|
162 |
+
|
163 |
+
join_btns = gen_inline_keyboard(btns_list, 2)
|
164 |
+
join_btns.append(
|
165 |
+
[
|
166 |
+
InlineKeyboardButton("Unmute 🗣️", f"forcesub:unmute:{message.from_user.id}:{message.chat.id}")
|
167 |
+
]
|
168 |
+
)
|
169 |
+
await message.reply_text(
|
170 |
+
f"**👋 Welcome to {message.chat.title}!**\n\n"
|
171 |
+
f"To be able to chat here, you must follow the instructions below:\n"
|
172 |
+
f" {Symbols.anchor} __Click the buttons below to join our important channels.__"
|
173 |
+
f" {Symbols.anchor} __After joining all channels, press the unmute button below.__"
|
174 |
+
f" {Symbols.anchor} __Then you can chat here.__",
|
175 |
+
disable_web_page_preview=True,
|
176 |
+
reply_markup=InlineKeyboardMarkup(join_btns),
|
177 |
+
)
|
178 |
+
|
179 |
+
|
180 |
+
@hellbot.bot.on_callback_query(filters.regex(r"forcesub"))
|
181 |
+
async def forcesub_cb(client: Client, cb: CallbackQuery):
|
182 |
+
data = cb.data.split(":")
|
183 |
+
if data[1] == "unmute":
|
184 |
+
try:
|
185 |
+
if not int(data[3]) == cb.message.chat.id:
|
186 |
+
return await cb.answer(
|
187 |
+
"**This is not for this chat!**", show_alert=True
|
188 |
+
)
|
189 |
+
|
190 |
+
must_join = await db.get_forcesub(cb.message.chat.id)
|
191 |
+
for chat in must_join["must_join"]:
|
192 |
+
try:
|
193 |
+
await client.get_chat_member(int(chat), cb.from_user.id)
|
194 |
+
except UserNotParticipant:
|
195 |
+
return await cb.answer(
|
196 |
+
"**You must join all channels first!**", show_alert=True
|
197 |
+
)
|
198 |
+
except ChatAdminRequired:
|
199 |
+
return await cb.answer(
|
200 |
+
"I'm not admin in some of the channels! Ask owner to make me admin.",
|
201 |
+
show_alert=True,
|
202 |
+
)
|
203 |
+
except Exception as e:
|
204 |
+
return await cb.answer(f"**Error:**\n`{e}`")
|
205 |
+
|
206 |
+
permissions = ChatPermissions(can_send_messages=True)
|
207 |
+
await cb.message.chat.restrict_member(int(data[2]), permissions)
|
208 |
+
except Exception as e:
|
209 |
+
return await cb.answer(f"**Error:**\n`{e}`")
|
210 |
+
|
211 |
+
await cb.answer("**📌 Unmuted!**", show_alert=True)
|
212 |
+
return await cb.message.delete()
|
213 |
+
|
214 |
+
|
215 |
+
BotHelp("ForceSub").add(
|
216 |
+
"forcesub",
|
217 |
+
"This command is used to force users to join some channels to chat in group.",
|
218 |
+
).add(
|
219 |
+
"unforcesub", "This command is used to remove channels from forcesub in group."
|
220 |
+
).add(
|
221 |
+
"listforcesub", "This command is used to list all forcesub in bot."
|
222 |
+
).add(
|
223 |
+
"getforcesub", "This command is used to get forcesub in group."
|
224 |
+
).info(
|
225 |
+
"ForceSub 🚀"
|
226 |
+
).done()
|
HellBot/plugins/bot/inline.py
ADDED
@@ -0,0 +1,39 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import filters
|
2 |
+
from pyrogram.types import (
|
3 |
+
InlineKeyboardMarkup,
|
4 |
+
InlineQuery,
|
5 |
+
InlineQueryResultArticle,
|
6 |
+
InputTextMessageContent,
|
7 |
+
)
|
8 |
+
|
9 |
+
from Hellbot.functions.templates import help_template
|
10 |
+
|
11 |
+
from ..btnsG import gen_inline_help_buttons
|
12 |
+
from . import Config, hellbot
|
13 |
+
|
14 |
+
|
15 |
+
@hellbot.bot.on_inline_query(filters.regex(r"help_menu"))
|
16 |
+
async def help_inline(_, query: InlineQuery):
|
17 |
+
if not query.from_user.id in Config.AUTH_USERS:
|
18 |
+
return
|
19 |
+
no_of_plugins = len(Config.CMD_MENU)
|
20 |
+
no_of_commands = len(Config.CMD_INFO)
|
21 |
+
buttons, pages = await gen_inline_help_buttons(0, sorted(Config.CMD_MENU))
|
22 |
+
caption = await help_template(
|
23 |
+
query.from_user.mention, (no_of_commands, no_of_plugins), (1, pages)
|
24 |
+
)
|
25 |
+
await query.answer(
|
26 |
+
results=[
|
27 |
+
(
|
28 |
+
InlineQueryResultArticle(
|
29 |
+
"HellBot Help Menu 🍀",
|
30 |
+
InputTextMessageContent(
|
31 |
+
caption,
|
32 |
+
disable_web_page_preview=True,
|
33 |
+
),
|
34 |
+
description="Inline Query for Help Menu of HellBot",
|
35 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
36 |
+
)
|
37 |
+
)
|
38 |
+
],
|
39 |
+
)
|
HellBot/plugins/bot/sessions.py
ADDED
@@ -0,0 +1,183 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import Client, filters
|
2 |
+
from pyrogram.errors import SessionPasswordNeeded
|
3 |
+
from pyrogram.types import (
|
4 |
+
CallbackQuery,
|
5 |
+
InlineKeyboardButton,
|
6 |
+
InlineKeyboardMarkup,
|
7 |
+
Message,
|
8 |
+
ReplyKeyboardRemove,
|
9 |
+
)
|
10 |
+
|
11 |
+
from ..btnsG import gen_inline_keyboard, start_button
|
12 |
+
from ..btnsK import session_keyboard
|
13 |
+
from . import START_MSG, BotHelp, Config, Symbols, db, hellbot
|
14 |
+
|
15 |
+
|
16 |
+
@hellbot.bot.on_message(
|
17 |
+
filters.command("session") & Config.AUTH_USERS & filters.private
|
18 |
+
)
|
19 |
+
async def session_menu(_, message: Message):
|
20 |
+
await message.reply_text(
|
21 |
+
"**🍀 𝖯𝗅𝖾𝖺𝗌𝖾 𝖼𝗁𝗈𝗈𝗌𝖾 𝖺𝗇 𝗈𝗉𝗍𝗂𝗈𝗇 𝖿𝗋𝗈𝗆 𝖻𝖾𝗅𝗈𝗐:**",
|
22 |
+
reply_markup=session_keyboard(),
|
23 |
+
)
|
24 |
+
|
25 |
+
|
26 |
+
@hellbot.bot.on_message(filters.regex(r"New 💫") & Config.AUTH_USERS & filters.private)
|
27 |
+
async def new_session(_, message: Message):
|
28 |
+
await message.reply_text(
|
29 |
+
"**𝖮𝗄𝖺𝗒!** 𝖫𝖾𝗍'𝗌 𝗌𝖾𝗍𝗎𝗉 𝖺 𝗇𝖾𝗐 𝗌𝖾𝗌𝗌𝗂𝗈𝗇",
|
30 |
+
reply_markup=ReplyKeyboardRemove(),
|
31 |
+
)
|
32 |
+
|
33 |
+
phone_number = await hellbot.bot.ask(
|
34 |
+
message.chat.id,
|
35 |
+
"**1.** 𝖤𝗇𝗍𝖾𝗋 𝗒𝗈𝗎𝗋 𝗍𝖾𝗅𝖾𝗀𝗋𝖺𝗆 𝖺𝖼𝖼𝗈𝗎𝗇𝗍 𝗉𝗁𝗈𝗇𝖾 𝗇𝗎𝗆𝖻𝖾𝗋 𝗍𝗈 𝖺𝖽𝖽 𝗍𝗁𝖾 𝗌𝖾𝗌𝗌𝗂𝗈𝗇: \n\n__𝖲𝖾𝗇𝖽 /cancel 𝗍𝗈 𝖼𝖺𝗇𝖼𝖾𝗅 𝗍𝗁𝖾 𝗈𝗉𝖾𝗋𝖺𝗍𝗂𝗈𝗇.__",
|
36 |
+
filters=filters.text,
|
37 |
+
timeout=120,
|
38 |
+
)
|
39 |
+
|
40 |
+
if phone_number.text == "/cancel":
|
41 |
+
return await message.reply_text("**𝖢𝖺𝗇𝖼𝖾𝗅𝗅𝖾𝖽!**")
|
42 |
+
elif not phone_number.text.startswith("+") and not phone_number.text[1:].isdigit():
|
43 |
+
return await message.reply_text(
|
44 |
+
"**𝖤𝗋𝗋𝗈𝗋!** 𝖯𝗁𝗈𝗇𝖾 𝗇𝗎𝗆𝖻𝖾𝗋 𝗆𝗎𝗌𝗍 𝖻𝖾 𝗂𝗇 𝖽𝗂𝗀𝗂𝗍𝗌 𝖺𝗇𝖽 𝗌𝗁𝗈𝗎𝗅𝖽 𝖼𝗈𝗇𝗍𝖺𝗂𝗇 𝖼𝗈𝗎𝗇𝗍𝗋𝗒 𝖼𝗈𝖽𝖾."
|
45 |
+
)
|
46 |
+
|
47 |
+
try:
|
48 |
+
client = Client(
|
49 |
+
name="Hellbot",
|
50 |
+
api_id=Config.API_ID,
|
51 |
+
api_hash=Config.API_HASH,
|
52 |
+
in_memory=True,
|
53 |
+
)
|
54 |
+
await client.connect()
|
55 |
+
|
56 |
+
code = await client.send_code(phone_number.text)
|
57 |
+
ask_otp = await hellbot.bot.ask(
|
58 |
+
message.chat.id,
|
59 |
+
"**2.** 𝖤𝗇𝗍𝖾𝗋 𝗍𝗁𝖾 𝖮𝖳𝖯 𝗌𝖾𝗇𝗍 𝗍𝗈 𝗒𝗈𝗎𝗋 𝗍𝖾𝗅𝖾𝗀𝗋𝖺𝗆 𝖺𝖼𝖼𝗈𝗎𝗇𝗍 𝖻𝗒 𝗌𝖾𝗉𝖺𝗋𝖺𝗍𝗂𝗇𝗀 𝖾𝗏𝖾𝗋𝗒 𝗇𝗎𝗆𝖻𝖾𝗋 𝗐𝗂𝗍𝗁 𝖺 𝗌𝗉𝖺𝖼𝖾. \n\n**𝖤𝗑𝖺𝗆𝗉𝗅𝖾:** `2 4 1 7 4`\n\n__𝖲𝖾𝗇𝖽 /cancel 𝗍𝗈 𝖼𝖺𝗇𝖼𝖾𝗅 𝗍𝗁𝖾 𝗈𝗉𝖾𝗋𝖺𝗍𝗂𝗈𝗇.__",
|
60 |
+
filters=filters.text,
|
61 |
+
timeout=300,
|
62 |
+
)
|
63 |
+
if ask_otp.text == "/cancel":
|
64 |
+
return await message.reply_text("**𝖢𝖺𝗇𝖼𝖾𝗅𝗅𝖾𝖽!**")
|
65 |
+
otp = ask_otp.text.replace(" ", "")
|
66 |
+
|
67 |
+
try:
|
68 |
+
await client.sign_in(phone_number.text, code.phone_code_hash, otp)
|
69 |
+
except SessionPasswordNeeded:
|
70 |
+
two_step_pass = await hellbot.bot.ask(
|
71 |
+
message.chat.id,
|
72 |
+
"**3.** 𝖤𝗇𝗍𝖾𝗋 𝗒𝗈𝗎𝗋 𝗍𝗐𝗈 𝗌𝗍𝖾𝗉 𝗏𝖾𝗋𝗂𝖿𝗂𝖼𝖺𝗍𝗂𝗈𝗇 𝗉𝖺𝗌𝗌𝗐𝗈𝗋𝖽: \n\n__𝖲𝖾𝗇𝖽 /cancel 𝗍𝗈 𝖼𝖺𝗇𝖼𝖾𝗅 𝗍𝗁𝖾 𝗈𝗉𝖾𝗋𝖺𝗍𝗂𝗈𝗇.__",
|
73 |
+
filters=filters.text,
|
74 |
+
timeout=120,
|
75 |
+
)
|
76 |
+
if two_step_pass.text == "/cancel":
|
77 |
+
return await message.reply_text("**𝖢𝖺𝗇𝖼𝖾𝗅𝗅𝖾𝖽!**")
|
78 |
+
await client.check_password(two_step_pass.text)
|
79 |
+
|
80 |
+
session_string = await client.export_session_string()
|
81 |
+
await message.reply_text(
|
82 |
+
f"**𝖲𝗎𝖼𝖼𝖾𝗌𝗌!** 𝖸𝗈𝗎𝗋 𝗌𝖾𝗌𝗌𝗂𝗈𝗇 𝗌𝗍𝗋𝗂𝗇𝗀 𝗂𝗌 𝗀𝖾𝗇𝖾𝗋𝖺𝗍𝖾𝖽. 𝖠𝖽𝖽𝗂𝗇𝗀 𝗂𝗍 𝗍𝗈 𝖽𝖺𝗍𝖺𝖻𝖺𝗌𝖾..."
|
83 |
+
)
|
84 |
+
user_id = (await client.get_me()).id
|
85 |
+
await db.update_session(user_id, session_string)
|
86 |
+
await client.disconnect()
|
87 |
+
await message.reply_text(
|
88 |
+
"**𝖲𝗎𝖼𝖼𝖾𝗌𝗌!** 𝖲𝖾𝗌𝗌𝗂𝗈𝗇 𝗌𝗍𝗋𝗂𝗇𝗀 𝖺𝖽𝖽𝖾𝖽 𝗍𝗈 𝖽𝖺𝗍𝖺𝖻𝖺𝗌𝖾. 𝖸𝗈𝗎 𝖼𝖺𝗇 𝗇𝗈𝗐 𝗎𝗌𝖾 𝖧𝖾𝗅𝗅𝖡𝗈𝗍 𝗈𝗇 𝗍𝗁𝗂𝗌 𝖺𝖼𝖼𝗈𝗎𝗇𝗍 𝖺𝖿𝗍𝖾𝗋 𝗋𝖾𝗌𝗍𝖺𝗋𝗍𝗂𝗇𝗀 𝗍𝗁𝖾 𝖻𝗈𝗍.\n\n**𝖭𝖮𝖳𝖤:** 𝖥𝗈𝗋 𝗌𝖾𝖼𝗎𝗋𝗂𝗍𝗒 𝗉𝗎𝗋𝗉𝗈���𝖾𝗌 𝗇𝗈𝖻𝗈𝖽𝗒 𝗐𝗂𝗅𝗅 𝗁𝖺𝗏𝖾 𝗍𝗁𝖾 𝖺𝖼𝖼𝖾𝗌𝗌 𝗍𝗈 𝗒𝗈𝗎𝗋 𝗌𝖾𝗌𝗌𝗂𝗈𝗇 𝗌𝗍𝗋𝗂𝗇𝗀. 𝖭𝗈𝗍 𝖾𝗏𝖾𝗇 𝗒𝗈𝗎 𝗈𝗋 𝗍𝗁𝖾 𝖻𝗈𝗍."
|
89 |
+
)
|
90 |
+
except TimeoutError:
|
91 |
+
await message.reply_text(
|
92 |
+
"**𝖳𝗂𝗆𝖾𝗈𝗎𝗍𝖤𝗋𝗋𝗈𝗋!** 𝖸𝗈𝗎 𝗍𝗈𝗈𝗄 𝗅𝗈𝗇𝗀𝖾𝗋 𝗍𝗁𝖺𝗇 𝖾𝗑𝖼𝗉𝖾𝖼𝗍𝖾𝖽 𝗍𝗈 𝖼𝗈𝗆𝗉𝗅𝖾𝗍𝖾 𝗍𝗁𝖾 𝗉𝗋𝗈𝖼𝖾𝗌𝗌. 𝖯𝗅𝖾𝖺𝗌𝖾 𝗍𝗋𝗒 𝖺𝗀𝖺𝗂𝗇."
|
93 |
+
)
|
94 |
+
except Exception as e:
|
95 |
+
await message.reply_text(f"**𝖤𝗋𝗋𝗈𝗋!** {e}")
|
96 |
+
|
97 |
+
|
98 |
+
@hellbot.bot.on_message(
|
99 |
+
filters.regex(r"Delete ❌") & Config.AUTH_USERS & filters.private
|
100 |
+
)
|
101 |
+
async def delete_session(_, message: Message):
|
102 |
+
all_sessions = await db.get_all_sessions()
|
103 |
+
if not all_sessions:
|
104 |
+
return await message.reply_text("𝖭𝗈 𝗌𝖾𝗌𝗌𝗂𝗈𝗇𝗌 𝖿𝗈𝗎𝗇𝖽 𝗂𝗇 𝖽𝖺𝗍𝖺𝖻𝖺𝗌𝖾.")
|
105 |
+
|
106 |
+
collection = []
|
107 |
+
for i in all_sessions:
|
108 |
+
collection.append((i["user_id"], f"rm_session:{i['user_id']}"))
|
109 |
+
|
110 |
+
buttons = gen_inline_keyboard(collection, 2)
|
111 |
+
buttons.append([InlineKeyboardButton("Cancel ❌", "auth_close")])
|
112 |
+
|
113 |
+
await message.reply_text(
|
114 |
+
"**𝖢𝗁𝗈𝗈𝗌𝖾 𝖺 𝗌𝖾𝗌𝗌𝗂𝗈𝗇 𝗍𝗈 𝖽𝖾𝗅𝖾𝗍𝖾:**",
|
115 |
+
reply_markup=InlineKeyboardMarkup(buttons),
|
116 |
+
)
|
117 |
+
|
118 |
+
|
119 |
+
@hellbot.bot.on_callback_query(filters.regex(r"rm_session"))
|
120 |
+
async def rm_session_cb(client: Client, cb: CallbackQuery):
|
121 |
+
collection = []
|
122 |
+
user_id = int(cb.data.split(":")[1])
|
123 |
+
all_sessions = await db.get_all_sessions()
|
124 |
+
|
125 |
+
if not all_sessions:
|
126 |
+
return await cb.message.delete()
|
127 |
+
|
128 |
+
try:
|
129 |
+
owner = await client.get_users(Config.OWNER_ID)
|
130 |
+
owner_id = owner.id
|
131 |
+
owner_name = owner.first_name
|
132 |
+
except:
|
133 |
+
owner_id = Config.OWNER_ID
|
134 |
+
owner_name = "𝖮𝗐𝗇𝖾𝗋"
|
135 |
+
if cb.from_user.id not in [user_id, owner_id]:
|
136 |
+
return await cb.answer(
|
137 |
+
f"𝖠𝖼𝖼𝖾𝗌𝗌 𝗋𝖾𝗌𝗍𝗋𝗂𝖼𝗍𝖾𝖽 𝗍𝗈 𝖺𝗇𝗈𝗍𝗁𝖾𝗋 𝗎𝗌𝖾𝗋𝗌. Only {owner_name} and session client can delete this session!",
|
138 |
+
show_alert=True,
|
139 |
+
)
|
140 |
+
|
141 |
+
await db.rm_session(user_id)
|
142 |
+
await cb.answer("**𝖲𝗎𝖼𝖼𝖾𝗌𝗌!** 𝖲𝖾𝗌𝗌𝗂𝗈𝗇 𝖽𝖾𝗅𝖾𝗍𝖾𝖽 𝖿𝗋𝗈𝗆 𝖽𝖺𝗍𝖺𝖻𝖺𝗌𝖾. \n__Restart the bot to apply changes.__", show_alert=True)
|
143 |
+
|
144 |
+
for i in all_sessions:
|
145 |
+
collection.append((i["user_id"], f"rm_session:{i['user_id']}"))
|
146 |
+
|
147 |
+
buttons = gen_inline_keyboard(collection, 2)
|
148 |
+
buttons.append([InlineKeyboardButton("Cancel ❌", "auth_close")])
|
149 |
+
|
150 |
+
await cb.message.edit_reply_markup(InlineKeyboardMarkup(buttons))
|
151 |
+
|
152 |
+
|
153 |
+
@hellbot.bot.on_message(filters.regex(r"List 📜") & Config.AUTH_USERS & filters.private)
|
154 |
+
async def list_sessions(_, message: Message):
|
155 |
+
all_sessions = await db.get_all_sessions()
|
156 |
+
if not all_sessions:
|
157 |
+
return await message.reply_text("𝖭𝗈 𝗌𝖾𝗌𝗌𝗂𝗈𝗇𝗌 𝖿𝗈𝗎𝗇𝖽 𝗂𝗇 𝖽𝖺𝗍𝖺𝖻𝖺𝗌𝖾.")
|
158 |
+
|
159 |
+
text = f"**{Symbols.cross_mark} 𝖫𝗂𝗌𝗍 𝗈𝖿 𝗌𝖾𝗌𝗌𝗂𝗈𝗇𝗌:**\n\n"
|
160 |
+
for i, session in enumerate(all_sessions):
|
161 |
+
text += f"[{'0' if i <= 9 else ''}{i+1}] {Symbols.bullet} **𝖴𝗌𝖾𝗋 𝖨𝖣:** `{session['user_id']}`\n"
|
162 |
+
|
163 |
+
await message.reply_text(text)
|
164 |
+
|
165 |
+
|
166 |
+
@hellbot.bot.on_message(filters.regex(r"Home 🏠") & filters.private & Config.AUTH_USERS)
|
167 |
+
async def go_home(_, message: Message):
|
168 |
+
await message.reply_text(
|
169 |
+
"**Home 🏠**",
|
170 |
+
reply_markup=ReplyKeyboardRemove(),
|
171 |
+
)
|
172 |
+
await message.reply_text(
|
173 |
+
START_MSG.format(message.from_user.mention),
|
174 |
+
disable_web_page_preview=True,
|
175 |
+
reply_markup=InlineKeyboardMarkup(start_button()),
|
176 |
+
)
|
177 |
+
|
178 |
+
|
179 |
+
BotHelp("Sessions").add(
|
180 |
+
"session", "This command is packed with tools to manage userbot sessions."
|
181 |
+
).info(
|
182 |
+
"Session 🚀"
|
183 |
+
).done()
|
HellBot/plugins/bot/users.py
ADDED
@@ -0,0 +1,82 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import Client, filters
|
2 |
+
from pyrogram.types import Message
|
3 |
+
|
4 |
+
from . import BotHelp, Config, Symbols, hellbot
|
5 |
+
|
6 |
+
|
7 |
+
@hellbot.bot.on_message(
|
8 |
+
filters.command("addauth") & Config.AUTH_USERS
|
9 |
+
)
|
10 |
+
async def addauth(client: Client, message: Message):
|
11 |
+
if not message.reply_to_message:
|
12 |
+
if len(message.command) < 2:
|
13 |
+
return await message.reply_text(
|
14 |
+
"Reply to a user or give me a userid/username to add them as an auth user!"
|
15 |
+
)
|
16 |
+
try:
|
17 |
+
user = await client.get_users(message.command[1])
|
18 |
+
except Exception:
|
19 |
+
return await message.reply_text(
|
20 |
+
"Give me a valid userid/username to add them as an auth user!"
|
21 |
+
)
|
22 |
+
else:
|
23 |
+
user = message.reply_to_message.from_user
|
24 |
+
|
25 |
+
if user.is_self:
|
26 |
+
return await message.reply_text("I can't add myself as an auth user!")
|
27 |
+
|
28 |
+
if user.id in Config.AUTH_USERS:
|
29 |
+
return await message.reply_text(f"**{user.mention} is already authorized**")
|
30 |
+
|
31 |
+
Config.AUTH_USERS.add(user.id)
|
32 |
+
await message.reply_text(f"**Added {user.mention} to auth users!**")
|
33 |
+
|
34 |
+
|
35 |
+
@hellbot.bot.on_message(
|
36 |
+
filters.command("delauth") & Config.AUTH_USERS
|
37 |
+
)
|
38 |
+
async def delauth(client: Client, message: Message):
|
39 |
+
if not message.reply_to_message:
|
40 |
+
if len(message.command) < 2:
|
41 |
+
return await message.reply_text(
|
42 |
+
"Reply to a user or give me a userid/username to add them as an auth user!"
|
43 |
+
)
|
44 |
+
try:
|
45 |
+
user = await client.get_users(message.command[1])
|
46 |
+
except Exception:
|
47 |
+
return await message.reply_text(
|
48 |
+
"Give me a valid userid/username to add them as an auth user!"
|
49 |
+
)
|
50 |
+
else:
|
51 |
+
user = message.reply_to_message.from_user
|
52 |
+
|
53 |
+
if user.id in Config.AUTH_USERS:
|
54 |
+
Config.AUTH_USERS.remove(user.id)
|
55 |
+
await message.reply_text(f"**Removed {user.mention} from auth users!**")
|
56 |
+
else:
|
57 |
+
await message.reply_text(f"**{user.mention} is not authorized**")
|
58 |
+
|
59 |
+
|
60 |
+
@hellbot.bot.on_message(
|
61 |
+
filters.command("authlist") & Config.AUTH_USERS
|
62 |
+
)
|
63 |
+
async def authlist(client: Client, message: Message):
|
64 |
+
text = "**🍀 Authorized Users:**\n\n"
|
65 |
+
for i, userid in enumerate(Config.AUTH_USERS):
|
66 |
+
try:
|
67 |
+
user = await client.get_users(userid)
|
68 |
+
text += f" {Symbols.anchor} {user.mention} (`{user.id}`)\n"
|
69 |
+
except:
|
70 |
+
text += f" {Symbols.anchor} Auth User #{i+1} (`{userid}`)\n"
|
71 |
+
|
72 |
+
await message.reply_text(text)
|
73 |
+
|
74 |
+
|
75 |
+
BotHelp("Users").add(
|
76 |
+
"addauth",
|
77 |
+
"This command is used to add a user as an authorized user. An authorized user can create and manage userbot session!",
|
78 |
+
).add("delauth", "This command is used to remove a user from authorized users.").add(
|
79 |
+
"authlist", "This command is used to list all authorized users."
|
80 |
+
).info(
|
81 |
+
"Users Command 🚀"
|
82 |
+
).done()
|
HellBot/plugins/btnsG.py
ADDED
@@ -0,0 +1,106 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# G: Glass Buttons
|
2 |
+
|
3 |
+
from math import ceil
|
4 |
+
|
5 |
+
from pyrogram.types import InlineKeyboardButton
|
6 |
+
|
7 |
+
from Hellbot.core import ENV, Symbols, db, Config
|
8 |
+
|
9 |
+
|
10 |
+
def gen_inline_keyboard(collection: list, row: int = 2) -> list[list[InlineKeyboardButton]]:
|
11 |
+
keyboard = []
|
12 |
+
for i in range(0, len(collection), row):
|
13 |
+
kyb = []
|
14 |
+
for x in collection[i : i + row]:
|
15 |
+
button = btn(*x)
|
16 |
+
kyb.append(button)
|
17 |
+
keyboard.append(kyb)
|
18 |
+
return keyboard
|
19 |
+
|
20 |
+
|
21 |
+
def btn(text, value, type="callback_data") -> InlineKeyboardButton:
|
22 |
+
return InlineKeyboardButton(text, **{type: value})
|
23 |
+
|
24 |
+
|
25 |
+
async def gen_inline_help_buttons(page: int, plugins: list) -> tuple[list, int]:
|
26 |
+
buttons = []
|
27 |
+
column = await db.get_env(ENV.btn_in_help) or 5
|
28 |
+
column = int(column)
|
29 |
+
emoji = await db.get_env(ENV.help_emoji) or "✧"
|
30 |
+
pairs = list(map(list, zip(plugins[::2], plugins[1::2])))
|
31 |
+
|
32 |
+
if len(plugins) % 2 == 1:
|
33 |
+
pairs.append([plugins[-1]])
|
34 |
+
|
35 |
+
max_pages = ceil(len(pairs) / column)
|
36 |
+
pairs = [pairs[i : i + column] for i in range(0, len(pairs), column)]
|
37 |
+
|
38 |
+
for pair in pairs[page]:
|
39 |
+
btn_pair = []
|
40 |
+
for i, plugin in enumerate(pair):
|
41 |
+
if i % 2 == 0:
|
42 |
+
btn_pair.append(
|
43 |
+
InlineKeyboardButton(f"{emoji} {plugin}", f"help_menu:{page}:{plugin}")
|
44 |
+
)
|
45 |
+
else:
|
46 |
+
btn_pair.append(
|
47 |
+
InlineKeyboardButton(f"{plugin} {emoji}", f"help_menu:{page}:{plugin}")
|
48 |
+
)
|
49 |
+
buttons.append(btn_pair)
|
50 |
+
|
51 |
+
buttons.append(
|
52 |
+
[
|
53 |
+
InlineKeyboardButton(
|
54 |
+
Symbols.previous, f"help_page:{(max_pages - 1) if page == 0 else (page - 1)}",
|
55 |
+
),
|
56 |
+
InlineKeyboardButton(
|
57 |
+
Symbols.close, "help_data:c"
|
58 |
+
),
|
59 |
+
InlineKeyboardButton(
|
60 |
+
Symbols.next, f"help_page:{0 if page == (max_pages - 1) else (page + 1)}",
|
61 |
+
),
|
62 |
+
]
|
63 |
+
)
|
64 |
+
|
65 |
+
return buttons, max_pages
|
66 |
+
|
67 |
+
|
68 |
+
async def gen_bot_help_buttons() -> list[list[InlineKeyboardButton]]:
|
69 |
+
buttons = []
|
70 |
+
plugins = sorted(Config.BOT_CMD_MENU)
|
71 |
+
emoji = await db.get_env(ENV.help_emoji) or "✧"
|
72 |
+
pairs = list(map(list, zip(plugins[::2], plugins[1::2])))
|
73 |
+
|
74 |
+
if len(plugins) % 2 == 1:
|
75 |
+
pairs.append([plugins[-1]])
|
76 |
+
|
77 |
+
for pair in pairs:
|
78 |
+
btn_pair = []
|
79 |
+
for i, plugin in enumerate(pair):
|
80 |
+
if i % 2 == 0:
|
81 |
+
btn_pair.append(
|
82 |
+
InlineKeyboardButton(f"{emoji} {plugin}", f"bot_help_menu:{plugin}")
|
83 |
+
)
|
84 |
+
else:
|
85 |
+
btn_pair.append(
|
86 |
+
InlineKeyboardButton(f"{plugin} {emoji}", f"bot_help_menu:{plugin}")
|
87 |
+
)
|
88 |
+
buttons.append(btn_pair)
|
89 |
+
|
90 |
+
buttons.append(
|
91 |
+
[
|
92 |
+
InlineKeyboardButton("🏠", "help_data:start"),
|
93 |
+
InlineKeyboardButton(Symbols.close, "help_data:botclose"),
|
94 |
+
]
|
95 |
+
)
|
96 |
+
|
97 |
+
return buttons
|
98 |
+
|
99 |
+
|
100 |
+
def start_button() -> list[list[InlineKeyboardButton]]:
|
101 |
+
return [
|
102 |
+
[
|
103 |
+
InlineKeyboardButton("⚙️ Help", "help_data:bothelp"),
|
104 |
+
InlineKeyboardButton("Source 📦", "help_data:source"),
|
105 |
+
]
|
106 |
+
]
|
HellBot/plugins/btnsK.py
ADDED
@@ -0,0 +1,45 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# K: Keyboard Buttons
|
2 |
+
|
3 |
+
from pyrogram.types import KeyboardButton, ReplyKeyboardMarkup
|
4 |
+
|
5 |
+
|
6 |
+
def gen_keyboard(collection: list, row: int = 2) -> list[list[KeyboardButton]]:
|
7 |
+
keyboard = []
|
8 |
+
for i in range(0, len(collection), row):
|
9 |
+
kyb = []
|
10 |
+
for x in collection[i : i + row]:
|
11 |
+
kyb.append(KeyboardButton(x))
|
12 |
+
keyboard.append(kyb)
|
13 |
+
return keyboard
|
14 |
+
|
15 |
+
|
16 |
+
def session_keyboard() -> ReplyKeyboardMarkup:
|
17 |
+
return ReplyKeyboardMarkup(
|
18 |
+
[
|
19 |
+
[
|
20 |
+
KeyboardButton("New 💫"),
|
21 |
+
KeyboardButton("Delete ❌"),
|
22 |
+
],
|
23 |
+
[
|
24 |
+
KeyboardButton("List 📜"),
|
25 |
+
KeyboardButton("Home 🏠"),
|
26 |
+
],
|
27 |
+
],
|
28 |
+
resize_keyboard=True,
|
29 |
+
)
|
30 |
+
|
31 |
+
|
32 |
+
def start_keyboard() -> ReplyKeyboardMarkup:
|
33 |
+
return ReplyKeyboardMarkup(
|
34 |
+
[
|
35 |
+
[
|
36 |
+
KeyboardButton("📟 Session"),
|
37 |
+
KeyboardButton("Force Sub ✨"),
|
38 |
+
],
|
39 |
+
[
|
40 |
+
KeyboardButton("👥 Users"),
|
41 |
+
KeyboardButton("Others 📣"),
|
42 |
+
],
|
43 |
+
],
|
44 |
+
resize_keyboard=True,
|
45 |
+
)
|
HellBot/plugins/decorator.py
ADDED
@@ -0,0 +1,67 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import Client, filters
|
2 |
+
from pyrogram.enums import ChatType
|
3 |
+
from pyrogram.handlers import MessageHandler
|
4 |
+
from pyrogram.types import Message
|
5 |
+
|
6 |
+
from Hellbot.core import Config, db, hellbot
|
7 |
+
from Hellbot.functions.admins import is_user_admin
|
8 |
+
|
9 |
+
|
10 |
+
def on_message(
|
11 |
+
command: str | list[str],
|
12 |
+
group: int = 0,
|
13 |
+
chat_type: list[ChatType] = None,
|
14 |
+
admin_only: bool = False,
|
15 |
+
allow_stan: bool = False,
|
16 |
+
):
|
17 |
+
if allow_stan:
|
18 |
+
_filter = (
|
19 |
+
filters.command(command, Config.HANDLERS)
|
20 |
+
& (filters.me | Config.STAN_USERS)
|
21 |
+
& ~filters.forwarded
|
22 |
+
& ~filters.via_bot
|
23 |
+
)
|
24 |
+
else:
|
25 |
+
_filter = (
|
26 |
+
filters.command(command, Config.HANDLERS)
|
27 |
+
& filters.me
|
28 |
+
& ~filters.forwarded
|
29 |
+
& ~filters.via_bot
|
30 |
+
)
|
31 |
+
|
32 |
+
def decorator(func):
|
33 |
+
async def wrapper(client: Client, message: Message):
|
34 |
+
if client.me.id != message.from_user.id:
|
35 |
+
if not await db.is_stan(client.me.id, message.from_user.id):
|
36 |
+
return
|
37 |
+
|
38 |
+
if admin_only and not message.chat.type == ChatType.PRIVATE:
|
39 |
+
if not await is_user_admin(message.chat, client.me.id):
|
40 |
+
return await hellbot.edit(message, "𝖨 𝖺𝗆 𝗇𝗈𝗍 𝖺𝗇 𝖺𝖽𝗆𝗂𝗇 𝗁𝖾𝗋𝖾!")
|
41 |
+
|
42 |
+
if chat_type and message.chat.type not in chat_type:
|
43 |
+
return await hellbot.edit(message, "𝖢𝖺𝗇'𝗍 𝗎𝗌𝖾 𝗍𝗁𝗂𝗌 𝖼𝗈𝗆𝗆𝖺𝗇𝖽 𝗁𝖾𝗋𝖾!")
|
44 |
+
|
45 |
+
await func(client, message)
|
46 |
+
message.continue_propagation()
|
47 |
+
|
48 |
+
for user in hellbot.users:
|
49 |
+
user.add_handler(MessageHandler(wrapper, _filter), group)
|
50 |
+
|
51 |
+
return wrapper
|
52 |
+
|
53 |
+
return decorator
|
54 |
+
|
55 |
+
|
56 |
+
def custom_handler(filters: filters.Filter, group: int = 0):
|
57 |
+
def decorator(func):
|
58 |
+
async def wrapper(client: Client, message: Message):
|
59 |
+
await func(client, message)
|
60 |
+
message.continue_propagation()
|
61 |
+
|
62 |
+
for user in hellbot.users:
|
63 |
+
user.add_handler(MessageHandler(wrapper, filters), group)
|
64 |
+
|
65 |
+
return wrapper
|
66 |
+
|
67 |
+
return decorator
|
HellBot/plugins/help.py
ADDED
@@ -0,0 +1,132 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from Hellbot.core.config import Config, Symbols
|
2 |
+
|
3 |
+
|
4 |
+
class HelpMenu:
|
5 |
+
def __init__(self, file: str) -> None:
|
6 |
+
self.filename = file
|
7 |
+
self.command_dict = {}
|
8 |
+
self.command_info = ""
|
9 |
+
|
10 |
+
def add(
|
11 |
+
self,
|
12 |
+
command: str,
|
13 |
+
parameters: str = None,
|
14 |
+
description: str = None,
|
15 |
+
example: str = None,
|
16 |
+
note: str = None,
|
17 |
+
):
|
18 |
+
self.command_dict[command] = {
|
19 |
+
"command": command,
|
20 |
+
"parameters": parameters,
|
21 |
+
"description": description,
|
22 |
+
"example": example,
|
23 |
+
"note": note,
|
24 |
+
}
|
25 |
+
return self
|
26 |
+
|
27 |
+
def info(self, command_info: str):
|
28 |
+
self.command_info = command_info
|
29 |
+
return self
|
30 |
+
|
31 |
+
def get_menu(self) -> str:
|
32 |
+
result = f"**𝖯𝗅𝗎𝗀𝗂𝗇 𝖥𝗂𝗅𝖾:** `{self.filename}`"
|
33 |
+
if self.command_info:
|
34 |
+
result += f"\n**𝖯𝗅𝗎𝗀𝗂𝗇 𝖨𝗇𝖿𝗈:** __{self.command_info} 🍀__"
|
35 |
+
result += "\n\n"
|
36 |
+
for command in self.command_dict:
|
37 |
+
command = self.command_dict[command]
|
38 |
+
result += f"**{Symbols.radio_select} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `{Config.HANDLERS[0]}{command['command']}"
|
39 |
+
if command["parameters"]:
|
40 |
+
result += f" {command['parameters']}`\n"
|
41 |
+
else:
|
42 |
+
result += "`\n"
|
43 |
+
if command["description"]:
|
44 |
+
result += (
|
45 |
+
f"**{Symbols.arrow_right} 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** __{command['description']}__\n"
|
46 |
+
)
|
47 |
+
if command["example"]:
|
48 |
+
result += f"**{Symbols.arrow_right} 𝖤𝗑𝖺𝗆𝗉𝗅𝖾:** `{Config.HANDLERS[0]}{command['example']}`\n"
|
49 |
+
if command["note"]:
|
50 |
+
result += f"**{Symbols.arrow_right} 𝖭𝗈𝗍𝖾:** __{command['note']}__\n"
|
51 |
+
|
52 |
+
result += "\n"
|
53 |
+
|
54 |
+
Config.CMD_INFO[command["command"]] = {
|
55 |
+
"command": f"{command['command']} {command['parameters'] if command['parameters'] else ''}",
|
56 |
+
"description": command["description"],
|
57 |
+
"example": command["example"],
|
58 |
+
"note": command["note"],
|
59 |
+
"plugin": self.filename,
|
60 |
+
}
|
61 |
+
|
62 |
+
return result
|
63 |
+
|
64 |
+
def done(self) -> None:
|
65 |
+
Config.HELP_DICT[self.filename] = {
|
66 |
+
"commands": self.command_dict,
|
67 |
+
"info": self.command_info,
|
68 |
+
}
|
69 |
+
Config.CMD_MENU[self.filename] = self.get_menu()
|
70 |
+
|
71 |
+
|
72 |
+
class BotHelp:
|
73 |
+
def __init__(self, file: str) -> None:
|
74 |
+
self.category = file
|
75 |
+
self.command_dict = {}
|
76 |
+
self.command_info = ""
|
77 |
+
|
78 |
+
def add(self, command: str, description: str):
|
79 |
+
self.command_dict[command] = {"command": command, "description": description}
|
80 |
+
return self
|
81 |
+
|
82 |
+
def info(self, command_info: str):
|
83 |
+
self.command_info = command_info
|
84 |
+
return self
|
85 |
+
|
86 |
+
def get_menu(self) -> str:
|
87 |
+
result = f"**𝖯𝗅𝗎𝗀𝗂𝗇 𝖢𝖺𝗍𝖾𝗀𝗈𝗋𝗒:** `{self.category}`"
|
88 |
+
if self.command_info:
|
89 |
+
result += f"\n**𝖯𝗅𝗎𝗀𝗂𝗇 𝖨𝗇𝖿𝗈:** __{self.command_info}__"
|
90 |
+
result += "\n\n"
|
91 |
+
for command in self.command_dict:
|
92 |
+
command = self.command_dict[command]
|
93 |
+
result += f"**{Symbols.radio_select} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `/{command['command']}`\n"
|
94 |
+
if command["description"]:
|
95 |
+
result += (
|
96 |
+
f"**{Symbols.arrow_right} 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** __{command['description']}__\n"
|
97 |
+
)
|
98 |
+
result += "\n"
|
99 |
+
|
100 |
+
Config.BOT_CMD_INFO[command["command"]] = {
|
101 |
+
"command": command["command"],
|
102 |
+
"description": command["description"],
|
103 |
+
"category": self.category,
|
104 |
+
}
|
105 |
+
|
106 |
+
return result
|
107 |
+
|
108 |
+
def done(self) -> None:
|
109 |
+
Config.BOT_HELP[self.category] = {
|
110 |
+
"commands": self.command_dict,
|
111 |
+
"info": self.command_info,
|
112 |
+
}
|
113 |
+
Config.BOT_CMD_MENU[self.category] = self.get_menu()
|
114 |
+
|
115 |
+
|
116 |
+
# example usage of HelpMenu class
|
117 |
+
"""
|
118 |
+
HelpMenu("example").add(
|
119 |
+
"example", "<text>", "description of command", "example of command", "note of command"
|
120 |
+
).info(
|
121 |
+
"information of plugin"
|
122 |
+
).done()
|
123 |
+
"""
|
124 |
+
|
125 |
+
# example usage of BotHelp class
|
126 |
+
"""
|
127 |
+
BotHelp("example").add(
|
128 |
+
"example", "description of command"
|
129 |
+
).info(
|
130 |
+
"information of category"
|
131 |
+
).done()
|
132 |
+
"""
|
HellBot/plugins/user/__init__.py
ADDED
@@ -0,0 +1,16 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram.enums import ChatType
|
2 |
+
|
3 |
+
from Hellbot.core.clients import hellbot
|
4 |
+
from Hellbot.core.config import Config, Symbols
|
5 |
+
from Hellbot.core.database import db
|
6 |
+
from Hellbot.plugins.decorator import custom_handler, on_message
|
7 |
+
from Hellbot.plugins.help import HelpMenu
|
8 |
+
|
9 |
+
handler = Config.HANDLERS[0]
|
10 |
+
bot = hellbot.bot
|
11 |
+
|
12 |
+
bot_only = [ChatType.BOT]
|
13 |
+
group_n_channel = [ChatType.GROUP, ChatType.SUPERGROUP, ChatType.CHANNEL]
|
14 |
+
group_only = [ChatType.GROUP, ChatType.SUPERGROUP]
|
15 |
+
private_n_bot = [ChatType.PRIVATE, ChatType.BOT]
|
16 |
+
private_only = [ChatType.PRIVATE]
|
HellBot/plugins/user/admins.py
ADDED
@@ -0,0 +1,510 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
|
3 |
+
from pyrogram import Client, filters
|
4 |
+
from pyrogram.types import ChatPermissions, ChatPrivileges, Message
|
5 |
+
|
6 |
+
from Hellbot.core import LOGS
|
7 |
+
|
8 |
+
from . import HelpMenu, custom_handler, db, group_only, handler, hellbot, on_message
|
9 |
+
|
10 |
+
|
11 |
+
@on_message(
|
12 |
+
"promote",
|
13 |
+
chat_type=group_only,
|
14 |
+
admin_only=True,
|
15 |
+
allow_stan=True,
|
16 |
+
)
|
17 |
+
async def promote(client: Client, message: Message):
|
18 |
+
if len(message.command) < 2 and not message.reply_to_message:
|
19 |
+
return await hellbot.delete(
|
20 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗉𝗋𝗈𝗆𝗈𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
21 |
+
)
|
22 |
+
|
23 |
+
if message.reply_to_message:
|
24 |
+
user = message.reply_to_message.from_user
|
25 |
+
title = await hellbot.input(message)
|
26 |
+
else:
|
27 |
+
user = await client.get_users(message.command[1])
|
28 |
+
title = (await hellbot.input(message)).split(" ", 1)[1].strip() or ""
|
29 |
+
|
30 |
+
try:
|
31 |
+
privileges = ChatPrivileges(
|
32 |
+
can_manage_chat=True,
|
33 |
+
can_delete_messages=True,
|
34 |
+
can_manage_video_chats=True,
|
35 |
+
can_restrict_members=False,
|
36 |
+
can_promote_members=False,
|
37 |
+
can_change_info=False,
|
38 |
+
can_invite_users=True,
|
39 |
+
can_pin_messages=True,
|
40 |
+
is_anonymous=False,
|
41 |
+
)
|
42 |
+
await message.chat.promote_member(user.id, privileges)
|
43 |
+
await client.set_administrator_title(message.chat.id, user.id, title)
|
44 |
+
except Exception as e:
|
45 |
+
return await hellbot.error(message, e)
|
46 |
+
|
47 |
+
await hellbot.delete(message, f"**💫 𝖯𝗋𝗈𝗆𝗈𝗍𝖾𝖽 {user.mention} 𝗌𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**")
|
48 |
+
await hellbot.check_and_log(
|
49 |
+
"promote",
|
50 |
+
f"**Promoted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
51 |
+
)
|
52 |
+
|
53 |
+
|
54 |
+
@on_message(
|
55 |
+
"demote",
|
56 |
+
chat_type=group_only,
|
57 |
+
admin_only=True,
|
58 |
+
allow_stan=True,
|
59 |
+
)
|
60 |
+
async def demote(client: Client, message: Message):
|
61 |
+
if len(message.command) < 2 and not message.reply_to_message:
|
62 |
+
return await hellbot.delete(
|
63 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝖽𝖾𝗆𝗈𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
64 |
+
)
|
65 |
+
|
66 |
+
if message.reply_to_message:
|
67 |
+
user = message.reply_to_message.from_user
|
68 |
+
else:
|
69 |
+
user = await client.get_users(message.command[1])
|
70 |
+
try:
|
71 |
+
privileges = ChatPrivileges(
|
72 |
+
can_manage_chat=False,
|
73 |
+
can_delete_messages=False,
|
74 |
+
can_manage_video_chats=False,
|
75 |
+
can_restrict_members=False,
|
76 |
+
can_promote_members=False,
|
77 |
+
can_change_info=False,
|
78 |
+
can_invite_users=False,
|
79 |
+
can_pin_messages=False,
|
80 |
+
is_anonymous=False,
|
81 |
+
)
|
82 |
+
await message.chat.promote_member(user.id, privileges)
|
83 |
+
except Exception as e:
|
84 |
+
return await hellbot.error(message, e)
|
85 |
+
|
86 |
+
await hellbot.delete(message, f"**🙄 𝖣𝖾𝗆𝗈𝗍𝖾𝖽 {user.mention} 𝗌𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**")
|
87 |
+
await hellbot.check_and_log(
|
88 |
+
"demote",
|
89 |
+
f"**Demoted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
90 |
+
)
|
91 |
+
|
92 |
+
|
93 |
+
@on_message(
|
94 |
+
["ban", "dban"],
|
95 |
+
chat_type=group_only,
|
96 |
+
admin_only=True,
|
97 |
+
allow_stan=True,
|
98 |
+
)
|
99 |
+
async def ban(client: Client, message: Message):
|
100 |
+
if message.reply_to_message:
|
101 |
+
user = message.reply_to_message.from_user
|
102 |
+
if len(message.command) < 2:
|
103 |
+
reason = None
|
104 |
+
else:
|
105 |
+
reason = await hellbot.input(message)
|
106 |
+
if message.command[0][0].lower() == "d":
|
107 |
+
await message.reply_to_message.delete()
|
108 |
+
elif len(message.command) == 2:
|
109 |
+
user = await client.get_users(message.command[1])
|
110 |
+
reason = None
|
111 |
+
elif len(message.command) > 2:
|
112 |
+
user = await client.get_users(message.command[1])
|
113 |
+
reason = (await hellbot.input(message)).split(" ", 1)[1].strip()
|
114 |
+
else:
|
115 |
+
return await hellbot.delete(
|
116 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝖻𝖺𝗇 𝗍𝗁𝖾𝗆!"
|
117 |
+
)
|
118 |
+
|
119 |
+
try:
|
120 |
+
await message.chat.ban_member(user.id)
|
121 |
+
except Exception as e:
|
122 |
+
return await hellbot.error(message, e)
|
123 |
+
|
124 |
+
reason = reason if reason else "Not Specified"
|
125 |
+
await hellbot.delete(
|
126 |
+
message,
|
127 |
+
f"**☠️ 𝖡𝖺𝗇𝗇𝖾𝖽 {user.mention} 𝗌𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**\n**𝖱𝖾𝖺𝗌𝗈𝗇:** `{reason}`",
|
128 |
+
30,
|
129 |
+
)
|
130 |
+
await hellbot.check_and_log(
|
131 |
+
"ban",
|
132 |
+
f"**Banned User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Reason:** `{reason}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
133 |
+
)
|
134 |
+
|
135 |
+
|
136 |
+
@on_message(
|
137 |
+
"unban",
|
138 |
+
chat_type=group_only,
|
139 |
+
admin_only=True,
|
140 |
+
allow_stan=True,
|
141 |
+
)
|
142 |
+
async def unban(client: Client, message: Message):
|
143 |
+
if len(message.command) < 2 and not message.reply_to_message:
|
144 |
+
return await hellbot.delete(
|
145 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗎𝗇𝖻𝖺𝗇 𝗍𝗁𝖾𝗆!"
|
146 |
+
)
|
147 |
+
|
148 |
+
if message.reply_to_message:
|
149 |
+
user = message.reply_to_message.from_user
|
150 |
+
else:
|
151 |
+
user = await client.get_users(message.command[1])
|
152 |
+
|
153 |
+
try:
|
154 |
+
await message.chat.unban_member(user.id)
|
155 |
+
except Exception as e:
|
156 |
+
return await hellbot.error(message, e)
|
157 |
+
|
158 |
+
await hellbot.delete(message, f"**🤗 𝖴𝗇𝖻𝖺𝗇𝗇𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**", 30)
|
159 |
+
await hellbot.check_and_log(
|
160 |
+
"unban",
|
161 |
+
f"**Unbanned User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
162 |
+
)
|
163 |
+
|
164 |
+
|
165 |
+
@on_message(
|
166 |
+
["kick", "dkick"],
|
167 |
+
chat_type=group_only,
|
168 |
+
admin_only=True,
|
169 |
+
allow_stan=True,
|
170 |
+
)
|
171 |
+
async def kick(client: Client, message: Message):
|
172 |
+
if message.reply_to_message:
|
173 |
+
user = message.reply_to_message.from_user
|
174 |
+
if len(message.command) < 2:
|
175 |
+
reason = None
|
176 |
+
else:
|
177 |
+
reason = await hellbot.input(message)
|
178 |
+
if message.command[0][0].lower() == "d":
|
179 |
+
await message.reply_to_message.delete()
|
180 |
+
elif len(message.command) == 2:
|
181 |
+
user = await client.get_users(message.command[1])
|
182 |
+
reason = None
|
183 |
+
elif len(message.command) > 2:
|
184 |
+
user = await client.get_users(message.command[1])
|
185 |
+
reason = (await hellbot.input(message)).split(" ", 1)[1].strip()
|
186 |
+
else:
|
187 |
+
return await hellbot.delete(
|
188 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗄𝗂𝖼𝗄 𝗍𝗁𝖾𝗆!"
|
189 |
+
)
|
190 |
+
|
191 |
+
try:
|
192 |
+
await message.chat.ban_member(user.id)
|
193 |
+
except Exception as e:
|
194 |
+
return await hellbot.error(message, e)
|
195 |
+
|
196 |
+
reason = reason if reason else "Not Specified"
|
197 |
+
await hellbot.delete(
|
198 |
+
message,
|
199 |
+
f"**👋 𝖪𝗂𝖼𝗄𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**\n**𝖱𝖾𝖺𝗌𝗈𝗇:** `{reason}`",
|
200 |
+
30,
|
201 |
+
)
|
202 |
+
await hellbot.check_and_log(
|
203 |
+
"kick",
|
204 |
+
f"**Kicked User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Reason:** `{reason}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
205 |
+
)
|
206 |
+
await asyncio.sleep(5)
|
207 |
+
await message.chat.unban_member(user.id)
|
208 |
+
|
209 |
+
|
210 |
+
@on_message(
|
211 |
+
"mute",
|
212 |
+
chat_type=group_only,
|
213 |
+
admin_only=True,
|
214 |
+
allow_stan=True,
|
215 |
+
)
|
216 |
+
async def mute(client: Client, message: Message):
|
217 |
+
if message.reply_to_message:
|
218 |
+
user = message.reply_to_message.from_user
|
219 |
+
if len(message.command) < 2:
|
220 |
+
reason = None
|
221 |
+
else:
|
222 |
+
reason = await hellbot.input(message)
|
223 |
+
elif len(message.command) == 2:
|
224 |
+
user = await client.get_users(message.command[1])
|
225 |
+
reason = None
|
226 |
+
elif len(message.command) > 2:
|
227 |
+
user = await client.get_users(message.command[1])
|
228 |
+
reason = (await hellbot.input(message)).split(" ", 1)[1].strip()
|
229 |
+
else:
|
230 |
+
return await hellbot.delete(
|
231 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗆𝗎𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
232 |
+
)
|
233 |
+
|
234 |
+
try:
|
235 |
+
permissions = ChatPermissions(
|
236 |
+
can_send_messages=False,
|
237 |
+
)
|
238 |
+
await message.chat.restrict_member(user.id, permissions)
|
239 |
+
except Exception as e:
|
240 |
+
return await hellbot.error(message, e)
|
241 |
+
|
242 |
+
reason = reason if reason else "Not Specified"
|
243 |
+
await hellbot.delete(
|
244 |
+
message,
|
245 |
+
f"**🤐 𝖬𝗎𝗍𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**\n**𝖱𝖾𝖺𝗌𝗈𝗇:** `{reason}`",
|
246 |
+
30,
|
247 |
+
)
|
248 |
+
await hellbot.check_and_log(
|
249 |
+
"mute",
|
250 |
+
f"**Muted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Reason:** `{reason}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
251 |
+
)
|
252 |
+
|
253 |
+
|
254 |
+
@on_message(
|
255 |
+
"unmute",
|
256 |
+
chat_type=group_only,
|
257 |
+
admin_only=True,
|
258 |
+
allow_stan=True,
|
259 |
+
)
|
260 |
+
async def unmute(client: Client, message: Message):
|
261 |
+
if len(message.command) < 2 and not message.reply_to_message:
|
262 |
+
return await hellbot.delete(
|
263 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗎𝗇𝗆𝗎𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
264 |
+
)
|
265 |
+
|
266 |
+
if message.reply_to_message:
|
267 |
+
user = message.reply_to_message.from_user
|
268 |
+
else:
|
269 |
+
user = await client.get_users(message.command[1])
|
270 |
+
|
271 |
+
try:
|
272 |
+
permissions = ChatPermissions(
|
273 |
+
can_send_messages=True,
|
274 |
+
)
|
275 |
+
await message.chat.restrict_member(user.id, permissions)
|
276 |
+
except Exception as e:
|
277 |
+
return await hellbot.error(message, e)
|
278 |
+
|
279 |
+
await hellbot.delete(message, f"**😁 𝖴𝗇𝗆𝗎𝗍𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**", 30)
|
280 |
+
await hellbot.check_and_log(
|
281 |
+
"unmute",
|
282 |
+
f"**Unmuted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
283 |
+
)
|
284 |
+
|
285 |
+
|
286 |
+
@on_message("dmute", allow_stan=True)
|
287 |
+
async def dmute(client: Client, message: Message):
|
288 |
+
if message.reply_to_message:
|
289 |
+
user = message.reply_to_message.from_user
|
290 |
+
if len(message.command) < 2:
|
291 |
+
reason = None
|
292 |
+
else:
|
293 |
+
reason = await hellbot.input(message)
|
294 |
+
elif len(message.command) == 2:
|
295 |
+
user = await client.get_users(message.command[1])
|
296 |
+
reason = None
|
297 |
+
elif len(message.command) > 2:
|
298 |
+
user = await client.get_users(message.command[1])
|
299 |
+
reason = (await hellbot.input(message)).split(" ", 1)[1].strip()
|
300 |
+
else:
|
301 |
+
return await hellbot.delete(
|
302 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗆𝗎𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
303 |
+
)
|
304 |
+
|
305 |
+
if await db.is_muted(client.me.id, user.id, message.chat.id):
|
306 |
+
return await hellbot.delete(message, "This user is already dmuted.")
|
307 |
+
|
308 |
+
reason = reason if reason else "Not Specified"
|
309 |
+
await db.add_mute(client.me.id, user.id, message.chat.id, reason)
|
310 |
+
await hellbot.delete(
|
311 |
+
message,
|
312 |
+
f"**🤐 𝖬𝗎𝗍𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**\n**𝖱𝖾𝖺𝗌𝗈𝗇:** `{reason}`",
|
313 |
+
30,
|
314 |
+
)
|
315 |
+
await hellbot.check_and_log(
|
316 |
+
"dmute",
|
317 |
+
f"**D-Muted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Reason:** `{reason}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title or message.chat.first_name}`\n**Group ID:** `{message.chat.id}`",
|
318 |
+
)
|
319 |
+
|
320 |
+
|
321 |
+
@on_message("undmute", allow_stan=True)
|
322 |
+
async def undmute(client: Client, message: Message):
|
323 |
+
if len(message.command) < 2 and not message.reply_to_message:
|
324 |
+
return await hellbot.delete(
|
325 |
+
message, "𝖭𝖾𝖾𝖽 𝖺 𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽 𝗈𝗋 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝖺 𝗎𝗌𝖾𝗋 𝗍𝗈 𝗎𝗇𝗆𝗎𝗍𝖾 𝗍𝗁𝖾𝗆!"
|
326 |
+
)
|
327 |
+
|
328 |
+
if message.reply_to_message:
|
329 |
+
user = message.reply_to_message.from_user
|
330 |
+
else:
|
331 |
+
user = await client.get_users(message.command[1])
|
332 |
+
|
333 |
+
if not await db.is_muted(client.me.id, user.id, message.chat.id):
|
334 |
+
return await hellbot.delete(message, "𝖳𝗁𝖾 𝗎𝗌𝖾𝗋 𝗂𝗌 𝗇𝗈𝗍 𝗆𝗎𝗍𝖾𝖽!")
|
335 |
+
|
336 |
+
reason = await db.rm_mute(client.me.id, user.id, message.chat.id)
|
337 |
+
await hellbot.delete(
|
338 |
+
message,
|
339 |
+
f"**😁 𝖴𝗇𝗆𝗎𝗍𝖾𝖽 {user.mention} 𝖲𝗎𝖼𝖼𝖾𝗌𝗌𝖿𝗎𝗅𝗅𝗒!**\n\n**Mute reason was:** `{reason}`",
|
340 |
+
30,
|
341 |
+
)
|
342 |
+
await hellbot.check_and_log(
|
343 |
+
"unmute",
|
344 |
+
f"**D-Unmuted User**\n\n**User:** {user.mention}\n**User ID:** `{user.id}`\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
345 |
+
)
|
346 |
+
|
347 |
+
|
348 |
+
@on_message(
|
349 |
+
"pin",
|
350 |
+
chat_type=group_only,
|
351 |
+
admin_only=True,
|
352 |
+
allow_stan=True,
|
353 |
+
)
|
354 |
+
async def pin(_, message: Message):
|
355 |
+
if not message.reply_to_message:
|
356 |
+
return await hellbot.delete(message, "𝖭𝖾𝖾𝖽 𝖺 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝗉𝗂𝗇 𝖺 𝗆𝖾𝗌𝗌𝖺𝗀𝖾!")
|
357 |
+
|
358 |
+
try:
|
359 |
+
await message.reply_to_message.pin()
|
360 |
+
except Exception as e:
|
361 |
+
return await hellbot.error(message, e)
|
362 |
+
|
363 |
+
await hellbot.delete(
|
364 |
+
message,
|
365 |
+
f"**📌 𝖯𝗂𝗇𝗇𝖾𝖽 [𝖬𝖾𝗌𝗌𝖺𝗀𝖾]({message.reply_to_message.link}) 𝗂𝗇 {message.chat.title}!**",
|
366 |
+
30,
|
367 |
+
)
|
368 |
+
await hellbot.check_and_log(
|
369 |
+
"pin",
|
370 |
+
f"**Pinned Message**\n\n**Message:** [Click Here]({message.reply_to_message.link})\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
371 |
+
)
|
372 |
+
|
373 |
+
|
374 |
+
@on_message(
|
375 |
+
"unpin",
|
376 |
+
chat_type=group_only,
|
377 |
+
admin_only=True,
|
378 |
+
allow_stan=True,
|
379 |
+
)
|
380 |
+
async def unpin(_, message: Message):
|
381 |
+
if not message.reply_to_message:
|
382 |
+
return await hellbot.delete(message, "𝖭𝖾𝖾𝖽 𝖺 𝗋𝖾𝗉𝗅𝗒 𝗍𝗈 𝗎𝗇𝗉𝗂𝗇 𝖺 𝗆𝖾𝗌𝗌𝖺𝗀𝖾!")
|
383 |
+
|
384 |
+
try:
|
385 |
+
await message.reply_to_message.unpin()
|
386 |
+
except Exception as e:
|
387 |
+
return await hellbot.error(message, e)
|
388 |
+
|
389 |
+
await hellbot.delete(
|
390 |
+
message,
|
391 |
+
f"**📌 𝖴𝗇𝗉𝗂𝗇𝗇𝖾𝖽 [𝖬𝖾𝗌𝗌𝖺𝗀𝖾]({message.reply_to_message.link}) 𝗂𝗇 {message.chat.title}!**",
|
392 |
+
30,
|
393 |
+
)
|
394 |
+
await hellbot.check_and_log(
|
395 |
+
"unpin",
|
396 |
+
f"**Unpinned Message**\n\n**Message:** [Click Here]({message.reply_to_message.link})\n**Admin:** `{message.from_user.mention}`\n**Group:** `{message.chat.title}`\n**Group ID:** `{message.chat.id}`",
|
397 |
+
)
|
398 |
+
|
399 |
+
|
400 |
+
@on_message(
|
401 |
+
"zombies",
|
402 |
+
chat_type=group_only,
|
403 |
+
admin_only=True,
|
404 |
+
allow_stan=True,
|
405 |
+
)
|
406 |
+
async def zombies(_, message: Message):
|
407 |
+
hell = await hellbot.edit(message, "☠️ 𝖣𝖾𝗍𝖾𝖼𝗍𝗂𝗇𝗀 𝗓𝗈𝗆𝖻𝗂𝖾𝗌...")
|
408 |
+
ded_users = []
|
409 |
+
async for members in message.chat.get_members():
|
410 |
+
if members.user.is_deleted:
|
411 |
+
ded_users.append(members.user.id)
|
412 |
+
|
413 |
+
if not ded_users:
|
414 |
+
return await hell.edit(
|
415 |
+
"🫡 𝖣𝗈𝗇'𝗍 𝗁𝖺𝗏𝖾 𝖺𝗇𝗒 𝗓𝗈𝗆𝖻𝗂𝖾𝗌 𝗂𝗇 𝗍𝗁𝗂𝗌 𝗀𝗋𝗈𝗎𝗉. **𝖦𝗋𝗈𝗎𝗉𝗌' 𝖼𝗅𝖾𝖺𝗇 𝖠𝖥!**"
|
416 |
+
)
|
417 |
+
|
418 |
+
if len(message.command) > 1 and message.command[1].lower() == "clean":
|
419 |
+
await hell.edit(
|
420 |
+
f"☠️ 𝖥𝗈𝗎𝗇𝖽 {len(ded_users)} 𝗓𝗈𝗆𝖻𝗂𝖾𝗌... **🔫 𝖳𝗂𝗆𝖾 𝗍𝗈 𝗉𝗎𝗋𝗀𝖾 𝗍𝗁𝖾𝗆!**"
|
421 |
+
)
|
422 |
+
failed = 0
|
423 |
+
success = 0
|
424 |
+
for user in ded_users:
|
425 |
+
try:
|
426 |
+
await message.chat.ban_member(user)
|
427 |
+
success += 1
|
428 |
+
except Exception as e:
|
429 |
+
LOGS.error(e)
|
430 |
+
failed += 1
|
431 |
+
|
432 |
+
await hell.edit(f"**𝖯𝗎𝗋𝗀𝖾𝖽 {success} 𝗓𝗈𝗆𝖻𝗂𝖾𝗌!**\n`{failed}` holds immunity!")
|
433 |
+
else:
|
434 |
+
await hell.edit(
|
435 |
+
f"**☠️ 𝖥𝗈𝗎𝗇𝖽 {len(ded_users)} 𝗓𝗈𝗆𝖻𝗂𝖾𝗌!**\n\n__Use__ `{handler}zombies clean` __to kill them!__"
|
436 |
+
)
|
437 |
+
|
438 |
+
|
439 |
+
@custom_handler(filters.incoming)
|
440 |
+
async def multiple_handler(client: Client, message: Message):
|
441 |
+
if not message.from_user:
|
442 |
+
return
|
443 |
+
|
444 |
+
if await db.is_muted(client.me.id, message.from_user.id, message.chat.id):
|
445 |
+
try:
|
446 |
+
await message.delete()
|
447 |
+
except:
|
448 |
+
pass
|
449 |
+
|
450 |
+
elif await db.is_gmuted(message.from_user.id):
|
451 |
+
try:
|
452 |
+
await message.delete()
|
453 |
+
except:
|
454 |
+
pass
|
455 |
+
|
456 |
+
elif await db.is_echo(client.me.id, message.chat.id, message.from_user.id):
|
457 |
+
await message.copy(message.chat.id, reply_to_message_id=message.id)
|
458 |
+
|
459 |
+
|
460 |
+
HelpMenu("admin").add(
|
461 |
+
"promote",
|
462 |
+
"<𝗎𝗌𝖾𝗋𝗇𝖺𝗆𝖾/𝗂𝖽/reply> <𝗍𝗂𝗍𝗅𝖾>",
|
463 |
+
"Promote a user to admin.",
|
464 |
+
"promote @ForGo10God hellboy",
|
465 |
+
).add(
|
466 |
+
"demote", "<username/id/reply>", "Demote a user from admin.", "demote @ForGo10God"
|
467 |
+
).add(
|
468 |
+
"ban",
|
469 |
+
"<username/id/reply> <reason>",
|
470 |
+
"Ban a user from the group.",
|
471 |
+
"ban @ForGo10God",
|
472 |
+
"You can also use dban to delete the message of the user.",
|
473 |
+
).add(
|
474 |
+
"unban", "<username/id/reply>", "Unban a user from the group.", "unban @ForGo10God"
|
475 |
+
).add(
|
476 |
+
"kick",
|
477 |
+
"<username/id/reply> <reason>",
|
478 |
+
"Kick a user from the group.",
|
479 |
+
"kick @ForGo10God",
|
480 |
+
"You can also use dkick to delete the message of the user.",
|
481 |
+
).add(
|
482 |
+
"mute",
|
483 |
+
"<username/id/reply> <reason>",
|
484 |
+
"Mute a user in the group",
|
485 |
+
"mute @ForGo10God",
|
486 |
+
"You can also use dmute to delete the message of the user.",
|
487 |
+
).add(
|
488 |
+
"unmute", "<username/id/reply>", "Unmute a user in the group.", "unmute @ForGo10God"
|
489 |
+
).add(
|
490 |
+
"dmute",
|
491 |
+
"<username/id/reply>",
|
492 |
+
"Mute a user by deleting their new messages in the group.",
|
493 |
+
"dmute @ForGo10God",
|
494 |
+
"Need delete message permission for proper functioning.",
|
495 |
+
).add(
|
496 |
+
"undmute",
|
497 |
+
"<username/id/reply>",
|
498 |
+
"Unmute a user who's muted using 'dmute' command in the group.",
|
499 |
+
"undmute @ForGo10God",
|
500 |
+
).add(
|
501 |
+
"pin", "<reply>", "Pin the replied message in the group."
|
502 |
+
).add(
|
503 |
+
"unpin", "<reply>", "Unpin the replied pinned message in the group."
|
504 |
+
).add(
|
505 |
+
"zombies",
|
506 |
+
"clean",
|
507 |
+
"Finds the total number of deleted users present in that group and ban them.",
|
508 |
+
).info(
|
509 |
+
"Admin Menu"
|
510 |
+
).done()
|
HellBot/plugins/user/afk.py
ADDED
@@ -0,0 +1,157 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import random
|
3 |
+
import time
|
4 |
+
|
5 |
+
from pyrogram import Client, filters
|
6 |
+
from pyrogram.enums import MessageMediaType
|
7 |
+
from pyrogram.types import Message
|
8 |
+
|
9 |
+
from Hellbot.core import Config, db, hellbot
|
10 |
+
from Hellbot.functions.formatter import add_to_dict, get_from_dict, readable_time
|
11 |
+
|
12 |
+
from . import HelpMenu, custom_handler, group_only, on_message
|
13 |
+
|
14 |
+
afk_quotes = [
|
15 |
+
"🚶♂️ Taking a break, be back soon!",
|
16 |
+
"⏳ AFK - Away From the Keyboard momentarily.",
|
17 |
+
"🔜 Stepped away, but I'll return shortly.",
|
18 |
+
"👋 Gone for a moment, not forgotten.",
|
19 |
+
"🌿 Taking a breather, back in a bit.",
|
20 |
+
"📵 Away for a while, feel free to leave a message!",
|
21 |
+
"⏰ On a short break, back shortly.",
|
22 |
+
"🌈 Away from the screen, catching a breath.",
|
23 |
+
"💤 Offline for a moment, but still here in spirit.",
|
24 |
+
"🚀 Exploring the real world, back in a moment!",
|
25 |
+
"🍵 Taking a tea break, back shortly!",
|
26 |
+
"🌙 Resting my keyboard, back after a short nap.",
|
27 |
+
"🚶♀️ Stepping away for a moment of peace.",
|
28 |
+
"🎵 AFK but humming along, back shortly!",
|
29 |
+
"🌞 Taking a sunshine break, back soon!",
|
30 |
+
"🌊 Away, catching some waves of relaxation.",
|
31 |
+
"🚪 Temporarily closed, be back in a bit!",
|
32 |
+
"🌸 Taking a moment to smell the digital roses.",
|
33 |
+
"🍃 Stepped into the real world for a while.",
|
34 |
+
]
|
35 |
+
|
36 |
+
|
37 |
+
@on_message("afk")
|
38 |
+
async def afk(_, message: Message):
|
39 |
+
if await db.is_afk(message.from_user.id):
|
40 |
+
return await hellbot.delete(message, "🙄 𝖨'𝗆 𝖺𝗅𝗋𝖾𝖺𝖽𝗒 𝖠𝖥𝖪!")
|
41 |
+
|
42 |
+
media_type = None
|
43 |
+
media = None
|
44 |
+
|
45 |
+
if message.reply_to_message and message.reply_to_message.media:
|
46 |
+
if message.reply_to_message.media == MessageMediaType.ANIMATION:
|
47 |
+
media_type = "animation"
|
48 |
+
elif message.reply_to_message.media == MessageMediaType.AUDIO:
|
49 |
+
media_type = "audio"
|
50 |
+
elif message.reply_to_message.media == MessageMediaType.PHOTO:
|
51 |
+
media_type = "photo"
|
52 |
+
elif message.reply_to_message.media == MessageMediaType.STICKER:
|
53 |
+
media_type = "sticker"
|
54 |
+
elif message.reply_to_message.media == MessageMediaType.VIDEO:
|
55 |
+
media_type = "video"
|
56 |
+
elif message.reply_to_message.media == MessageMediaType.VOICE:
|
57 |
+
media_type = "voice"
|
58 |
+
|
59 |
+
media = await message.reply_to_message.forward(Config.LOGGER_ID)
|
60 |
+
|
61 |
+
reason = await hellbot.input(message)
|
62 |
+
reason = reason if reason else "Not specified"
|
63 |
+
|
64 |
+
await db.set_afk(
|
65 |
+
message.from_user.id, reason, media.id if media else None, media_type
|
66 |
+
)
|
67 |
+
await hellbot.delete(message, "🫡 𝖦𝗈𝗂𝗇𝗀 𝖠𝖥𝖪! 𝖲𝖾𝖾 𝗒𝖺'𝗅𝗅 𝗅𝖺𝗍𝖾𝗋.")
|
68 |
+
await hellbot.check_and_log(
|
69 |
+
"afk",
|
70 |
+
f"Going AFK! \n\n**Reason:** `{reason}`",
|
71 |
+
)
|
72 |
+
add_to_dict(Config.AFK_CACHE, [message.from_user.id, message.chat.id])
|
73 |
+
|
74 |
+
|
75 |
+
@custom_handler(filters.incoming & ~filters.bot & ~filters.service)
|
76 |
+
async def afk_watch(client: Client, message: Message):
|
77 |
+
afk_data = await db.get_afk(client.me.id)
|
78 |
+
if not afk_data:
|
79 |
+
return
|
80 |
+
|
81 |
+
if message.from_user.id == afk_data["user_id"]:
|
82 |
+
return
|
83 |
+
|
84 |
+
if message.chat.type in group_only:
|
85 |
+
if not message.mentioned:
|
86 |
+
return
|
87 |
+
|
88 |
+
afk_time = readable_time(round(time.time() - afk_data["time"]))
|
89 |
+
caption = f"**{random.choice(afk_quotes)}**\n\n**💫 𝖱𝖾𝖺𝗌𝗈𝗇:** {afk_data['reason']}\n**⏰ 𝖠𝖥𝖪 𝖥𝗋𝗈𝗆:** `{afk_time}`"
|
90 |
+
|
91 |
+
if afk_data["media_type"] == "animation":
|
92 |
+
media = await client.get_messages(Config.LOGGER_ID, afk_data["media"])
|
93 |
+
sent = await client.send_animation(
|
94 |
+
message.chat.id, media.animation.file_id, caption, True
|
95 |
+
)
|
96 |
+
|
97 |
+
elif afk_data["media_type"] in ["audio", "photo", "video", "voice"]:
|
98 |
+
sent = await client.copy_message(
|
99 |
+
message.chat.id,
|
100 |
+
Config.LOGGER_ID,
|
101 |
+
afk_data["media"],
|
102 |
+
caption,
|
103 |
+
reply_to_message_id=message.id,
|
104 |
+
)
|
105 |
+
|
106 |
+
elif afk_data["media_type"] == "sticker":
|
107 |
+
media = await client.get_messages(Config.LOGGER_ID, afk_data["media"])
|
108 |
+
await client.download_media(media, "afk.png")
|
109 |
+
sent = await message.reply_photo("afk.png", caption=caption)
|
110 |
+
os.remove("afk.png")
|
111 |
+
|
112 |
+
else:
|
113 |
+
sent = await message.reply_text(caption)
|
114 |
+
|
115 |
+
link = message.link if message.chat.type in group_only else "No DM Link"
|
116 |
+
|
117 |
+
await hellbot.check_and_log(
|
118 |
+
"afk",
|
119 |
+
f"{message.from_user.mention} mentioned you when you were AFK! \n\n**Link:** {link}",
|
120 |
+
)
|
121 |
+
try:
|
122 |
+
data = get_from_dict(Config.AFK_CACHE, [afk_data["user_id"], message.chat.id])
|
123 |
+
if data:
|
124 |
+
await client.delete_messages(message.chat.id, data)
|
125 |
+
add_to_dict(Config.AFK_CACHE, [afk_data["user_id"], message.chat.id], sent.id)
|
126 |
+
except KeyError:
|
127 |
+
add_to_dict(Config.AFK_CACHE, [afk_data["user_id"], message.chat.id], sent.id)
|
128 |
+
|
129 |
+
|
130 |
+
@custom_handler(filters.outgoing, 2)
|
131 |
+
async def remove_afk(_, message: Message):
|
132 |
+
if await db.is_afk(message.from_user.id):
|
133 |
+
if "afk" in message.text:
|
134 |
+
return
|
135 |
+
|
136 |
+
data = await db.get_afk(message.from_user.id)
|
137 |
+
total_afk_time = readable_time(round(time.time() - data["time"]))
|
138 |
+
|
139 |
+
hell = await message.reply_text(
|
140 |
+
f"🫡 **𝖡𝖺𝖼𝗄 𝗍𝗈 𝗏𝗂𝗋𝗍𝗎𝖺𝗅 𝗐𝗈𝗋𝗅𝖽! \n\n⌚ Was away for:** `{total_afk_time}`"
|
141 |
+
)
|
142 |
+
await message.delete()
|
143 |
+
|
144 |
+
await db.rm_afk(message.from_user.id)
|
145 |
+
await hellbot.check_and_log(
|
146 |
+
"afk",
|
147 |
+
f"Returned from AFK! \n\n**Time:** `{total_afk_time}`\n**Link:** {hell.link}",
|
148 |
+
)
|
149 |
+
|
150 |
+
|
151 |
+
HelpMenu("afk").add(
|
152 |
+
"afk",
|
153 |
+
"<reason>",
|
154 |
+
"Set your status as AFK. When someone mentions' you, the bot will tell them you're currently Offline! You can also use a media by replying to it.",
|
155 |
+
"afk good night!",
|
156 |
+
"To unset afk you can send a message to any chat and it'll automaticslly get disabled! You can use 'afk' in your message to bypass automatic disabling of afk.",
|
157 |
+
).info("Away From Keyboard").done()
|
HellBot/plugins/user/anime.py
ADDED
@@ -0,0 +1,184 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
|
3 |
+
from pyrogram.errors import ChatSendMediaForbidden
|
4 |
+
from pyrogram.types import Message
|
5 |
+
|
6 |
+
from Hellbot.core import hellbot
|
7 |
+
from Hellbot.functions.scraping import (
|
8 |
+
get_airing_info,
|
9 |
+
get_anilist_user_info,
|
10 |
+
get_anime_info,
|
11 |
+
get_character_info,
|
12 |
+
get_filler_info,
|
13 |
+
get_manga_info,
|
14 |
+
get_watch_order,
|
15 |
+
)
|
16 |
+
|
17 |
+
from . import HelpMenu, on_message
|
18 |
+
|
19 |
+
|
20 |
+
@on_message("anime", allow_stan=True)
|
21 |
+
async def anime(_, message: Message):
|
22 |
+
if len(message.command) < 2:
|
23 |
+
return await hellbot.delete(message, "Give me an anime name to search!")
|
24 |
+
|
25 |
+
query = await hellbot.input(message)
|
26 |
+
hell = await hellbot.edit(message, "Searching ...")
|
27 |
+
caption, photo = await get_anime_info(query)
|
28 |
+
|
29 |
+
try:
|
30 |
+
await message.reply_photo(photo, caption=caption)
|
31 |
+
await hell.delete()
|
32 |
+
except ChatSendMediaForbidden:
|
33 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
34 |
+
|
35 |
+
if os.path.exists(photo):
|
36 |
+
os.remove(photo)
|
37 |
+
|
38 |
+
|
39 |
+
@on_message("manga", allow_stan=True)
|
40 |
+
async def manga(_, message: Message):
|
41 |
+
if len(message.command) < 2:
|
42 |
+
return await hellbot.delete(message, "Give me a manga name to search!")
|
43 |
+
|
44 |
+
query = await hellbot.input(message)
|
45 |
+
hell = await hellbot.edit(message, "Searching ...")
|
46 |
+
caption, photo = await get_manga_info(query)
|
47 |
+
|
48 |
+
try:
|
49 |
+
await message.reply_photo(photo, caption=caption)
|
50 |
+
await hell.delete()
|
51 |
+
except ChatSendMediaForbidden:
|
52 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
53 |
+
|
54 |
+
if os.path.exists(photo):
|
55 |
+
os.remove(photo)
|
56 |
+
|
57 |
+
|
58 |
+
@on_message("character", allow_stan=True)
|
59 |
+
async def character(_, message: Message):
|
60 |
+
if len(message.command) < 2:
|
61 |
+
return await hellbot.delete(message, "Give me a character name to search!")
|
62 |
+
|
63 |
+
query = await hellbot.input(message)
|
64 |
+
hell = await hellbot.edit(message, "Searching ...")
|
65 |
+
caption, photo = await get_character_info(query)
|
66 |
+
|
67 |
+
try:
|
68 |
+
await message.reply_photo(photo, caption=caption)
|
69 |
+
await hell.delete()
|
70 |
+
except ChatSendMediaForbidden:
|
71 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
72 |
+
|
73 |
+
if os.path.exists(photo):
|
74 |
+
os.remove(photo)
|
75 |
+
|
76 |
+
|
77 |
+
@on_message("airing", allow_stan=True)
|
78 |
+
async def airing(_, message: Message):
|
79 |
+
if len(message.command) < 2:
|
80 |
+
return await hellbot.delete(message, "Give me an anime name to search!")
|
81 |
+
|
82 |
+
query = await hellbot.input(message)
|
83 |
+
hell = await hellbot.edit(message, "Searching ...")
|
84 |
+
caption, photo = await get_airing_info(query)
|
85 |
+
|
86 |
+
try:
|
87 |
+
await message.reply_photo(photo, caption=caption)
|
88 |
+
await hell.delete()
|
89 |
+
except ChatSendMediaForbidden:
|
90 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
91 |
+
|
92 |
+
if os.path.exists(photo):
|
93 |
+
os.remove(photo)
|
94 |
+
|
95 |
+
|
96 |
+
@on_message(["anilistuser", "aniuser"], allow_stan=True)
|
97 |
+
async def anilist_user(_, message: Message):
|
98 |
+
if len(message.command) < 2:
|
99 |
+
return await hellbot.delete(message, "Give me an anilist username to search!")
|
100 |
+
|
101 |
+
query = await hellbot.input(message)
|
102 |
+
hell = await hellbot.edit(message, "Searching ...")
|
103 |
+
caption, photo = await get_anilist_user_info(query)
|
104 |
+
|
105 |
+
try:
|
106 |
+
await message.reply_photo(photo, caption=caption)
|
107 |
+
await hell.delete()
|
108 |
+
except ChatSendMediaForbidden:
|
109 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
110 |
+
|
111 |
+
if os.path.exists(photo):
|
112 |
+
os.remove(photo)
|
113 |
+
|
114 |
+
|
115 |
+
@on_message(["filler", "canon"], allow_stan=True)
|
116 |
+
async def fillers(_, message: Message):
|
117 |
+
if len(message.command) < 2:
|
118 |
+
return await hellbot.delete(message, "Give me an anime name to search!")
|
119 |
+
|
120 |
+
query = await hellbot.input(message)
|
121 |
+
hell = await hellbot.edit(message, "Searching ...")
|
122 |
+
|
123 |
+
caption = await get_filler_info(query)
|
124 |
+
if caption == "":
|
125 |
+
return await hellbot.delete(hell, "No results found!")
|
126 |
+
|
127 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
128 |
+
|
129 |
+
|
130 |
+
@on_message("watchorder", allow_stan=True)
|
131 |
+
async def watch_order(_, message: Message):
|
132 |
+
if len(message.command) < 2:
|
133 |
+
return await hellbot.delete(message, "Give me an anime name to search!")
|
134 |
+
|
135 |
+
query = await hellbot.input(message)
|
136 |
+
hell = await hellbot.edit(message, "Searching ...")
|
137 |
+
|
138 |
+
caption = await get_watch_order(query)
|
139 |
+
if caption == "":
|
140 |
+
return await hellbot.delete(hell, "No results found!")
|
141 |
+
|
142 |
+
await hell.edit(caption, disable_web_page_preview=True)
|
143 |
+
|
144 |
+
|
145 |
+
HelpMenu("anime").add(
|
146 |
+
"anime",
|
147 |
+
"<name>",
|
148 |
+
"Get a detailed information about the mentioned anime.",
|
149 |
+
"anime one piece",
|
150 |
+
).add(
|
151 |
+
"manga",
|
152 |
+
"<name>",
|
153 |
+
"Get a detailed information about the mentioned manga.",
|
154 |
+
"manga one piece",
|
155 |
+
).add(
|
156 |
+
"character",
|
157 |
+
"<name>",
|
158 |
+
"Get a detailed information about the mentioned character.",
|
159 |
+
"character monkey d luffy",
|
160 |
+
).add(
|
161 |
+
"airing",
|
162 |
+
"<name>",
|
163 |
+
"Get a detailed airing information about the mentioned anime.",
|
164 |
+
"airing one piece",
|
165 |
+
).add(
|
166 |
+
"anilistuser",
|
167 |
+
"<username>",
|
168 |
+
"Get a detailed information about the mentioned anilist user.",
|
169 |
+
"anilistuser meizhellboy",
|
170 |
+
"You can also use 'aniuser' as alias",
|
171 |
+
).add(
|
172 |
+
"filler",
|
173 |
+
"<name>",
|
174 |
+
"Get the list of filler/canon episodes about the mentioned anime.",
|
175 |
+
"filler one piece",
|
176 |
+
"You can also use 'canon' as alias",
|
177 |
+
).add(
|
178 |
+
"watchorder",
|
179 |
+
"<name>",
|
180 |
+
"Get the watch order about the mentioned anime.",
|
181 |
+
"watchorder one piece",
|
182 |
+
).info(
|
183 |
+
"Anime Menu"
|
184 |
+
).done()
|
HellBot/plugins/user/antiflood.py
ADDED
@@ -0,0 +1,200 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import datetime
|
3 |
+
import time
|
4 |
+
|
5 |
+
from pyrogram import Client, filters
|
6 |
+
from pyrogram.types import ChatPermissions, Message
|
7 |
+
|
8 |
+
from Hellbot.core import Config, Symbols, db
|
9 |
+
from Hellbot.functions.utility import Flood
|
10 |
+
|
11 |
+
from . import HelpMenu, custom_handler, group_only, hellbot, on_message
|
12 |
+
|
13 |
+
|
14 |
+
@on_message("setflood", chat_type=group_only, admin_only=True, allow_stan=True)
|
15 |
+
async def setflood(client: Client, message: Message):
|
16 |
+
count = 5
|
17 |
+
mtime = 0
|
18 |
+
mode = "mute"
|
19 |
+
|
20 |
+
try:
|
21 |
+
time_data = "N/A"
|
22 |
+
if len(message.command) == 2:
|
23 |
+
count = int(message.command[1])
|
24 |
+
elif len(message.command) == 3:
|
25 |
+
count = int(message.command[1])
|
26 |
+
mode = message.command[2]
|
27 |
+
elif len(message.command) >= 4:
|
28 |
+
count = int(message.command[1])
|
29 |
+
mode = message.command[2]
|
30 |
+
time_data = message.command[3]
|
31 |
+
if time_data.endswith(("d", "day", "days")):
|
32 |
+
mtime = int(time_data.split("d")[0].strip()) * 24 * 60 * 60
|
33 |
+
elif time_data.endswith(("h", "hrs", "hour", "hours")):
|
34 |
+
mtime = int(time_data.split("h")[0].strip()) * 60 * 60
|
35 |
+
elif time_data.endswith(("m", "mins", "minute", "minutes")):
|
36 |
+
mtime = int(time_data.split("m")[0].strip()) * 60
|
37 |
+
else:
|
38 |
+
return await hellbot.error(
|
39 |
+
message,
|
40 |
+
"Please pass time in correct format!\n\nExample: 12d or 12h or 12m",
|
41 |
+
)
|
42 |
+
except Exception as e:
|
43 |
+
return await hellbot.error(message, str(e))
|
44 |
+
|
45 |
+
if mode.lower() not in ["mute", "kick", "ban"]:
|
46 |
+
return await hellbot.error(
|
47 |
+
message, "**Invalid mode! Choose one: **\n`mute`, `kick`, `ban`"
|
48 |
+
)
|
49 |
+
|
50 |
+
settings = {
|
51 |
+
"mode": mode,
|
52 |
+
"limit": count,
|
53 |
+
"time": mtime,
|
54 |
+
}
|
55 |
+
|
56 |
+
await db.set_flood((client.me.id, message.chat.id), settings)
|
57 |
+
Flood.updateSettings(client.me.id, message.chat.id, settings)
|
58 |
+
|
59 |
+
if count == 0:
|
60 |
+
return await hellbot.delete(message, "Antiflood disabled!")
|
61 |
+
|
62 |
+
await hellbot.delete(
|
63 |
+
message,
|
64 |
+
f"**Antiflood enabled!**\n\n**{Symbols.triangle_right} Mode:** `{mode}`\n**{Symbols.triangle_right} Limit:** `{count}`\n**{Symbols.triangle_right} Time:** `{time_data}`",
|
65 |
+
20,
|
66 |
+
)
|
67 |
+
|
68 |
+
|
69 |
+
@custom_handler(
|
70 |
+
filters.all
|
71 |
+
& filters.group
|
72 |
+
& filters.incoming
|
73 |
+
& ~filters.bot
|
74 |
+
& ~Config.AUTH_USERS
|
75 |
+
& ~filters.me
|
76 |
+
& ~filters.service
|
77 |
+
)
|
78 |
+
async def antiflood(client: Client, message: Message):
|
79 |
+
mode, mtime, limit = Flood.getSettings(client.me.id, message.chat.id)
|
80 |
+
|
81 |
+
if limit == 0:
|
82 |
+
return
|
83 |
+
if not Flood.check_client_chat(client.me.id, message.chat.id):
|
84 |
+
return
|
85 |
+
|
86 |
+
last_user, count = Flood.getLastUser(client.me.id, message.chat.id)
|
87 |
+
|
88 |
+
if last_user == message.from_user.id:
|
89 |
+
if (count + 1) >= limit:
|
90 |
+
template = (
|
91 |
+
"**🤫 𝖠𝗇𝗍𝗂𝖥𝗅𝗈𝗈𝖽 {mode}!!** \n\n"
|
92 |
+
"**{symbol} 𝖴𝗌𝖾𝗋:** `{mention}`\n"
|
93 |
+
"**{symbol} 𝖳𝗂𝗅𝗅 𝖣𝖺𝗍𝖾:** `🗓️ {till_date}`\n"
|
94 |
+
)
|
95 |
+
hell = await message.reply_text("Flood Detected!")
|
96 |
+
|
97 |
+
if mode == "mute":
|
98 |
+
permission = ChatPermissions(can_send_messages=False)
|
99 |
+
until_date = datetime.datetime.fromtimestamp(time.time() + mtime)
|
100 |
+
try:
|
101 |
+
await client.restrict_chat_member(
|
102 |
+
message.chat.id,
|
103 |
+
message.from_user.id,
|
104 |
+
permission,
|
105 |
+
until_date,
|
106 |
+
)
|
107 |
+
except Exception as e:
|
108 |
+
return await hellbot.error(
|
109 |
+
hell, f"__Error in Antiflood while trying to mute!__\n{str(e)}"
|
110 |
+
)
|
111 |
+
|
112 |
+
Flood.updateFlood(
|
113 |
+
client.me.id, message.chat.id, message.from_user.id, 0
|
114 |
+
)
|
115 |
+
till_date = "Forever" if mtime == 0 else until_date.ctime()
|
116 |
+
|
117 |
+
return await hell.edit(
|
118 |
+
template.format(
|
119 |
+
mode=mode.title(),
|
120 |
+
symbol=Symbols.triangle_right,
|
121 |
+
mention=message.from_user.mention,
|
122 |
+
till_date=till_date,
|
123 |
+
)
|
124 |
+
)
|
125 |
+
|
126 |
+
elif mode == "kick":
|
127 |
+
try:
|
128 |
+
await client.ban_chat_member(message.chat.id, message.from_user.id)
|
129 |
+
except Exception as e:
|
130 |
+
return await hellbot.error(
|
131 |
+
hell, f"__Error in Antiflood while trying to kick!__\n{str(e)}"
|
132 |
+
)
|
133 |
+
|
134 |
+
await hell.edit(
|
135 |
+
template.format(
|
136 |
+
mode=mode.title(),
|
137 |
+
symbol=Symbols.triangle_right,
|
138 |
+
mention=message.from_user.mention,
|
139 |
+
till_date="Kicked Users can join back after 5 seconds!",
|
140 |
+
)
|
141 |
+
)
|
142 |
+
Flood.updateFlood(
|
143 |
+
client.me.id, message.chat.id, message.from_user.id, 0
|
144 |
+
)
|
145 |
+
await asyncio.sleep(5)
|
146 |
+
await client.unban_chat_member(message.chat.id, message.from_user.id)
|
147 |
+
return
|
148 |
+
|
149 |
+
elif mode == "ban":
|
150 |
+
until_date = datetime.datetime.fromtimestamp(time.time() + mtime)
|
151 |
+
try:
|
152 |
+
await client.ban_chat_member(
|
153 |
+
message.chat.id,
|
154 |
+
message.from_user.id,
|
155 |
+
until_date,
|
156 |
+
)
|
157 |
+
except Exception as e:
|
158 |
+
return await hellbot.error(
|
159 |
+
hell, f"__Error in Antiflood while trying to ban!__\n{str(e)}"
|
160 |
+
)
|
161 |
+
|
162 |
+
Flood.updateFlood(
|
163 |
+
client.me.id, message.chat.id, message.from_user.id, 0
|
164 |
+
)
|
165 |
+
till_date = "Forever" if mtime == 0 else until_date.ctime()
|
166 |
+
|
167 |
+
return await hell.edit(
|
168 |
+
template.format(
|
169 |
+
mode=mode.title(),
|
170 |
+
symbol=Symbols.triangle_right,
|
171 |
+
mention=message.from_user.mention,
|
172 |
+
till_date=till_date,
|
173 |
+
)
|
174 |
+
)
|
175 |
+
else:
|
176 |
+
return
|
177 |
+
else:
|
178 |
+
count += 1
|
179 |
+
Flood.updateFlood(
|
180 |
+
client.me.id, message.chat.id, message.from_user.id, count
|
181 |
+
)
|
182 |
+
return
|
183 |
+
else:
|
184 |
+
Flood.updateFlood(client.me.id, message.chat.id, message.from_user.id, 1)
|
185 |
+
|
186 |
+
|
187 |
+
HelpMenu("antiflood").add(
|
188 |
+
"setflood",
|
189 |
+
"<limit> <mode> <time>",
|
190 |
+
"Set antiflood in the chat! All arguments are optional, bydefault limit is 5 and mode is permanent mute.",
|
191 |
+
"setflood 10 ban 1d",
|
192 |
+
"Mode can be mute, kick or ban. Time can be xd (days), xh (hours) or xm (minutes) where x is number.",
|
193 |
+
).add(
|
194 |
+
"setflood 0",
|
195 |
+
None,
|
196 |
+
"Disable antiflood in the chat!",
|
197 |
+
"setflood 0",
|
198 |
+
).info(
|
199 |
+
"Control Flood in the chat!"
|
200 |
+
).done()
|
HellBot/plugins/user/archiver.py
ADDED
@@ -0,0 +1,99 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import time
|
3 |
+
import zipfile
|
4 |
+
|
5 |
+
from pyrogram.types import Message
|
6 |
+
|
7 |
+
from Hellbot.functions.formatter import readable_time
|
8 |
+
from Hellbot.functions.tools import get_files_from_directory, progress
|
9 |
+
|
10 |
+
from . import Config, HelpMenu, hellbot, on_message
|
11 |
+
|
12 |
+
|
13 |
+
@on_message("zip", allow_stan=True)
|
14 |
+
async def zip_files(_, message: Message):
|
15 |
+
if not message.reply_to_message:
|
16 |
+
return await hellbot.delete(message, "Reply to a message to zip it.")
|
17 |
+
|
18 |
+
media = message.reply_to_message.media
|
19 |
+
if not media:
|
20 |
+
return await hellbot.delete(message, "Reply to a media message to zip it.")
|
21 |
+
|
22 |
+
hell = await hellbot.edit(message, "`Zipping...`")
|
23 |
+
start = time.time()
|
24 |
+
download_path = await message.reply_to_message.download(
|
25 |
+
f"{Config.TEMP_DIR}temp_{round(time.time())}",
|
26 |
+
progress=progress,
|
27 |
+
progress_args=(hell, start, "📦 Zipping"),
|
28 |
+
)
|
29 |
+
|
30 |
+
zip_path = Config.TEMP_DIR + f"zipped_{int(time.time())}.zip"
|
31 |
+
with zipfile.ZipFile(zip_path, "w", zipfile.ZIP_DEFLATED) as zip_file:
|
32 |
+
zip_file.write(download_path)
|
33 |
+
|
34 |
+
await hellbot.delete(hell, "Zipped Successfully.")
|
35 |
+
await message.reply_document(
|
36 |
+
zip_path,
|
37 |
+
caption=f"**Zipped in {readable_time(time.time() - start)}.**",
|
38 |
+
progress=progress,
|
39 |
+
progress_args=(hell, start, "⬆️ Uploading"),
|
40 |
+
)
|
41 |
+
|
42 |
+
os.remove(zip_path)
|
43 |
+
os.remove(download_path)
|
44 |
+
|
45 |
+
|
46 |
+
@on_message("unzip", allow_stan=True)
|
47 |
+
async def unzip_file(_, message: Message):
|
48 |
+
if not message.reply_to_message:
|
49 |
+
return await hellbot.delete(message, "Reply to a message to unzip it.")
|
50 |
+
|
51 |
+
media = message.reply_to_message.media
|
52 |
+
if not media:
|
53 |
+
return await hellbot.delete(message, "Reply to a media message to unzip it.")
|
54 |
+
|
55 |
+
hell = await hellbot.edit(message, "`Unzipping...`")
|
56 |
+
start = time.time()
|
57 |
+
download_path = await message.reply_to_message.download(
|
58 |
+
f"{Config.TEMP_DIR}temp_{round(time.time())}",
|
59 |
+
progress=progress,
|
60 |
+
progress_args=(hell, start, "📦 Unzipping"),
|
61 |
+
)
|
62 |
+
|
63 |
+
with zipfile.ZipFile(download_path, "r") as zip_file:
|
64 |
+
if not os.path.isdir(Config.TEMP_DIR + "unzipped/"):
|
65 |
+
os.mkdir(Config.TEMP_DIR + "unzipped/")
|
66 |
+
zip_file.extractall(Config.TEMP_DIR + "unzipped/")
|
67 |
+
|
68 |
+
await hellbot.delete(hell, "Unzipped Successfully.")
|
69 |
+
files = await get_files_from_directory(Config.TEMP_DIR + "unzipped/")
|
70 |
+
|
71 |
+
for file in files:
|
72 |
+
if os.path.exists(file):
|
73 |
+
try:
|
74 |
+
await message.reply_document(
|
75 |
+
file,
|
76 |
+
caption=f"**Unzipped {os.path.basename(file)}.**",
|
77 |
+
force_document=True,
|
78 |
+
progress=progress,
|
79 |
+
progress_args=(hell, start, "⬆️ Uploading"),
|
80 |
+
)
|
81 |
+
except Exception as e:
|
82 |
+
await message.reply_text(f"**{file}:** `{e}`")
|
83 |
+
continue
|
84 |
+
os.remove(file)
|
85 |
+
|
86 |
+
os.remove(download_path)
|
87 |
+
|
88 |
+
|
89 |
+
HelpMenu("archiver").add(
|
90 |
+
"zip",
|
91 |
+
"<reply to a media>",
|
92 |
+
"Zip the replied media and upload it in the chat.",
|
93 |
+
).add(
|
94 |
+
"unzip",
|
95 |
+
"<reply to a zip file>",
|
96 |
+
"Unzip the replied zip file and upload it in the chat.",
|
97 |
+
).info(
|
98 |
+
"Manage Archives"
|
99 |
+
).done()
|
HellBot/plugins/user/autopost.py
ADDED
@@ -0,0 +1,143 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram import Client, filters
|
2 |
+
from pyrogram.types import Message
|
3 |
+
|
4 |
+
from Hellbot.core import Symbols
|
5 |
+
|
6 |
+
from . import HelpMenu, custom_handler, db, group_n_channel, hellbot, on_message
|
7 |
+
|
8 |
+
|
9 |
+
@on_message("autopost", chat_type=group_n_channel, allow_stan=True)
|
10 |
+
async def autopost(client: Client, message: Message):
|
11 |
+
if len(message.command) != 2:
|
12 |
+
return await hellbot.delete(
|
13 |
+
message, "Wrong usage of command.\nCheck help menu for more info."
|
14 |
+
)
|
15 |
+
|
16 |
+
hell = await hellbot.edit(message, "Starting Autopost in this group/channel...")
|
17 |
+
|
18 |
+
post_from = message.command[1]
|
19 |
+
_chat = await client.get_chat(post_from)
|
20 |
+
|
21 |
+
if not _chat:
|
22 |
+
return await hellbot.delete(hell, "Invalid chat/channel id.")
|
23 |
+
|
24 |
+
if _chat.type not in group_n_channel:
|
25 |
+
return await hellbot.delete(
|
26 |
+
hell, "You can only autopost in groups and channels."
|
27 |
+
)
|
28 |
+
|
29 |
+
if _chat.id == message.chat.id:
|
30 |
+
return await hellbot.delete(
|
31 |
+
hell, "You can't autopost in the same group/channel."
|
32 |
+
)
|
33 |
+
|
34 |
+
if _chat.id in await db.is_autopost(client.me.id, _chat.id, message.chat.id):
|
35 |
+
return await hellbot.delete(
|
36 |
+
hell, "This group/channel is already in autopost list."
|
37 |
+
)
|
38 |
+
|
39 |
+
await db.set_autopost(client.me.id, _chat.id, message.chat.id)
|
40 |
+
|
41 |
+
await hellbot.delete(
|
42 |
+
hell, f"Autopost started from {_chat.title} to {message.chat.title}."
|
43 |
+
)
|
44 |
+
await hellbot.check_and_log(
|
45 |
+
"autopost start",
|
46 |
+
f"**AutoPost From:** {_chat.title} \n**AutoPost To:** {message.chat.title}\n**AutoPost By:** {client.me.mention}",
|
47 |
+
)
|
48 |
+
|
49 |
+
|
50 |
+
@on_message("stopautopost", chat_type=group_n_channel, allow_stan=True)
|
51 |
+
async def stop_autopost(client: Client, message: Message):
|
52 |
+
if len(message.command) != 2:
|
53 |
+
return await hellbot.delete(
|
54 |
+
message, "Wrong usage of command.\nCheck help menu for more info."
|
55 |
+
)
|
56 |
+
|
57 |
+
hell = await hellbot.edit(message, "Stopping Autopost in this group/channel...")
|
58 |
+
|
59 |
+
post_from = message.command[1]
|
60 |
+
_chat = await client.get_chat(post_from)
|
61 |
+
|
62 |
+
if not _chat:
|
63 |
+
return await hellbot.delete(hell, "Invalid chat/channel id.")
|
64 |
+
|
65 |
+
if _chat.type not in group_n_channel:
|
66 |
+
return await hellbot.delete(
|
67 |
+
hell, "You can only autopost in groups and channels."
|
68 |
+
)
|
69 |
+
|
70 |
+
if _chat.id not in await db.is_autopost(client.me.id, _chat.id, message.chat.id):
|
71 |
+
return await hellbot.delete(hell, "This group/channel is not in autopost list.")
|
72 |
+
|
73 |
+
await db.rm_autopost(client.me.id, _chat.id, message.chat.id)
|
74 |
+
|
75 |
+
await hellbot.delete(
|
76 |
+
hell, f"Autopost stopped from {_chat.title} to {message.chat.title}."
|
77 |
+
)
|
78 |
+
await hellbot.check_and_log(
|
79 |
+
"autopost stop",
|
80 |
+
f"**AutoPost From:** {_chat.title} \n**AutoPost To:** {message.chat.title}\n**AutoPost By:** {client.me.mention}",
|
81 |
+
)
|
82 |
+
|
83 |
+
|
84 |
+
@on_message("autoposts", chat_type=group_n_channel, allow_stan=True)
|
85 |
+
async def autoposts(client: Client, message: Message):
|
86 |
+
hell = await hellbot.edit(message, "Getting autopost list...")
|
87 |
+
|
88 |
+
data = await db.get_all_autoposts(client.me.id)
|
89 |
+
if not data:
|
90 |
+
return await hellbot.delete(hell, "No autoposts found.")
|
91 |
+
|
92 |
+
text = f"**𝖠𝖼𝗍𝗂𝗏𝖾 𝖠𝗎𝗍𝗈𝗉𝗈𝗌𝗍𝗌 𝖿𝗈𝗋: {client.me.mention}**\n\n"
|
93 |
+
for i in data:
|
94 |
+
from_chat = await client.get_chat(i["from_channel"])
|
95 |
+
to_chat = await client.get_chat(i["to_channel"])
|
96 |
+
|
97 |
+
from_chat_name = (
|
98 |
+
f"{from_chat.title} [{from_chat.id}]" if from_chat else i["from_channel"]
|
99 |
+
)
|
100 |
+
to_chat_name = f"{to_chat.title} [{to_chat.id}]" if to_chat else i["to_channel"]
|
101 |
+
|
102 |
+
text += f" {Symbols.anchor} **From:** {from_chat_name}\n"
|
103 |
+
text += f" {Symbols.anchor} **To:** {to_chat_name}\n"
|
104 |
+
text += f" {Symbols.anchor} **Date:** {i['date']}\n\n"
|
105 |
+
|
106 |
+
await hellbot.edit(hell, text)
|
107 |
+
|
108 |
+
|
109 |
+
@custom_handler(filters.incoming & filters.group & filters.channel & ~filters.service)
|
110 |
+
async def handle_autopost(client: Client, message: Message):
|
111 |
+
if not await db.is_autopost(client.me.id, message.chat.id):
|
112 |
+
return
|
113 |
+
|
114 |
+
data = await db.get_autopost(client.me.id, message.chat.id)
|
115 |
+
if not data:
|
116 |
+
return
|
117 |
+
|
118 |
+
from_chat = await client.get_chat(data["from_channel"])
|
119 |
+
if not from_chat:
|
120 |
+
return
|
121 |
+
|
122 |
+
if message.chat.id != data["to_channel"]:
|
123 |
+
return
|
124 |
+
|
125 |
+
await message.copy(int(data["to_channel"]))
|
126 |
+
|
127 |
+
|
128 |
+
HelpMenu("autopost").add(
|
129 |
+
"autopost",
|
130 |
+
"<channel id>",
|
131 |
+
"Start autoposting in current group/channel from the mentioned chatid/username of channel.",
|
132 |
+
"autopost @Its_HellBot",
|
133 |
+
"This module will post all incoming post from the target channel to the current chat without forward tag!",
|
134 |
+
).add(
|
135 |
+
"stopautopost",
|
136 |
+
"<channel id>",
|
137 |
+
"Stops autoposting in current chroup/channel from the mentioned chatid/username of channel.",
|
138 |
+
"stopautopost @Its_HellBot",
|
139 |
+
).add(
|
140 |
+
"autoposts", None, "Get all active autoposts!", "autoposts"
|
141 |
+
).info(
|
142 |
+
"AutoPost Module"
|
143 |
+
).done()
|
HellBot/plugins/user/blacklist.py
ADDED
@@ -0,0 +1,78 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import re
|
2 |
+
|
3 |
+
from pyrogram import Client, filters
|
4 |
+
from pyrogram.types import Message
|
5 |
+
|
6 |
+
from Hellbot.core import Config, Symbols
|
7 |
+
from Hellbot.functions.utility import BList
|
8 |
+
|
9 |
+
from . import HelpMenu, custom_handler, db, hellbot, on_message
|
10 |
+
|
11 |
+
|
12 |
+
@on_message("blacklist", admin_only=True, allow_stan=True)
|
13 |
+
async def blacklist(client: Client, message: Message):
|
14 |
+
if len(message.command) < 2:
|
15 |
+
return await hellbot.delete(message, "Give me something to blacklist.")
|
16 |
+
|
17 |
+
text = await hellbot.input(message)
|
18 |
+
|
19 |
+
if await db.is_blacklist(client.me.id, message.chat.id, text):
|
20 |
+
return await hellbot.delete(message, f"**Already blacklisted** `{text}`")
|
21 |
+
|
22 |
+
await BList.addBlacklist(client.me.id, message.chat.id, text)
|
23 |
+
await hellbot.delete(message, f"**Blacklisted:** `{text}`")
|
24 |
+
|
25 |
+
|
26 |
+
@on_message("unblacklist", admin_only=True, allow_stan=True)
|
27 |
+
async def unblacklist(client: Client, message: Message):
|
28 |
+
if len(message.command) < 2:
|
29 |
+
return await hellbot.delete(message, "Give me something to unblacklist.")
|
30 |
+
|
31 |
+
text = await hellbot.input(message)
|
32 |
+
|
33 |
+
if not await db.is_blacklist(client.me.id, message.chat.id, text):
|
34 |
+
return await hellbot.delete(message, f"`{text}` does not exist in blacklist.")
|
35 |
+
|
36 |
+
await BList.rmBlacklist(client.me.id, message.chat.id, text)
|
37 |
+
await hellbot.delete(message, f"**Unblacklisted:** `{text}`")
|
38 |
+
|
39 |
+
|
40 |
+
@on_message("blacklists", admin_only=True, allow_stan=True)
|
41 |
+
async def blacklists(client: Client, message: Message):
|
42 |
+
blacklists = await db.get_all_blacklists(client.me.id, message.chat.id)
|
43 |
+
|
44 |
+
if not blacklists:
|
45 |
+
return await hellbot.delete(message, "No blacklists found.")
|
46 |
+
|
47 |
+
text = f"**{Symbols.bullet} 𝖡𝗅𝖺𝖼𝗄𝗅𝗂𝗌𝗍𝗌 𝗂𝗇 {message.chat.title}:**\n\n"
|
48 |
+
for i in blacklists:
|
49 |
+
text += f" {Symbols.anchor} `{i}`\n"
|
50 |
+
|
51 |
+
await hellbot.edit(message, text)
|
52 |
+
|
53 |
+
|
54 |
+
@custom_handler(filters.text & filters.incoming & ~Config.AUTH_USERS & ~filters.service)
|
55 |
+
async def handle_blacklists(client: Client, message: Message):
|
56 |
+
if BList.check_client_chat(client.me.id, message.chat.id):
|
57 |
+
blacklists = BList.getBlacklists(client.me.id, message.chat.id)
|
58 |
+
for blacklist in blacklists:
|
59 |
+
pattern = r"( |^|[^\w])" + re.escape(blacklist) + r"( |$|[^\w])"
|
60 |
+
if re.search(pattern, message.text, flags=re.IGNORECASE):
|
61 |
+
try:
|
62 |
+
await message.delete()
|
63 |
+
except Exception:
|
64 |
+
await BList.rmBlacklist(client.me.id, message.chat.id, blacklist)
|
65 |
+
|
66 |
+
|
67 |
+
HelpMenu("blacklist").add(
|
68 |
+
"blacklist",
|
69 |
+
"<text>",
|
70 |
+
"Add the text to blacklist. If the text is sent in the chat it will be deleted.",
|
71 |
+
"blacklist hello",
|
72 |
+
).add(
|
73 |
+
"unblacklist", "<text>", "Remove the text from blacklist.", "unblacklist hello"
|
74 |
+
).add(
|
75 |
+
"blacklists", None, "List all the blacklisted words in the chat.", "blacklists"
|
76 |
+
).info(
|
77 |
+
"Blacklist Module"
|
78 |
+
).done()
|
HellBot/plugins/user/bot.py
ADDED
@@ -0,0 +1,164 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import random
|
3 |
+
import time
|
4 |
+
|
5 |
+
from pyrogram import Client
|
6 |
+
from pyrogram.types import Message
|
7 |
+
|
8 |
+
from Hellbot import START_TIME
|
9 |
+
from Hellbot.core import ENV
|
10 |
+
from Hellbot.functions.formatter import readable_time
|
11 |
+
from Hellbot.functions.images import generate_alive_image
|
12 |
+
from Hellbot.functions.templates import alive_template, ping_template
|
13 |
+
|
14 |
+
from . import Config, HelpMenu, db, hellbot, on_message
|
15 |
+
|
16 |
+
|
17 |
+
@on_message("alive", allow_stan=True)
|
18 |
+
async def alive(client: Client, message: Message):
|
19 |
+
hell = await hellbot.edit(message, "Processing ...")
|
20 |
+
|
21 |
+
img = await db.get_env(ENV.alive_pic)
|
22 |
+
if not img:
|
23 |
+
if message.from_user.photo:
|
24 |
+
user_pfp = await client.download_media(message.from_user.photo.big_file_id)
|
25 |
+
del_path = True
|
26 |
+
else:
|
27 |
+
user_pfp = "./Hellbot/resources/images/hellbot_logo.png"
|
28 |
+
del_path = False
|
29 |
+
img = [
|
30 |
+
generate_alive_image(
|
31 |
+
message.from_user.first_name, user_pfp, del_path, Config.FONT_PATH
|
32 |
+
)
|
33 |
+
]
|
34 |
+
else:
|
35 |
+
img = img.split(" ")
|
36 |
+
|
37 |
+
img = random.choice(img)
|
38 |
+
uptime = readable_time(time.time() - START_TIME)
|
39 |
+
caption = await alive_template(client.me.first_name, uptime)
|
40 |
+
|
41 |
+
if img.endswith(".mp4"):
|
42 |
+
await message.reply_video(img, caption=caption)
|
43 |
+
else:
|
44 |
+
await message.reply_photo(img, caption=caption)
|
45 |
+
await hell.delete()
|
46 |
+
|
47 |
+
try:
|
48 |
+
os.remove(img)
|
49 |
+
except:
|
50 |
+
pass
|
51 |
+
|
52 |
+
|
53 |
+
@on_message("ping", allow_stan=True)
|
54 |
+
async def ping(client: Client, message: Message):
|
55 |
+
start_time = time.time()
|
56 |
+
hell = await hellbot.edit(message, "**Pong ~**")
|
57 |
+
uptime = readable_time(time.time() - START_TIME)
|
58 |
+
img = await db.get_env(ENV.ping_pic)
|
59 |
+
end_time = time.time()
|
60 |
+
speed = end_time - start_time
|
61 |
+
caption = await ping_template(round(speed, 3), uptime, client.me.mention)
|
62 |
+
if img:
|
63 |
+
img = random.choice(img.split(" "))
|
64 |
+
if img.endswith(".mp4"):
|
65 |
+
await message.reply_video(
|
66 |
+
img,
|
67 |
+
caption=caption,
|
68 |
+
)
|
69 |
+
else:
|
70 |
+
await message.reply_photo(
|
71 |
+
img,
|
72 |
+
caption=caption,
|
73 |
+
)
|
74 |
+
await hell.delete()
|
75 |
+
return
|
76 |
+
await hellbot.edit(hell, caption, no_link_preview=True)
|
77 |
+
|
78 |
+
|
79 |
+
@on_message("history", allow_stan=True)
|
80 |
+
async def history(client: Client, message: Message):
|
81 |
+
if not message.reply_to_message:
|
82 |
+
if len(message.command) < 2:
|
83 |
+
return await hellbot.delete(
|
84 |
+
message, "Either reply to an user or give me a username to get history."
|
85 |
+
)
|
86 |
+
try:
|
87 |
+
user = await client.get_users(message.command[1])
|
88 |
+
except Exception as e:
|
89 |
+
return await hellbot.error(message, f"`{str(e)}`")
|
90 |
+
else:
|
91 |
+
user = message.reply_to_message.from_user
|
92 |
+
|
93 |
+
hell = await hellbot.edit(message, "Processing ...")
|
94 |
+
|
95 |
+
try:
|
96 |
+
response = await client.ask("@SangMata_BOT", f"{user.id}", timeout=60)
|
97 |
+
except Exception as e:
|
98 |
+
return await hellbot.error(hell, f"`{str(e)}`")
|
99 |
+
|
100 |
+
if "you have used up your quota for today" in response.text:
|
101 |
+
return await hellbot.delete(
|
102 |
+
hell,
|
103 |
+
f"Your quota of using SangMata Bot is over. Wait till 00:00 UTC before using it again.",
|
104 |
+
)
|
105 |
+
|
106 |
+
try:
|
107 |
+
await response.delete()
|
108 |
+
await response.request.delete()
|
109 |
+
except:
|
110 |
+
pass
|
111 |
+
|
112 |
+
await hellbot.edit(hell, response.text)
|
113 |
+
|
114 |
+
|
115 |
+
@on_message("template", allow_stan=True)
|
116 |
+
async def template_example(_, message: Message):
|
117 |
+
variable_names = list(Config.TEMPLATES.keys())
|
118 |
+
if len(message.command) < 2:
|
119 |
+
return await hellbot.delete(
|
120 |
+
message,
|
121 |
+
f"__Give a template name to get template example.__\n\n**Available Templates:**\n`{'`, `'.join(variable_names)}`",
|
122 |
+
30,
|
123 |
+
)
|
124 |
+
|
125 |
+
if message.command[1].upper() not in variable_names:
|
126 |
+
return await hellbot.delete(
|
127 |
+
message,
|
128 |
+
f"__Invalid template name:__ `{message.command[1].upper()}`\n\n**Available Templates:**\n`{'`, `'.join(variable_names)}`",
|
129 |
+
30,
|
130 |
+
)
|
131 |
+
|
132 |
+
await hellbot.edit(
|
133 |
+
message,
|
134 |
+
f"**{message.command[1].upper()} Template Example:**\n\n```{Config.TEMPLATES[message.command[1].upper()]}```"
|
135 |
+
)
|
136 |
+
|
137 |
+
|
138 |
+
HelpMenu("bot").add(
|
139 |
+
"alive",
|
140 |
+
None,
|
141 |
+
"Get the alive message of the bot.",
|
142 |
+
"alive",
|
143 |
+
"You can also customize alive message with suitable variables for it.",
|
144 |
+
).add(
|
145 |
+
"ping",
|
146 |
+
None,
|
147 |
+
"Check the ping speed and uptime of bot.",
|
148 |
+
"ping",
|
149 |
+
"You can also customize ping message by adding a media to it.",
|
150 |
+
).add(
|
151 |
+
"history",
|
152 |
+
"<reply to user>/<username/id>",
|
153 |
+
"Get the username, name history of an user.",
|
154 |
+
"history @ForGo10_God",
|
155 |
+
"This command uses SangMata Bot to get the history.",
|
156 |
+
).add(
|
157 |
+
"template",
|
158 |
+
"<template name>",
|
159 |
+
"Get the example of a template.",
|
160 |
+
"template alive_templates",
|
161 |
+
"This command is used to get the example of a template and a list of customisable templates.",
|
162 |
+
).info(
|
163 |
+
"Alive Menu"
|
164 |
+
).done()
|
HellBot/plugins/user/carbon.py
ADDED
@@ -0,0 +1,73 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import asyncio
|
2 |
+
import os
|
3 |
+
|
4 |
+
from pyrogram.types import Message
|
5 |
+
|
6 |
+
from Hellbot.functions.driver import Driver
|
7 |
+
|
8 |
+
from . import HelpMenu, hellbot, on_message
|
9 |
+
|
10 |
+
|
11 |
+
@on_message("carbon", allow_stan=True)
|
12 |
+
async def carbon(_, message: Message):
|
13 |
+
if len(message.command) < 2:
|
14 |
+
return await hellbot.delete(message, "Give me some code to make carbon.")
|
15 |
+
|
16 |
+
code = await hellbot.input(message)
|
17 |
+
hell = await hellbot.edit(message, "**[ 50% ]** __Making carbon...__")
|
18 |
+
|
19 |
+
driver, resp = Driver.get()
|
20 |
+
if not driver:
|
21 |
+
return await hellbot.error(message, resp)
|
22 |
+
|
23 |
+
await hell.edit("**[ 75% ]** __Making carbon...__")
|
24 |
+
image = await Driver.generate_carbon(driver, code)
|
25 |
+
await asyncio.sleep(4)
|
26 |
+
|
27 |
+
await hell.edit("**[ 100% ]** __Uploading carbon...__")
|
28 |
+
Driver.close(driver)
|
29 |
+
|
30 |
+
await hell.reply_photo(image, caption=f"**𝖢𝖺𝗋𝖻𝗈𝗇𝖾𝖽:**\n`{code}`")
|
31 |
+
await hell.delete()
|
32 |
+
os.remove(image)
|
33 |
+
|
34 |
+
|
35 |
+
@on_message("karbon", allow_stan=True)
|
36 |
+
async def karbon(_, message: Message):
|
37 |
+
if len(message.command) < 2:
|
38 |
+
return await hellbot.delete(message, "Give me some code to make karbon.")
|
39 |
+
|
40 |
+
code = await hellbot.input(message)
|
41 |
+
hell = await hellbot.edit(message, "**[ 50% ]** __Making karbon...__")
|
42 |
+
|
43 |
+
driver, resp = Driver.get()
|
44 |
+
if not driver:
|
45 |
+
return await hellbot.error(message, resp)
|
46 |
+
|
47 |
+
await hell.edit("**[ 75% ]** __Making karbon...__")
|
48 |
+
image = await Driver.generate_carbon(driver, code, True)
|
49 |
+
await asyncio.sleep(4)
|
50 |
+
|
51 |
+
await hell.edit("**[ 100% ]** __Uploading karbon...__")
|
52 |
+
Driver.close(driver)
|
53 |
+
|
54 |
+
await hell.reply_photo(image, caption=f"**𝖢𝖺𝗋𝖻𝗈𝗇𝖾𝖽:**\n`{code}`")
|
55 |
+
await hell.delete()
|
56 |
+
os.remove(image)
|
57 |
+
|
58 |
+
|
59 |
+
HelpMenu("carbon").add(
|
60 |
+
"carbon",
|
61 |
+
"<code snippet>",
|
62 |
+
"Makes carbon of given code snippet.",
|
63 |
+
"carbon print('Hello World!')",
|
64 |
+
"The style is fixed and cannot be changed.",
|
65 |
+
).add(
|
66 |
+
"karbon",
|
67 |
+
"<code snippet>",
|
68 |
+
"Makes carbon of given code snippet.",
|
69 |
+
"karbon print('Hello World!')",
|
70 |
+
"The style is randomly choosed.",
|
71 |
+
).info(
|
72 |
+
"Carbon is a code snippet sharing service. You can make carbon of your code and share it with others."
|
73 |
+
).done()
|
HellBot/plugins/user/climate.py
ADDED
@@ -0,0 +1,140 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from pyrogram.types import Message
|
2 |
+
|
3 |
+
from Hellbot.core import ENV
|
4 |
+
from Hellbot.functions.driver import Climate
|
5 |
+
from Hellbot.functions.formatter import subscript, superscript
|
6 |
+
from Hellbot.functions.templates import airpollution_templates, climate_templates
|
7 |
+
|
8 |
+
from . import HelpMenu, db, hellbot, on_message
|
9 |
+
|
10 |
+
|
11 |
+
@on_message("climate", allow_stan=True)
|
12 |
+
async def climate(_, message: Message):
|
13 |
+
if len(message.command) < 2:
|
14 |
+
city = "Delhi"
|
15 |
+
else:
|
16 |
+
city = await hellbot.input(message)
|
17 |
+
|
18 |
+
hell = await hellbot.edit(message, f"**Fetching climate data for `{city}`**")
|
19 |
+
|
20 |
+
apiKey = await db.get_env(ENV.climate_api)
|
21 |
+
if not apiKey:
|
22 |
+
return await hellbot.delete(
|
23 |
+
hell, f"**Climate API not found!** Setup `{ENV.climate_api}` first."
|
24 |
+
)
|
25 |
+
|
26 |
+
data = await Climate.fetchWeather(city, apiKey)
|
27 |
+
if not data:
|
28 |
+
return await hellbot.delete(hell, f"**Climate data not found for `{city}`**")
|
29 |
+
|
30 |
+
city_name = data["name"]
|
31 |
+
country = Climate.getCountry(data["sys"]["country"])
|
32 |
+
weather = data["weather"][0]["main"]
|
33 |
+
timezone = Climate.getCountryTimezone(data["sys"]["country"])
|
34 |
+
sunrise = await Climate.getTime(data["sys"]["sunrise"])
|
35 |
+
sunset = await Climate.getTime(data["sys"]["sunset"])
|
36 |
+
wind = Climate.getWindData(data["wind"]["speed"], data["wind"]["deg"])
|
37 |
+
temperature = data["main"]["temp"]
|
38 |
+
feels_like = data["main"]["feels_like"]
|
39 |
+
temp_min = data["main"]["temp_min"]
|
40 |
+
temp_max = data["main"]["temp_max"]
|
41 |
+
pressure = data["main"]["pressure"]
|
42 |
+
humidity = data["main"]["humidity"]
|
43 |
+
visibility = data["visibility"]
|
44 |
+
clouds = data["clouds"]["all"]
|
45 |
+
|
46 |
+
await hell.edit(
|
47 |
+
await climate_templates(
|
48 |
+
city_name=city_name,
|
49 |
+
country=country,
|
50 |
+
weather=weather,
|
51 |
+
timezone=timezone,
|
52 |
+
sunrise=sunrise,
|
53 |
+
sunset=sunset,
|
54 |
+
wind=wind,
|
55 |
+
temperature=temperature,
|
56 |
+
feels_like=feels_like,
|
57 |
+
temp_min=temp_min,
|
58 |
+
temp_max=temp_max,
|
59 |
+
pressure=pressure,
|
60 |
+
humidity=humidity,
|
61 |
+
visibility=visibility,
|
62 |
+
clouds=clouds,
|
63 |
+
),
|
64 |
+
disable_web_page_preview=True,
|
65 |
+
)
|
66 |
+
|
67 |
+
|
68 |
+
@on_message("airpollution", allow_stan=True)
|
69 |
+
async def airpollution(_, message: Message):
|
70 |
+
if len(message.command) < 2:
|
71 |
+
city = "Delhi"
|
72 |
+
else:
|
73 |
+
city = await hellbot.input(message)
|
74 |
+
|
75 |
+
hell = await hellbot.edit(message, f"**Fetching air pollution data for `{city}`**")
|
76 |
+
|
77 |
+
apiKey = await db.get_env(ENV.climate_api)
|
78 |
+
if not apiKey:
|
79 |
+
return await hellbot.delete(
|
80 |
+
hell, f"**Climate API not found!** Setup `{ENV.climate_api}` first."
|
81 |
+
)
|
82 |
+
|
83 |
+
data = await Climate.fetchAirPollution(city, apiKey)
|
84 |
+
if not data:
|
85 |
+
return await hellbot.delete(
|
86 |
+
hell, f"**Air pollution data not found for `{city}`**"
|
87 |
+
)
|
88 |
+
|
89 |
+
data = data["list"][0]
|
90 |
+
ugm3 = superscript("µg/m³")
|
91 |
+
sub2_5 = subscript("2.5")
|
92 |
+
sub10 = subscript("10")
|
93 |
+
|
94 |
+
city_name = city.upper()
|
95 |
+
aqi_index = data["main"]["aqi"]
|
96 |
+
aqi_cond = Climate.AQI_DICT[aqi_index]
|
97 |
+
aqi = f"{aqi_index} ({aqi_cond})"
|
98 |
+
co = f"{data['components']['co']} {ugm3}"
|
99 |
+
no = f"{data['components']['no']} {ugm3}"
|
100 |
+
no2 = f"{data['components']['no2']} {ugm3}"
|
101 |
+
o3 = f"{data['components']['o3']} {ugm3}"
|
102 |
+
so2 = f"{data['components']['so2']} {ugm3}"
|
103 |
+
nh3 = f"{data['components']['nh3']} {ugm3}"
|
104 |
+
pm2_5 = data["components"]["pm2_5"]
|
105 |
+
pm10 = data["components"]["pm10"]
|
106 |
+
|
107 |
+
await hell.edit(
|
108 |
+
await airpollution_templates(
|
109 |
+
city_name=city_name,
|
110 |
+
aqi=aqi,
|
111 |
+
co=co,
|
112 |
+
no=no,
|
113 |
+
no2=no2,
|
114 |
+
o3=o3,
|
115 |
+
so2=so2,
|
116 |
+
nh3=nh3,
|
117 |
+
pm2_5=pm2_5,
|
118 |
+
pm10=pm10,
|
119 |
+
sub2_5=sub2_5,
|
120 |
+
sub10=sub10,
|
121 |
+
),
|
122 |
+
disable_web_page_preview=True,
|
123 |
+
)
|
124 |
+
|
125 |
+
|
126 |
+
HelpMenu("climate").add(
|
127 |
+
"climate",
|
128 |
+
"<city name>",
|
129 |
+
"Get climate data of a city.",
|
130 |
+
"climate Delhi",
|
131 |
+
"City name is optional. Bydefault Delhi's climate data will be fetched.",
|
132 |
+
).add(
|
133 |
+
"airpollution",
|
134 |
+
"<city name>",
|
135 |
+
"Get air pollution data of a city.",
|
136 |
+
"airpollution Delhi",
|
137 |
+
"City name is optional. Bydefault Delhi's air pollution data will be fetched.",
|
138 |
+
).info(
|
139 |
+
"Get the API Key from [here](https://openweathermap.org/price)"
|
140 |
+
).done()
|
HellBot/plugins/user/clone.py
ADDED
@@ -0,0 +1,106 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
|
3 |
+
from pyrogram import Client
|
4 |
+
from pyrogram.raw.functions.users import GetFullUser
|
5 |
+
from pyrogram.types import Message
|
6 |
+
|
7 |
+
from . import HelpMenu, db, hellbot, on_message
|
8 |
+
|
9 |
+
|
10 |
+
@on_message("clone", allow_stan=True)
|
11 |
+
async def clone(client: Client, message: Message):
|
12 |
+
if not message.reply_to_message:
|
13 |
+
return await hellbot.delete(
|
14 |
+
message, "Reply to a user's message to clone their profile."
|
15 |
+
)
|
16 |
+
|
17 |
+
replied_user = message.reply_to_message.from_user
|
18 |
+
if replied_user.is_self:
|
19 |
+
return await hellbot.delete(message, "I can't clone myself!")
|
20 |
+
|
21 |
+
hell = await hellbot.edit(message, "Cloning ...")
|
22 |
+
|
23 |
+
try:
|
24 |
+
meh = await client.resolve_peer(client.me.id)
|
25 |
+
fullUser = await client.invoke(GetFullUser(id=meh))
|
26 |
+
about = fullUser.full_user.about or ""
|
27 |
+
except:
|
28 |
+
about = ""
|
29 |
+
|
30 |
+
first_name = client.me.first_name
|
31 |
+
last_name = client.me.last_name or ""
|
32 |
+
|
33 |
+
await db.set_env("CLONE_FIRST_NAME", first_name)
|
34 |
+
await db.set_env("CLONE_LAST_NAME", last_name)
|
35 |
+
await db.set_env("CLONE_ABOUT", about)
|
36 |
+
|
37 |
+
try:
|
38 |
+
targetUser = await client.resolve_peer(replied_user.id)
|
39 |
+
repliedFullUser = await client.invoke(GetFullUser(id=targetUser))
|
40 |
+
await client.update_profile(
|
41 |
+
first_name=replied_user.first_name,
|
42 |
+
last_name=replied_user.last_name or "",
|
43 |
+
about=repliedFullUser.full_user.about or "",
|
44 |
+
)
|
45 |
+
except:
|
46 |
+
await client.update_profile(
|
47 |
+
first_name=replied_user.first_name,
|
48 |
+
last_name=replied_user.last_name or "",
|
49 |
+
)
|
50 |
+
|
51 |
+
try:
|
52 |
+
profile_pic = await client.download_media(replied_user.photo.big_file_id)
|
53 |
+
await client.set_profile_photo(photo=profile_pic)
|
54 |
+
os.remove(profile_pic)
|
55 |
+
except:
|
56 |
+
pass
|
57 |
+
|
58 |
+
await hell.edit("**😁 𝖧𝖾𝗅𝗅𝗈 𝗆𝗒 𝖿𝗋𝗂𝖾𝗇𝖽!**")
|
59 |
+
await hellbot.check_and_log(
|
60 |
+
"clone",
|
61 |
+
f"**Cloned {replied_user.mention}** ({replied_user.id}) \n\n**By:** {first_name}",
|
62 |
+
)
|
63 |
+
|
64 |
+
|
65 |
+
@on_message("revert", allow_stan=True)
|
66 |
+
async def revert(client: Client, message: Message):
|
67 |
+
first_name = await db.get_env("CLONE_FIRST_NAME")
|
68 |
+
last_name = await db.get_env("CLONE_LAST_NAME")
|
69 |
+
about = await db.get_env("CLONE_ABOUT")
|
70 |
+
|
71 |
+
if not first_name:
|
72 |
+
return await hellbot.delete(message, "I'm not cloned yet.")
|
73 |
+
|
74 |
+
hell = await hellbot.edit(message, "Reverting ...")
|
75 |
+
|
76 |
+
await client.update_profile(first_name, last_name, about)
|
77 |
+
|
78 |
+
async for photos in client.get_chat_photos("me", 1):
|
79 |
+
await client.delete_profile_photos(photos.file_id)
|
80 |
+
|
81 |
+
await db.rm_env("CLONE_FIRST_NAME")
|
82 |
+
await db.rm_env("CLONE_LAST_NAME")
|
83 |
+
await db.rm_env("CLONE_ABOUT")
|
84 |
+
|
85 |
+
await hell.edit("**Reverted back!**")
|
86 |
+
await hellbot.check_and_log(
|
87 |
+
"revert",
|
88 |
+
f"**Reverted to my original profile.** \n\n**By:** {first_name}",
|
89 |
+
)
|
90 |
+
|
91 |
+
|
92 |
+
HelpMenu("clone").add(
|
93 |
+
"clone",
|
94 |
+
"<reply to user's message>",
|
95 |
+
"Clone the profile of replied user.",
|
96 |
+
"clone",
|
97 |
+
"You can revert back to last profile only. Clone with precaution.",
|
98 |
+
).add(
|
99 |
+
"revert",
|
100 |
+
None,
|
101 |
+
"Revert back to original profile.",
|
102 |
+
"revert",
|
103 |
+
"You can revert back to last profile only. Clone with precaution.",
|
104 |
+
).info(
|
105 |
+
"Clone Menu"
|
106 |
+
).done()
|
HellBot/plugins/user/convert.py
ADDED
@@ -0,0 +1,216 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import time
|
3 |
+
|
4 |
+
from pyrogram.types import Message
|
5 |
+
|
6 |
+
from Hellbot.functions.convert import convert_to_gif
|
7 |
+
from Hellbot.functions.tools import runcmd
|
8 |
+
|
9 |
+
from . import HelpMenu, hellbot, on_message, Config
|
10 |
+
|
11 |
+
|
12 |
+
@on_message("stog", allow_stan=True)
|
13 |
+
async def sticker_to_gif(_, message: Message):
|
14 |
+
if not message.reply_to_message or not message.reply_to_message.sticker:
|
15 |
+
return await hellbot.delete(
|
16 |
+
message, "Reply to an animated/video sticker to convert it to gif."
|
17 |
+
)
|
18 |
+
|
19 |
+
hell = await hellbot.edit(message, "Converting ...")
|
20 |
+
|
21 |
+
replied_sticker = message.reply_to_message.sticker
|
22 |
+
|
23 |
+
if replied_sticker.is_animated:
|
24 |
+
is_video = False
|
25 |
+
elif replied_sticker.is_video:
|
26 |
+
is_video = True
|
27 |
+
else:
|
28 |
+
return await hellbot.delete(hell, "Reply to an animated/video sticker.")
|
29 |
+
|
30 |
+
dwl_path = await message.reply_to_message.download(Config.TEMP_DIR)
|
31 |
+
gif_path = await convert_to_gif(dwl_path, is_video)
|
32 |
+
|
33 |
+
await message.reply_animation(gif_path)
|
34 |
+
await hellbot.delete(hell, "Converted to gif successfully!")
|
35 |
+
|
36 |
+
os.remove(dwl_path)
|
37 |
+
os.remove(gif_path)
|
38 |
+
|
39 |
+
|
40 |
+
@on_message("stoi", allow_stan=True)
|
41 |
+
async def sticker_to_image(_, message: Message):
|
42 |
+
if not message.reply_to_message or not message.reply_to_message.sticker:
|
43 |
+
return await hellbot.delete(
|
44 |
+
message, "Reply to an sticker to convert it to image."
|
45 |
+
)
|
46 |
+
|
47 |
+
hell = await hellbot.edit(message, "Converting ...")
|
48 |
+
fileName = f"image_{round(time.time())}.png"
|
49 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}{fileName}")
|
50 |
+
|
51 |
+
await message.reply_photo(dwl_path)
|
52 |
+
await hellbot.delete(hell, "Converted to image successfully!")
|
53 |
+
|
54 |
+
os.remove(dwl_path)
|
55 |
+
|
56 |
+
|
57 |
+
@on_message("itos", allow_stan=True)
|
58 |
+
async def image_to_sticker(_, message: Message):
|
59 |
+
if not message.reply_to_message or not message.reply_to_message.photo:
|
60 |
+
return await hellbot.delete(
|
61 |
+
message, "Reply to an image to convert it to sticker."
|
62 |
+
)
|
63 |
+
|
64 |
+
hell = await hellbot.edit(message, "Converting ...")
|
65 |
+
fileName = f"sticker_{round(time.time())}.webp"
|
66 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}{fileName}")
|
67 |
+
|
68 |
+
await message.reply_sticker(dwl_path)
|
69 |
+
await hellbot.delete(hell, "Converted to sticker successfully!")
|
70 |
+
|
71 |
+
os.remove(dwl_path)
|
72 |
+
|
73 |
+
|
74 |
+
@on_message("ftoi", allow_stan=True)
|
75 |
+
async def file_to_image(_, message: Message):
|
76 |
+
if not message.reply_to_message or not message.reply_to_message.document:
|
77 |
+
return await hellbot.delete(message, "Reply to a file to convert it to image.")
|
78 |
+
|
79 |
+
if message.reply_to_message.document.mime_type.split("/")[0] != "image":
|
80 |
+
return await hellbot.delete(message, "Reply to an image file.")
|
81 |
+
|
82 |
+
hell = await hellbot.edit(message, "Converting ...")
|
83 |
+
fileName = f"image_{round(time.time())}.png"
|
84 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}{fileName}")
|
85 |
+
|
86 |
+
await message.reply_photo(dwl_path)
|
87 |
+
await hellbot.delete(hell, "Converted to image successfully!")
|
88 |
+
|
89 |
+
os.remove(dwl_path)
|
90 |
+
|
91 |
+
|
92 |
+
@on_message("itof", allow_stan=True)
|
93 |
+
async def image_to_file(_, message: Message):
|
94 |
+
if not message.reply_to_message or not message.reply_to_message.photo:
|
95 |
+
return await hellbot.delete(message, "Reply to an image to convert it to file.")
|
96 |
+
|
97 |
+
hell = await hellbot.edit(message, "Converting ...")
|
98 |
+
fileName = f"file_{round(time.time())}.png"
|
99 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}{fileName}")
|
100 |
+
|
101 |
+
await message.reply_document(dwl_path)
|
102 |
+
await hellbot.delete(hell, "Converted to file successfully!")
|
103 |
+
|
104 |
+
os.remove(dwl_path)
|
105 |
+
|
106 |
+
|
107 |
+
@on_message("tovoice", allow_stan=True)
|
108 |
+
async def media_to_voice(_, message: Message):
|
109 |
+
if not message.reply_to_message or not message.reply_to_message.media:
|
110 |
+
return await hellbot.delete(message, "Reply to a media to convert it to voice.")
|
111 |
+
|
112 |
+
hell = await hellbot.edit(message, "Converting ...")
|
113 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}")
|
114 |
+
voice_path = f"{round(time.time())}.ogg"
|
115 |
+
|
116 |
+
cmd_list = [
|
117 |
+
"ffmpeg",
|
118 |
+
"-i",
|
119 |
+
dwl_path,
|
120 |
+
"-map",
|
121 |
+
"0:a",
|
122 |
+
"-codec:a",
|
123 |
+
"libopus",
|
124 |
+
"-b:a",
|
125 |
+
"100k",
|
126 |
+
"-vbr",
|
127 |
+
"on",
|
128 |
+
voice_path,
|
129 |
+
]
|
130 |
+
|
131 |
+
_, err, _, _ = await runcmd(" ".join(cmd_list))
|
132 |
+
|
133 |
+
if os.path.exists(voice_path):
|
134 |
+
await message.reply_voice(voice_path)
|
135 |
+
await hellbot.delete(hell, "Converted to voice successfully!")
|
136 |
+
os.remove(voice_path)
|
137 |
+
else:
|
138 |
+
await hellbot.error(hell, f"`{err}`")
|
139 |
+
|
140 |
+
os.remove(dwl_path)
|
141 |
+
|
142 |
+
|
143 |
+
@on_message("tomp3", allow_stan=True)
|
144 |
+
async def media_to_mp3(_, message: Message):
|
145 |
+
if not message.reply_to_message or not message.reply_to_message.media:
|
146 |
+
return await hellbot.delete(message, "Reply to a media to convert it to mp3.")
|
147 |
+
|
148 |
+
hell = await hellbot.edit(message, "Converting ...")
|
149 |
+
dwl_path = await message.reply_to_message.download(f"{Config.TEMP_DIR}")
|
150 |
+
mp3_path = f"{round(time.time())}.mp3"
|
151 |
+
|
152 |
+
cmd_list = [
|
153 |
+
"ffmpeg",
|
154 |
+
"-i",
|
155 |
+
dwl_path,
|
156 |
+
"-vn",
|
157 |
+
mp3_path,
|
158 |
+
]
|
159 |
+
|
160 |
+
_, stderr, _, _ = await runcmd(" ".join(cmd_list))
|
161 |
+
|
162 |
+
if os.path.exists(mp3_path):
|
163 |
+
await message.reply_audio(mp3_path)
|
164 |
+
await hellbot.delete(hell, "Converted to mp3 successfully!")
|
165 |
+
os.remove(mp3_path)
|
166 |
+
else:
|
167 |
+
await hellbot.error(hell, f"`{stderr}`")
|
168 |
+
|
169 |
+
os.remove(dwl_path)
|
170 |
+
|
171 |
+
|
172 |
+
HelpMenu("convert").add(
|
173 |
+
"stog", #Bugged: to-be-fixed
|
174 |
+
"<reply>",
|
175 |
+
"Converts animated sticker to gif.",
|
176 |
+
None,
|
177 |
+
"Only animated sticker and video sticker can be converted to gif.",
|
178 |
+
).add(
|
179 |
+
"stoi",
|
180 |
+
"<reply>",
|
181 |
+
"Converts sticker to image.",
|
182 |
+
None,
|
183 |
+
"Only static stickers can be converted to image.",
|
184 |
+
).add(
|
185 |
+
"itos",
|
186 |
+
"<reply>",
|
187 |
+
"Converts image to sticker.",
|
188 |
+
None,
|
189 |
+
"Only images can be converted to sticker.",
|
190 |
+
).add(
|
191 |
+
"ftoi",
|
192 |
+
"<reply>",
|
193 |
+
"Converts file to image.",
|
194 |
+
None,
|
195 |
+
"Only image files can be converted to image.",
|
196 |
+
).add(
|
197 |
+
"itof",
|
198 |
+
"<reply>",
|
199 |
+
"Converts image to file.",
|
200 |
+
None,
|
201 |
+
"Only images can be converted to file.",
|
202 |
+
).add(
|
203 |
+
"tovoice",
|
204 |
+
"<reply>",
|
205 |
+
"Converts media to voice.",
|
206 |
+
None,
|
207 |
+
"Only video/audio can be converted to voice.",
|
208 |
+
).add(
|
209 |
+
"tomp3",
|
210 |
+
"<reply>",
|
211 |
+
"Converts media to mp3.",
|
212 |
+
None,
|
213 |
+
"Only video/audio can be converted to mp3.",
|
214 |
+
).info(
|
215 |
+
"Converts media to other formats."
|
216 |
+
).done()
|
HellBot/plugins/user/core.py
ADDED
@@ -0,0 +1,321 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import importlib
|
2 |
+
import os
|
3 |
+
import sys
|
4 |
+
from pathlib import Path
|
5 |
+
|
6 |
+
from pyrogram import Client
|
7 |
+
from pyrogram.enums import MessagesFilter, ParseMode
|
8 |
+
from pyrogram.types import Message
|
9 |
+
|
10 |
+
from Hellbot.core import ENV, Config, Symbols
|
11 |
+
|
12 |
+
from . import HelpMenu, bot, db, handler, hellbot, on_message
|
13 |
+
|
14 |
+
|
15 |
+
@on_message("help", allow_stan=True)
|
16 |
+
async def help(client: Client, message: Message):
|
17 |
+
hell = await hellbot.edit(message, "**Processing...**")
|
18 |
+
if len(message.command) == 1:
|
19 |
+
try:
|
20 |
+
result = await client.get_inline_bot_results(bot.me.username, "help_menu")
|
21 |
+
await client.send_inline_bot_result(
|
22 |
+
message.chat.id,
|
23 |
+
result.query_id,
|
24 |
+
result.results[0].id,
|
25 |
+
True,
|
26 |
+
)
|
27 |
+
return await hell.delete()
|
28 |
+
except Exception as e:
|
29 |
+
await hellbot.error(hell, str(e), 20)
|
30 |
+
return
|
31 |
+
|
32 |
+
plugin = await hellbot.input(message)
|
33 |
+
if plugin.lower() in Config.CMD_MENU:
|
34 |
+
try:
|
35 |
+
await hellbot.edit(
|
36 |
+
hell, Config.CMD_MENU[plugin.lower()], ParseMode.MARKDOWN
|
37 |
+
)
|
38 |
+
return
|
39 |
+
except Exception as e:
|
40 |
+
await hellbot.error(hell, str(e), 20)
|
41 |
+
return
|
42 |
+
|
43 |
+
available_plugins = f"{Symbols.bullet} **𝖠𝗏𝖺𝗂𝗅𝖺𝖻𝗅𝖾 𝗉𝗅𝗎𝗀𝗂𝗇𝗌:**\n\n"
|
44 |
+
for i in sorted(list(Config.CMD_MENU.keys())):
|
45 |
+
available_plugins += f"`{i}`, "
|
46 |
+
available_plugins = available_plugins[:-2]
|
47 |
+
available_plugins += (
|
48 |
+
f"\n\n𝖣𝗈 `{handler}help <plugin name>` 𝗍𝗈 𝗀𝖾𝗍 𝖽𝖾𝗍𝖺𝗂𝗅𝖾𝖽 𝗂𝗇𝖿𝗈 𝗈𝖿 𝗍𝗁𝖺𝗍 𝗉𝗅𝗎𝗀𝗂𝗇."
|
49 |
+
)
|
50 |
+
await hellbot.edit(hell, available_plugins, ParseMode.MARKDOWN)
|
51 |
+
|
52 |
+
|
53 |
+
@on_message("repo", allow_stan=True)
|
54 |
+
async def repo(_, message: Message):
|
55 |
+
REPO_TEXT = (
|
56 |
+
"__🍀 𝖱𝖾𝗉𝗈:__ [Github](https://github.com/The-HellBot/HellBot)\n\n"
|
57 |
+
"__🍀 Updates:__ @Its_HellBot\n"
|
58 |
+
"__🍀 Support:__ @HellBot_Chats\n\n"
|
59 |
+
"**By ©️ @HellBot_Networks**"
|
60 |
+
)
|
61 |
+
await hellbot.edit(message, REPO_TEXT, no_link_preview=True)
|
62 |
+
|
63 |
+
|
64 |
+
@on_message("plinfo", allow_stan=True)
|
65 |
+
async def plugin_info(_, message: Message):
|
66 |
+
plugin = await hellbot.input(message)
|
67 |
+
if plugin.lower() in Config.CMD_MENU:
|
68 |
+
try:
|
69 |
+
await hellbot.edit(
|
70 |
+
message, Config.CMD_MENU[plugin.lower()], ParseMode.MARKDOWN
|
71 |
+
)
|
72 |
+
return
|
73 |
+
except Exception as e:
|
74 |
+
await hellbot.error(message, str(e), 20)
|
75 |
+
return
|
76 |
+
await hellbot.error(message, f"**Invalid Plugin Name:** `{plugin}`", 20)
|
77 |
+
|
78 |
+
|
79 |
+
@on_message("cmdinfo", allow_stan=True)
|
80 |
+
async def command_info(_, message: Message):
|
81 |
+
cmd = await hellbot.input(message)
|
82 |
+
if cmd.lower() in Config.CMD_INFO:
|
83 |
+
try:
|
84 |
+
cmd_dict = Config.CMD_INFO[cmd.lower()]
|
85 |
+
template = (
|
86 |
+
f"**🍀 𝖯𝗅𝗎𝗀𝗂𝗇:** `{cmd_dict['plugin']}.py`\n\n"
|
87 |
+
f"**{Symbols.anchor} 𝖢𝗈𝗆𝗆𝖺𝗇𝖽:** `{cmd_dict['command']}`\n"
|
88 |
+
f"**{Symbols.anchor} 𝖣𝖾𝗌𝖼𝗋𝗂𝗉𝗍𝗂𝗈𝗇:** __{cmd_dict['description']}__\n"
|
89 |
+
f"**{Symbols.anchor} 𝖤𝗑𝖺𝗆𝗉𝗅𝖾:** `{cmd_dict['example']}`\n"
|
90 |
+
f"**{Symbols.anchor} 𝖭𝗈𝗍𝖾:** __{cmd_dict['note']}__\n"
|
91 |
+
)
|
92 |
+
await hellbot.edit(message, template, ParseMode.MARKDOWN)
|
93 |
+
return
|
94 |
+
except Exception as e:
|
95 |
+
await hellbot.error(message, str(e), 20)
|
96 |
+
return
|
97 |
+
await hellbot.error(message, f"**Invalid Command Name:** `{cmd}`", 20)
|
98 |
+
|
99 |
+
|
100 |
+
@on_message("send", allow_stan=True)
|
101 |
+
async def send_plugin(client: Client, message: Message):
|
102 |
+
if len(message.command) < 2:
|
103 |
+
return await hellbot.delete(message, "Give me a plugin name to send.")
|
104 |
+
|
105 |
+
plugin = message.command[1].lower().replace(".py", "").strip()
|
106 |
+
if plugin not in Config.CMD_MENU:
|
107 |
+
return await hellbot.delete(message, f"**Invalid Plugin Name:** `{plugin}`")
|
108 |
+
|
109 |
+
try:
|
110 |
+
await client.send_document(
|
111 |
+
message.chat.id,
|
112 |
+
f"./Hellbot/plugins/user/{plugin}.py",
|
113 |
+
caption=f"**🍀 𝖯𝗅𝗎𝗀𝗂𝗇:** `{plugin}.py`",
|
114 |
+
)
|
115 |
+
await hellbot.delete(message, f"**Sent:** `{plugin}.py`")
|
116 |
+
except Exception as e:
|
117 |
+
await hellbot.error(message, str(e), 20)
|
118 |
+
|
119 |
+
|
120 |
+
@on_message("install", allow_stan=True)
|
121 |
+
async def install_plugins(_, message: Message):
|
122 |
+
if not message.reply_to_message or not message.reply_to_message.document:
|
123 |
+
return await hellbot.delete(message, "Reply to a plugin to install it.")
|
124 |
+
|
125 |
+
hell = await hellbot.edit(message, "**Installing...**")
|
126 |
+
plugin_path = await message.reply_to_message.download("./Hellbot/plugins/user/")
|
127 |
+
|
128 |
+
if not plugin_path.endswith(".py"):
|
129 |
+
os.remove(plugin_path)
|
130 |
+
return await hellbot.error(hell, "**Invalid Plugin:** Not a python file.", 20)
|
131 |
+
|
132 |
+
plugin = plugin_path.split("/")[-1].replace(".py", "").strip()
|
133 |
+
if plugin in Config.CMD_MENU:
|
134 |
+
os.remove(plugin_path)
|
135 |
+
return await hellbot.error(
|
136 |
+
hell, f"**Plugin Already Installed:** `{plugin}.py`", 20
|
137 |
+
)
|
138 |
+
|
139 |
+
if "(" in plugin:
|
140 |
+
os.remove(plugin_path)
|
141 |
+
return await hellbot.error(
|
142 |
+
hell, f"**Plugin Already Installed:** `{plugin}.py`", 20
|
143 |
+
)
|
144 |
+
|
145 |
+
try:
|
146 |
+
shortname = Path(plugin_path).stem.replace(".py", "")
|
147 |
+
path = Path(f"Hellbot/plugins/user/{shortname}.py")
|
148 |
+
name = "Hellbot.plugins.user." + shortname
|
149 |
+
spec = importlib.util.spec_from_file_location(name, path)
|
150 |
+
load = importlib.util.module_from_spec(spec)
|
151 |
+
spec.loader.exec_module(load)
|
152 |
+
sys.modules["Hellbot.plugins.user." + shortname] = load
|
153 |
+
await hellbot.edit(hell, f"**Installed:** `{plugin}.py`")
|
154 |
+
except Exception as e:
|
155 |
+
await hellbot.error(hell, str(e), 20)
|
156 |
+
os.remove(plugin_path)
|
157 |
+
|
158 |
+
|
159 |
+
@on_message("uninstall", allow_stan=True)
|
160 |
+
async def uninstall_plugins(_, message: Message):
|
161 |
+
if len(message.command) < 2:
|
162 |
+
return await hellbot.delete(message, "Give me a plugin name to uninstall.")
|
163 |
+
|
164 |
+
plugin = message.command[1].lower().replace(".py", "").strip()
|
165 |
+
if plugin not in Config.CMD_MENU:
|
166 |
+
return await hellbot.delete(message, f"**Invalid Plugin Name:** `{plugin}`")
|
167 |
+
|
168 |
+
try:
|
169 |
+
os.remove(f"./Hellbot/plugins/user/{plugin}.py")
|
170 |
+
for i in Config.HELP_DICT[plugin]["commands"]:
|
171 |
+
cmd = i["command"]
|
172 |
+
for i in hellbot.users:
|
173 |
+
i.remove_handler(cmd)
|
174 |
+
del Config.CMD_INFO[cmd]
|
175 |
+
del Config.HELP_DICT[plugin]
|
176 |
+
del Config.CMD_MENU[plugin]
|
177 |
+
await hellbot.delete(message, f"**Uninstalled:** `{plugin}.py`")
|
178 |
+
except Exception as e:
|
179 |
+
await hellbot.error(message, str(e), 20)
|
180 |
+
|
181 |
+
|
182 |
+
@on_message("installall", allow_stan=True)
|
183 |
+
async def installall(client: Client, message: Message):
|
184 |
+
if len(message.command) < 2:
|
185 |
+
return await hellbot.delete(
|
186 |
+
message, "Give me channel username to install plugins."
|
187 |
+
)
|
188 |
+
|
189 |
+
try:
|
190 |
+
chat = await client.get_chat(message.command[1])
|
191 |
+
except Exception as e:
|
192 |
+
return await hellbot.delete(message, f"**Invalid Channel Username:** `{e}`")
|
193 |
+
|
194 |
+
hell = await hellbot.edit(message, f"**Installing plugins from {chat.title}...**")
|
195 |
+
finalStr = f"{Symbols.check_mark} **𝖯𝗅𝗎𝗀𝗂𝗇𝗌 𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝖾𝖽: {chat.title}**\n\n"
|
196 |
+
count = 0
|
197 |
+
|
198 |
+
async for msg in client.search_messages(chat.id, filter=MessagesFilter.DOCUMENT):
|
199 |
+
if msg.document.file_name.endswith(".py"):
|
200 |
+
dwl_path = await msg.download("./Hellbot/plugins/user/")
|
201 |
+
plugin = dwl_path.split("/")[-1].replace(".py", "").strip()
|
202 |
+
if plugin in Config.CMD_MENU:
|
203 |
+
os.remove(dwl_path)
|
204 |
+
finalStr += (
|
205 |
+
f" {Symbols.anchor} `{plugin}.py` - **𝖠𝗅𝗋𝖾𝖺𝖽𝗒 𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝖾𝖽!**\n"
|
206 |
+
)
|
207 |
+
continue
|
208 |
+
if "(" in plugin:
|
209 |
+
os.remove(dwl_path)
|
210 |
+
finalStr += (
|
211 |
+
f" {Symbols.anchor} `{plugin}.py` - **𝖠𝗅𝗋𝖾𝖺𝖽𝗒 𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝖾𝖽!**\n"
|
212 |
+
)
|
213 |
+
continue
|
214 |
+
try:
|
215 |
+
shortname = Path(dwl_path).stem.replace(".py", "")
|
216 |
+
path = Path(f"Hellbot/plugins/user/{shortname}.py")
|
217 |
+
name = "Hellbot.plugins.user." + shortname
|
218 |
+
spec = importlib.util.spec_from_file_location(name, path)
|
219 |
+
load = importlib.util.module_from_spec(spec)
|
220 |
+
spec.loader.exec_module(load)
|
221 |
+
sys.modules["Hellbot.plugins.user." + shortname] = load
|
222 |
+
count += 1
|
223 |
+
finalStr += f" {Symbols.anchor} `{plugin}.py` - **𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝖾𝖽!**\n"
|
224 |
+
except Exception as e:
|
225 |
+
os.remove(dwl_path)
|
226 |
+
finalStr += (
|
227 |
+
f" {Symbols.anchor} `{plugin}.py` - **𝖤𝗋𝗋𝗈𝗋 𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝗂𝗇𝗀!**\n"
|
228 |
+
)
|
229 |
+
continue
|
230 |
+
|
231 |
+
finalStr += f"\n**🍀 𝖳𝗈𝗍𝖺𝗅 𝖯𝗅𝗎𝗀𝗂𝗇𝗌 𝖨𝗇𝗌𝗍𝖺𝗅𝗅𝖾𝖽:** `{count}`"
|
232 |
+
await hell.edit(finalStr, ParseMode.MARKDOWN, disable_web_page_preview=True)
|
233 |
+
|
234 |
+
|
235 |
+
@on_message("unload", allow_stan=True)
|
236 |
+
async def unload_plugins(_, message: Message):
|
237 |
+
if len(message.command) < 2:
|
238 |
+
return await hellbot.delete(message, "Give me a plugin name to unload.")
|
239 |
+
|
240 |
+
plugin = message.command[1].lower().replace(".py", "").strip()
|
241 |
+
if plugin not in Config.CMD_MENU:
|
242 |
+
return await hellbot.delete(message, f"**Invalid Plugin Name:** `{plugin}`")
|
243 |
+
|
244 |
+
unloaded = await db.get_env(ENV.unload_plugins) or ""
|
245 |
+
unloaded = unloaded.split(" ")
|
246 |
+
if plugin in unloaded:
|
247 |
+
return await hellbot.delete(message, f"**Already Unloaded:** `{plugin}.py`")
|
248 |
+
|
249 |
+
unloaded.append(plugin)
|
250 |
+
await db.set_env(ENV.unload_plugins, " ".join(unloaded))
|
251 |
+
await hellbot.delete(
|
252 |
+
message, f"**Unloaded:** `{plugin}.py` \n\n__Restart the bot to see changes.__"
|
253 |
+
)
|
254 |
+
|
255 |
+
|
256 |
+
@on_message("load", allow_stan=True)
|
257 |
+
async def load_plugins(_, message: Message):
|
258 |
+
if len(message.command) < 2:
|
259 |
+
return await hellbot.delete(message, "Give me a plugin name to load.")
|
260 |
+
|
261 |
+
plugin = message.command[1].lower().replace(".py", "").strip()
|
262 |
+
unloaded = await db.get_env(ENV.unload_plugins) or ""
|
263 |
+
unloaded = unloaded.split(" ")
|
264 |
+
if plugin not in unloaded:
|
265 |
+
return await hellbot.delete(message, f"**Already Loaded:** `{plugin}.py`")
|
266 |
+
|
267 |
+
unloaded.remove(plugin)
|
268 |
+
await db.set_env(ENV.unload_plugins, " ".join(unloaded))
|
269 |
+
await hellbot.delete(
|
270 |
+
message, f"**Loaded:** `{plugin}.py` \n\n__Restart the bot to see changes.__"
|
271 |
+
)
|
272 |
+
|
273 |
+
|
274 |
+
HelpMenu("help").add(
|
275 |
+
"help",
|
276 |
+
"<plugin name>",
|
277 |
+
"Get the detailed help menu for that mentioned plugin or get the whole help menu instead.",
|
278 |
+
"help alive",
|
279 |
+
).add("repo", None, "Get the repo link of the bot.", "repo").add(
|
280 |
+
"plinfo",
|
281 |
+
"<plugin name>",
|
282 |
+
"Get the detailed info of the mentioned plugin.",
|
283 |
+
"plinfo alive",
|
284 |
+
).add(
|
285 |
+
"cmdinfo",
|
286 |
+
"<command name>",
|
287 |
+
"Get the detailed info of the mentioned command.",
|
288 |
+
"cmdinfo alive",
|
289 |
+
).add(
|
290 |
+
"send", "<plugin name>", "Send the mentioned plugin.", "send alive"
|
291 |
+
).add(
|
292 |
+
"install",
|
293 |
+
"<reply to plugin>",
|
294 |
+
"Install the replied plugin.",
|
295 |
+
None,
|
296 |
+
"Do not install plugins from untrusted sources, they can be a malware. We're not responsible for any damage caused by them.",
|
297 |
+
).add(
|
298 |
+
"uninstall",
|
299 |
+
"<plugin name>",
|
300 |
+
"Uninstall the mentioned plugin.",
|
301 |
+
"uninstall alive",
|
302 |
+
"This will remove all the commands of that plugin from the bot till a restart is initiated.",
|
303 |
+
).add(
|
304 |
+
"installall",
|
305 |
+
"<channel username>",
|
306 |
+
"Install all the plugins from the mentioned channel.",
|
307 |
+
"installall @plugins_for_hellbot",
|
308 |
+
"Do not install plugins from untrusted sources, they can be a malware. We're not responsible for any damage caused by them.",
|
309 |
+
).add(
|
310 |
+
"unload",
|
311 |
+
"<plugin name>",
|
312 |
+
"Unload the mentioned plugin.",
|
313 |
+
"unload alive",
|
314 |
+
"This will remove all the commands of that plugin from the bot permanently.",
|
315 |
+
).add(
|
316 |
+
"load",
|
317 |
+
"<plugin name>",
|
318 |
+
"Load the mentioned plugin.",
|
319 |
+
"load alive",
|
320 |
+
"This will load all the commands of that plugin to the bot that was previously unloaded permanently.",
|
321 |
+
)
|
HellBot/plugins/user/downloads.py
ADDED
@@ -0,0 +1,85 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import time
|
2 |
+
|
3 |
+
from pyrogram.types import Message
|
4 |
+
from pySmartDL import SmartDL
|
5 |
+
|
6 |
+
from Hellbot.core import LOGS, Symbols
|
7 |
+
from Hellbot.functions.formatter import readable_time
|
8 |
+
from Hellbot.functions.tools import progress
|
9 |
+
|
10 |
+
from . import Config, HelpMenu, hellbot, on_message
|
11 |
+
|
12 |
+
|
13 |
+
@on_message("download", allow_stan=True)
|
14 |
+
async def download(_, message: Message):
|
15 |
+
hell = await hellbot.edit(message, "Starting to download...")
|
16 |
+
if message.reply_to_message and message.reply_to_message.media:
|
17 |
+
start_time = time.time()
|
18 |
+
try:
|
19 |
+
dwl_path = await message.reply_to_message.download(
|
20 |
+
Config.DWL_DIR,
|
21 |
+
progress=progress,
|
22 |
+
progress_args=(
|
23 |
+
hell,
|
24 |
+
start_time,
|
25 |
+
"⬇️ Downloading",
|
26 |
+
),
|
27 |
+
)
|
28 |
+
except Exception as e:
|
29 |
+
return await hellbot.error(hell, f"`{e}`", 10)
|
30 |
+
|
31 |
+
await hell.edit(
|
32 |
+
f"**Downloaded to** `{dwl_path}` **in** {readable_time(round(time.time() - start_time))} seconds.**"
|
33 |
+
)
|
34 |
+
|
35 |
+
elif len(message.command) > 2:
|
36 |
+
dwl_url = (await hellbot.input(message)).split(" ")
|
37 |
+
start_time = time.time()
|
38 |
+
try:
|
39 |
+
dl = SmartDL(dwl_url, Config.DWL_DIR, progress_bar=False)
|
40 |
+
dl.start(blocking=False)
|
41 |
+
while not dl.isFinished():
|
42 |
+
display_msg = ""
|
43 |
+
downloaded_size = dl.get_dl_size(human=True)
|
44 |
+
file_size = dl.filesize or "Unknown"
|
45 |
+
diff = time.time() - start_time
|
46 |
+
speed = dl.get_speed(human=True)
|
47 |
+
dl_percentage = round((dl.get_progress() * 100), 2)
|
48 |
+
eta = dl.get_eta(human=True)
|
49 |
+
try:
|
50 |
+
current_msg = (
|
51 |
+
f"**⬇️ Downloading...**\n\n"
|
52 |
+
f"**{Symbols.anchor} URL:** `{dwl_url}`\n"
|
53 |
+
f"**{Symbols.anchor} Downloaded:** `{downloaded_size}` of `{file_size}`\n"
|
54 |
+
f"**{Symbols.anchor} Speed:** `{speed}`\n"
|
55 |
+
f"**{Symbols.anchor} ETA:** `{eta}`\n"
|
56 |
+
f"**{Symbols.anchor} Progress:** `{dl_percentage}%`"
|
57 |
+
)
|
58 |
+
if round(diff % 10.00) == 0 and current_msg != display_msg:
|
59 |
+
await hell.edit(current_msg)
|
60 |
+
display_msg = current_msg
|
61 |
+
except Exception as e:
|
62 |
+
LOGS.warning(f"PySmartDl: {str(e)}")
|
63 |
+
|
64 |
+
end_time = readable_time(round(time.time() - start_time))
|
65 |
+
if dl.isSuccessful():
|
66 |
+
await hell.edit(
|
67 |
+
f"**Downloaded to** `{dl.get_dest()}` **in** `{end_time}` **seconds.**"
|
68 |
+
)
|
69 |
+
else:
|
70 |
+
await hellbot.error(hell, f"**Failed to download** `{len(dwl_url)} url(s)`")
|
71 |
+
except Exception as e:
|
72 |
+
return await hellbot.error(hell, f"`{e}`", 10)
|
73 |
+
else:
|
74 |
+
return await hellbot.delete(
|
75 |
+
message, "Reply to a media message or pass direct urls to download it."
|
76 |
+
)
|
77 |
+
|
78 |
+
|
79 |
+
HelpMenu("downloads").add(
|
80 |
+
"download",
|
81 |
+
"<reply to media> or <direct link>",
|
82 |
+
"Download media to server.",
|
83 |
+
"download https://example.com/file.mp4",
|
84 |
+
"You can pass multiple urls to download it on my server.",
|
85 |
+
).info("Downloader").done()
|