Initial commit
Signed-off-by: mctaylors <cantsendmails@mctaylors.ru>
This commit is contained in:
commit
6ec797a387
7 changed files with 198 additions and 0 deletions
6
.gitignore
vendored
Normal file
6
.gitignore
vendored
Normal file
|
@ -0,0 +1,6 @@
|
|||
.idea/
|
||||
.venv/
|
||||
build/
|
||||
dist/
|
||||
__pycache__/
|
||||
*.spec
|
0
LICENSE
Normal file
0
LICENSE
Normal file
46
commands.py
Normal file
46
commands.py
Normal file
|
@ -0,0 +1,46 @@
|
|||
from telegram import Update, InlineKeyboardButton, InlineKeyboardMarkup
|
||||
from telegram.constants import ParseMode
|
||||
from telegram.ext import ContextTypes
|
||||
|
||||
|
||||
async def start_command(update: Update, context: ContextTypes.DEFAULT_TYPE) -> None:
|
||||
await update.message.reply_text(
|
||||
f"hello, i'm <b>{context.bot.first_name}</b>, an inline image grabber.\n\n"
|
||||
f"to get help, use /help",
|
||||
parse_mode=ParseMode.HTML
|
||||
)
|
||||
|
||||
|
||||
async def help_command(update: Update, context: ContextTypes.DEFAULT_TYPE) -> None:
|
||||
await update.message.reply_text(
|
||||
f"<b>how to use {context.bot.first_name}</b>\n\n"
|
||||
f"1. open your message box and type:\n"
|
||||
f"<code>@{context.bot.username} <post ID></code>\n"
|
||||
f"2. click on the box that has popped up\n"
|
||||
f"3. ???\n"
|
||||
f"4. done!",
|
||||
parse_mode=ParseMode.HTML
|
||||
)
|
||||
|
||||
|
||||
async def about_command(update: Update, context: ContextTypes.DEFAULT_TYPE) -> None:
|
||||
from main import config
|
||||
reply_markup = None
|
||||
source_url = config.get('Source', 'Url')
|
||||
if source_url.startswith('http://') or source_url.startswith('https://'):
|
||||
keyboard = [
|
||||
[
|
||||
InlineKeyboardButton(f"source code", url=source_url)
|
||||
]
|
||||
]
|
||||
reply_markup = InlineKeyboardMarkup(keyboard)
|
||||
|
||||
await update.message.reply_text(
|
||||
f"<b>about {context.bot.first_name}</b>\n"
|
||||
f"{context.bot.first_name} is an inline image grabber written in python that grabs images from Danbooru "
|
||||
f"(or other similar services).\n\n"
|
||||
f"<b>currently configured instance:</b>\n"
|
||||
f"<i>{config.get('General', 'Name')}</i> ({config.get('General', 'Domain')})",
|
||||
parse_mode=ParseMode.HTML,
|
||||
reply_markup=reply_markup
|
||||
)
|
6
config.ini
Normal file
6
config.ini
Normal file
|
@ -0,0 +1,6 @@
|
|||
[General]
|
||||
Name = Danbooru
|
||||
Domain = danbooru.donmai.us
|
||||
|
||||
[Source]
|
||||
Url =
|
14
extensions.py
Normal file
14
extensions.py
Normal file
|
@ -0,0 +1,14 @@
|
|||
import re
|
||||
|
||||
|
||||
def humanize_tags_from_json(value, default):
|
||||
if value != "":
|
||||
output = str()
|
||||
tags = value.split()
|
||||
|
||||
for t in tags:
|
||||
output += f"{re.sub('_\\(.*', '', t)}, "
|
||||
output = output[:-2]
|
||||
|
||||
return output
|
||||
return default
|
78
inline_query.py
Normal file
78
inline_query.py
Normal file
|
@ -0,0 +1,78 @@
|
|||
from uuid import uuid4
|
||||
|
||||
import requests
|
||||
from telegram import Update, InlineKeyboardButton, InlineQueryResultArticle, InputTextMessageContent, \
|
||||
InlineKeyboardMarkup
|
||||
from telegram.constants import ParseMode
|
||||
from telegram.ext import ContextTypes
|
||||
|
||||
from extensions import humanize_tags_from_json
|
||||
|
||||
|
||||
# noinspection PyUnusedLocal
|
||||
async def inline_query(update: Update, context: ContextTypes.DEFAULT_TYPE) -> None:
|
||||
query = update.inline_query.query
|
||||
|
||||
if not query:
|
||||
return
|
||||
|
||||
if not query.isdigit():
|
||||
return
|
||||
|
||||
from main import config
|
||||
response = requests.get(f"https://{config.get('General', 'Domain')}/posts/{query}.json")
|
||||
|
||||
if response.status_code != 200:
|
||||
await invalid_query(update, query)
|
||||
return
|
||||
|
||||
data = response.json()
|
||||
|
||||
await answer_query(update, query, config, data)
|
||||
|
||||
|
||||
async def answer_query(update, query, config, data):
|
||||
characters = humanize_tags_from_json(data['tag_string_character'], "no characters")
|
||||
copyrights = humanize_tags_from_json(data['tag_string_copyright'], "unknown copyright")
|
||||
artists = humanize_tags_from_json(data['tag_string_artist'], "unknown artist")
|
||||
keyboard = [
|
||||
[
|
||||
InlineKeyboardButton(f"Open in {config.get('General', 'Name')}",
|
||||
url=f"https://{config.get('General', 'Domain')}/posts/{query}"),
|
||||
InlineKeyboardButton("View original", url=data['file_url'])
|
||||
]
|
||||
]
|
||||
|
||||
results = [
|
||||
InlineQueryResultArticle(
|
||||
id=str(uuid4()),
|
||||
title=f"ID: {query}",
|
||||
description=f"{characters} ({copyrights}) drawn by {artists}",
|
||||
thumbnail_url=data['preview_file_url'],
|
||||
input_message_content=InputTextMessageContent(
|
||||
f"ID: <code>{query}</code>\n"
|
||||
f"<a href='{data['large_file_url']}'><b>{characters} ({copyrights})</b> "
|
||||
f"drawn by <b>{artists}</b></a>",
|
||||
parse_mode=ParseMode.HTML
|
||||
),
|
||||
reply_markup=InlineKeyboardMarkup(keyboard)
|
||||
)
|
||||
]
|
||||
|
||||
await update.inline_query.answer(results)
|
||||
|
||||
|
||||
async def invalid_query(update, query):
|
||||
results = [
|
||||
InlineQueryResultArticle(
|
||||
id=str(uuid4()),
|
||||
title=f"ID: {query}",
|
||||
description="not found.",
|
||||
input_message_content=InputTextMessageContent(
|
||||
f"ID: <code>{query}</code>\n<b>requested post does not exist.</b>",
|
||||
parse_mode=ParseMode.HTML
|
||||
)
|
||||
)
|
||||
]
|
||||
|
||||
await update.inline_query.answer(results)
|
48
main.py
Normal file
48
main.py
Normal file
|
@ -0,0 +1,48 @@
|
|||
import configparser
|
||||
import logging
|
||||
import os
|
||||
|
||||
import requests
|
||||
from telegram.ext import Application, CommandHandler, InlineQueryHandler
|
||||
|
||||
from commands import *
|
||||
from inline_query import inline_query
|
||||
|
||||
logging.basicConfig(
|
||||
format="%(asctime)s - %(name)s - %(levelname)s - %(message)s", level=logging.INFO
|
||||
)
|
||||
# set higher logging level for httpx to avoid all GET and POST requests being logged
|
||||
logging.getLogger("httpx").setLevel(logging.WARNING)
|
||||
|
||||
logger = logging.getLogger(__name__)
|
||||
|
||||
config = configparser.ConfigParser()
|
||||
config.read('config.ini')
|
||||
|
||||
|
||||
def main() -> None:
|
||||
validate_config()
|
||||
application = Application.builder().token(os.getenv("BOT_TOKEN")).build()
|
||||
|
||||
application.add_handler(CommandHandler("start", start_command))
|
||||
application.add_handler(CommandHandler("help", help_command))
|
||||
application.add_handler(CommandHandler("about", about_command))
|
||||
|
||||
application.add_handler(InlineQueryHandler(inline_query))
|
||||
|
||||
application.run_polling(allowed_updates=Update.ALL_TYPES)
|
||||
|
||||
|
||||
def validate_config():
|
||||
# noinspection PyBroadException
|
||||
try:
|
||||
response = requests.get(f"https://{config.get('General', 'Domain')}/profile.json")
|
||||
if response.status_code != 200:
|
||||
raise
|
||||
except:
|
||||
print('Unable validate Domain in config.ini.')
|
||||
exit(1)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
Loading…
Add table
Reference in a new issue