diff --git a/components/callbacks.py b/components/callbacks.py index 06f18de..6630825 100644 --- a/components/callbacks.py +++ b/components/callbacks.py @@ -216,52 +216,54 @@ async def reply_search(update: Update, context: ContextTypes.DEFAULT_TYPE) -> No If the message is a reply, the bot will reply to the referenced message directly. """ message = cast(Message, update.effective_message) + if not message.text: + return + last = 0.0 github_matches: List[Tuple[int, Tuple[str, str, str, str, str]]] = [] found_entries: List[Tuple[int, BaseEntry]] = [] - no_entity_text = get_text_not_in_entities(message).strip() search = cast(Search, context.bot_data["search"]) github = search.github # Parse exact matches for GitHub threads & ptbcontrib found_entries first - if not (no_entity_text.startswith("!search") or no_entity_text.endswith("!search")): - for match in GITHUB_PATTERN.finditer(no_entity_text): - logging.debug(match.groupdict()) - owner, repo, number, sha, ptbcontrib = ( - cast(str, match.groupdict()[x]) - for x in ("owner", "repo", "number", "sha", "ptbcontrib") - ) - if number or sha or ptbcontrib: - github_matches.append((match.start(), (owner, repo, number, sha, ptbcontrib))) - - for gh_match in github_matches: - last = keep_typing( - last, - cast(Chat, update.effective_chat), - ChatAction.TYPING, - application=context.application, - ) - owner, repo, number, sha, ptbcontrib = gh_match[1] - owner = owner or DEFAULT_REPO_OWNER - repo = repo or DEFAULT_REPO_NAME - if number: - issue = await github.get_thread(int(number), owner, repo) - if issue is not None: - found_entries.append((gh_match[0], issue)) - elif sha: - commit = await github.get_commit(sha, owner, repo) - if commit is not None: - found_entries.append((gh_match[0], commit)) - elif ptbcontrib: - contrib = github.ptb_contribs.get(ptbcontrib) - if contrib: - found_entries.append((gh_match[0], contrib)) - - else: - # Parse fuzzy search next - for match in ENCLOSED_REGEX.finditer(no_entity_text): + for match in GITHUB_PATTERN.finditer(no_entity_text): + logging.debug(match.groupdict()) + owner, repo, number, sha, ptbcontrib = ( + cast(str, match.groupdict()[x]) + for x in ("owner", "repo", "number", "sha", "ptbcontrib") + ) + if number or sha or ptbcontrib: + github_matches.append((match.start(), (owner, repo, number, sha, ptbcontrib))) + + for gh_match in github_matches: + last = keep_typing( + last, + cast(Chat, update.effective_chat), + ChatAction.TYPING, + application=context.application, + ) + owner, repo, number, sha, ptbcontrib = gh_match[1] + owner = owner or DEFAULT_REPO_OWNER + repo = repo or DEFAULT_REPO_NAME + if number: + issue = await github.get_thread(int(number), owner, repo) + if issue is not None: + found_entries.append((gh_match[0], issue)) + elif sha: + commit = await github.get_commit(sha, owner, repo) + if commit is not None: + found_entries.append((gh_match[0], commit)) + elif ptbcontrib: + contrib = github.ptb_contribs.get(ptbcontrib) + if contrib: + found_entries.append((gh_match[0], contrib)) + + # Parse fuzzy search next, if requested. Here we use message.text instead of no_entity_text + # to avoid tlds like .bot and .app to mess things up for us + if message.text.startswith("!search") or message.text.endswith("!search"): + for match in ENCLOSED_REGEX.finditer(message.text): last = keep_typing( last, cast(Chat, update.effective_chat), diff --git a/components/errorhandler.py b/components/errorhandler.py index 53ad011..e9ddd23 100644 --- a/components/errorhandler.py +++ b/components/errorhandler.py @@ -2,7 +2,7 @@ import json import logging import traceback -from typing import Optional, cast +from typing import cast from telegram import Update from telegram.error import BadRequest @@ -34,20 +34,6 @@ async def error_handler(update: object, context: CallbackContext) -> None: ) message_2 = f"
{html.escape(tb_string)}" - user_id: Optional[int] = None - message_3 = "" - if isinstance(update, Update) and update.effective_user: - user_id = update.effective_user.id - if context.job: - user_id = context.job.user_id - if user_id: - data_str = html.escape( - json.dumps(context.application.user_data[user_id], indent=2, ensure_ascii=False) - ) - message_3 = ( - "
user_data
associated with this exception:\n\n" f"{data_str}" - ) - # Finally, send the messages # We send update and traceback in two parts to reduce the chance of hitting max length try: @@ -61,11 +47,6 @@ async def error_handler(update: object, context: CallbackContext) -> None: f"Hey.\nThe error
{html.escape(str(context.error))}
happened."
f" The traceback is too long to send, but it was written to the log."
)
- sent_message = await context.bot.send_message(
- chat_id=ERROR_CHANNEL_CHAT_ID, text=message
- )
+ await context.bot.send_message(chat_id=ERROR_CHANNEL_CHAT_ID, text=message)
else:
raise exc
- finally:
- if message_3:
- await sent_message.reply_html(message_3)
diff --git a/components/joinrequests.py b/components/joinrequests.py
index b03ea6a..1687f0a 100644
--- a/components/joinrequests.py
+++ b/components/joinrequests.py
@@ -1,5 +1,5 @@
import datetime
-from typing import Any, Dict, Tuple, Union, cast
+from typing import Tuple, Union, cast
from telegram import (
CallbackQuery,
@@ -30,19 +30,14 @@ def get_dtm_str() -> str:
async def approve_user(
user: Union[int, User], chat_id: int, group_name: str, context: ContextTypes.DEFAULT_TYPE
) -> None:
- user_data = cast(Dict[Any, Any], context.user_data)
try:
if isinstance(user, User):
await user.approve_join_request(chat_id=chat_id)
else:
await context.bot.approve_chat_join_request(user_id=user, chat_id=chat_id)
- user_data.setdefault(int(chat_id), {}).setdefault("approved", []).append(get_dtm_str())
except BadRequest as exc:
user_mention = f"{user.username} - {user.id}" if isinstance(user, User) else str(user)
error_message = f"{exc} - {user_mention} - {group_name}"
- user_data.setdefault(int(chat_id), {}).setdefault("approve failed", []).append(
- f"{get_dtm_str()}: {exc}"
- )
raise BadRequest(error_message) from exc
except Forbidden as exc:
if "user is deactivated" not in exc.message:
@@ -52,19 +47,14 @@ async def approve_user(
async def decline_user(
user: Union[int, User], chat_id: int, group_name: str, context: ContextTypes.DEFAULT_TYPE
) -> None:
- user_data = cast(Dict[Any, Any], context.user_data)
try:
if isinstance(user, User):
await user.decline_join_request(chat_id=chat_id)
else:
await context.bot.decline_chat_join_request(user_id=user, chat_id=chat_id)
- user_data.setdefault(int(chat_id), {}).setdefault("declined", []).append(get_dtm_str())
except BadRequest as exc:
user_mention = f"{user.username} - {user.id}" if isinstance(user, User) else str(user)
error_message = f"{exc} - {user_mention} - {group_name}"
- user_data.setdefault(int(chat_id), {}).setdefault("declined failed", []).append(
- f"{get_dtm_str()}: {exc}"
- )
raise BadRequest(error_message) from exc
except Forbidden as exc:
if "user is deactivated" not in exc.message:
@@ -81,16 +71,9 @@ async def join_request_callback(update: Update, context: ContextTypes.DEFAULT_TY
# No need to ping the user again if we already did
return
- user_data = cast(Dict[Any, Any], context.user_data)
on_topic = join_request.chat.username == ONTOPIC_USERNAME
group_mention = ONTOPIC_CHAT_ID if on_topic else OFFTOPIC_CHAT_ID
- user_data.setdefault(int(chat_id), {}).setdefault("received join request", []).append(
- get_dtm_str()
- )
- user_data[int(chat_id)]["user_chat_id"] = user_chat_id
- user_data[int(chat_id)]["user_id"] = user.id
-
text = (
f"Hi, {user.mention_html()}! I'm {context.bot.bot.mention_html()}, the "
f"guardian of the group {group_mention}, that you requested to join.\n\nBefore you can "
@@ -114,7 +97,6 @@ async def join_request_callback(update: Update, context: ContextTypes.DEFAULT_TY
# If the user blocked the bot, let's give the admins a chance to handle that
# TG also notifies the user and forwards the message once the user unblocks the bot, but
# forwarding it still doesn't hurt ...
- user_data.setdefault("blocked bot", []).append(get_dtm_str())
text = (
f"User {user.mention_html()} with id {user.id} requested to join the group "
f"{join_request.chat.username} but has blocked me. Please manually handle this."
@@ -133,23 +115,16 @@ async def join_request_callback(update: Update, context: ContextTypes.DEFAULT_TY
async def join_request_buttons(update: Update, context: ContextTypes.DEFAULT_TYPE) -> None:
- user_data = cast(Dict[Any, Any], context.user_data)
callback_query = cast(CallbackQuery, update.callback_query)
user = cast(User, update.effective_user)
_, press, chat_id = cast(str, callback_query.data).split()
if press == "2":
- user_data.setdefault(int(chat_id), {}).setdefault("pressed button 2", []).append(
- get_dtm_str()
- )
jobs = cast(JobQueue, context.job_queue).get_jobs_by_name(
f"JOIN_TIMEOUT {chat_id} {user.id}"
)
if jobs:
for job in jobs:
job.schedule_removal()
- user_data.setdefault(int(chat_id), {}).setdefault(
- "removed join timeout", []
- ).append(get_dtm_str())
try:
await approve_user(
@@ -162,9 +137,6 @@ async def join_request_buttons(update: Update, context: ContextTypes.DEFAULT_TYP
reply_markup = None
else:
- user_data.setdefault(int(chat_id), {}).setdefault("pressed button 1", []).append(
- get_dtm_str()
- )
reply_markup = InlineKeyboardMarkup.from_button(
InlineKeyboardButton(
text="⚠️ Tap again to confirm",
@@ -193,3 +165,11 @@ async def join_request_timeout_job(context: ContextTypes.DEFAULT_TYPE) -> None:
except Forbidden as exc:
if "user is deactivated" not in exc.message:
raise exc
+ except BadRequest as exc:
+ # These apparently happen frequently, e.g. when user clear the chat
+ if exc.message not in [
+ "Message to edit not found",
+ "Can't access the chat",
+ "Chat not found",
+ ]:
+ raise exc
diff --git a/components/taghints.py b/components/taghints.py
index ae873e3..0ac228f 100644
--- a/components/taghints.py
+++ b/components/taghints.py
@@ -120,8 +120,9 @@
"pastebin": {
"message": (
"{query} Please post code or tracebacks using a pastebin rather than via plain text "
- "or a picture. https://pastebin.com/ is quite popular, but there are many "
- "alternatives out there. Of course, for very short snippets, text is fine. Please at "
+ "or a picture. https://pastebin.com/ is quite popular, but there are "
+ "many alternatives "
+ "out there. Of course, for very short snippets, text is fine. Please at "
"least format it as monospace in that case."
),
"help": "Ask users not to post code as text or images.",
@@ -283,6 +284,18 @@
"help": "Tell users not to use AI/LLM generated answers",
"group_command": True,
},
+ "traceback": {
+ "message": (
+ "{query} Please show the full traceback via a pastebin. Make sure to include "
+ "everything from the first Traceback (most recent call last):
until the "
+ "last error message. https://pastebin.com/ is a popular pastebin service, but there "
+ "are many alternatives out "
+ "there."
+ ),
+ "default": "Hey.",
+ "help": "Ask for the full traceback",
+ "group_command": True,
+ },
}
diff --git a/rules_bot.py b/rules_bot.py
index 511632f..9051850 100644
--- a/rules_bot.py
+++ b/rules_bot.py
@@ -1,4 +1,3 @@
-import asyncio
import configparser
import logging
import os
@@ -111,6 +110,7 @@ def main() -> None:
.token(config["KEYS"]["bot_api"])
.defaults(defaults)
.post_init(post_init)
+ .post_shutdown(post_shutdown)
.job_queue(RulesJobQueue())
.build()
)
@@ -207,8 +207,6 @@ def main() -> None:
application.add_error_handler(error_handler)
application.run_polling(allowed_updates=Update.ALL_TYPES, close_loop=False)
- # Can be used in AppBuilder.post_shutdown once #3126 is released
- asyncio.get_event_loop().run_until_complete(post_shutdown(application))
if __name__ == "__main__":