mirror of
https://github.com/miloszowi/everyone-mention-telegram-bot.git
synced 2025-10-24 07:55:50 +00:00
Compare commits
10 Commits
Author | SHA1 | Date | |
---|---|---|---|
|
e44f4b75a5 | ||
|
e242584974 | ||
|
878091deae | ||
|
a6b441b197 | ||
|
4140098594 | ||
|
a02c09d19d | ||
|
df374e31ba | ||
|
48ceab008d | ||
|
568ac15b5e | ||
|
6790426ba2 |
14
CHANGELOG.md
14
CHANGELOG.md
@@ -1,6 +1,20 @@
|
||||
# Change Log
|
||||
All notable changes to this project will be documented in this file.
|
||||
|
||||
## [0.3.2] - 28.02.2023
|
||||
### Changed
|
||||
- fixed bug with missing '+' in replier
|
||||
## [0.3.1] - 28.02.2023
|
||||
### Changed
|
||||
- fixed markdown replier to respect restricted characters provided in the [api docs](https://core.telegram.org/bots/api#markdownv2-style)
|
||||
## [0.3.0] - 12.11.2021
|
||||
### Added
|
||||
- Dynamic mentioning by `@` character
|
||||
### Changed
|
||||
- `start` text
|
||||
- Group name validation - those are forbidden now - `all`, `channel`, `chat`, `everyone`, `group`, `here`
|
||||
### Deleted
|
||||
- mentioning user that performed `/everyone` or dynamic mention in bot response
|
||||
## [0.2.0] - 26.10.2021
|
||||
### Added
|
||||
- Inline Mode for `join`, `leave` & `everyone`
|
||||
|
36
README.md
36
README.md
@@ -5,6 +5,8 @@
|
||||
# Contents
|
||||
|
||||
* [Description](#description)
|
||||
* [Usage](#usage)
|
||||
* [Dynamic Mentioning](#dynamic-mentioning)
|
||||
* [Commands](#commands)
|
||||
* [`/join`](#join)
|
||||
* [`/leave`](#leave)
|
||||
@@ -12,8 +14,7 @@
|
||||
* [`/groups`](#groups)
|
||||
* [`/start`](#start)
|
||||
* [Example command flow](#example-command-flow)
|
||||
* [Inline Mode](#inline-mode)
|
||||
* [Usage](#usage)
|
||||
* [Inline Mode Usage](#inline-mode-usage)
|
||||
* [Getting started.](#getting-started)
|
||||
* [Requirements](#requirements)
|
||||
* [Installation](#installation)
|
||||
@@ -24,6 +25,32 @@ Everyone Mention Bot is simple, but useful telegram bot to gather group members
|
||||
|
||||
You can create groups per chat to mention every user that joined the group by calling one command instead of mentioning them one by one.
|
||||
|
||||
## Usage
|
||||
First, users need to join the group to let mentioning them, to do that, they simply need to join specific group.
|
||||
It can be done in 2 ways:
|
||||
- Command [`/join`](#join)
|
||||
- [Inline Mode](#inline-mode-usage)
|
||||
|
||||
Users that have joined the group can be mentioned in 3 ways:
|
||||
- [Dynaminc Mentioning](#dynamic-mentioning) by `@`, for example `@everyone`
|
||||
- Command [`/everyone`](#everyone)
|
||||
- [Inline Mode](#inline-mode-usage)
|
||||
|
||||
To leave the group use one of the two ways:
|
||||
- Command [`/leave`](#leave)
|
||||
- [Inline Mode](#inline-mode-usage)
|
||||
|
||||
To display available groups:
|
||||
- Command [`/groups`](#groups)
|
||||
|
||||
## Dynamic mentioning
|
||||
You can use `@` character (as you would mention a user) to mention specific group.
|
||||
|
||||
All the below will mention users from `default` group.
|
||||
|
||||
`@all`, `@channel`, `@chat`, `@everyone`, `@group`, `@here`.
|
||||
|
||||
If you did create a group named `gaming`, you can simply use `@gaming` in your text to mention them all.
|
||||
## Commands
|
||||
*Important*: `{group-name}` is not required, if not given, it will be set to `default`.
|
||||
### `/join`
|
||||
@@ -85,10 +112,7 @@ Start & Help message
|
||||
### Example command flow
|
||||

|
||||
|
||||
## Inline Mode
|
||||
Using Inline Mode is recommended because policy of bots with privacy mode enabled (https://core.telegram.org/bots/faq#what-messages-will-my-bot-get) says that command trigger is sent (without mentioning the bot) only to the last mentioned bot. So if you do have multiple bots in current chat, I might not receive your command!
|
||||
|
||||
### Usage
|
||||
### Inline Mode Usage
|
||||
To use inline mode, type `@everyone_mention_bot` in telegram message input or click on the `Inline Mode` button from `/start` command.
|
||||
|
||||

|
||||
|
@@ -1,5 +1,5 @@
|
||||
__all__ = [
|
||||
'abstractHandler', 'everyoneHandler', 'groupsHandler',
|
||||
'inlineQueryHandler', 'joinHandler', 'leaveHandler',
|
||||
'startHandler'
|
||||
'startHandler', 'dynamicMentionHandler'
|
||||
]
|
||||
|
@@ -9,6 +9,7 @@ from bot.message.replier import Replier
|
||||
from exception.actionNotAllowedException import ActionNotAllowedException
|
||||
from exception.invalidActionException import InvalidActionException
|
||||
from exception.invalidArgumentException import InvalidArgumentException
|
||||
from exception.notFoundException import NotFoundException
|
||||
from logger import Logger
|
||||
|
||||
|
||||
@@ -30,6 +31,8 @@ class AbstractHandler:
|
||||
Logger.action(self.inbound, self.action)
|
||||
except (InvalidActionException, InvalidArgumentException, ActionNotAllowedException) as e:
|
||||
Replier.markdown(update, str(e))
|
||||
except NotFoundException:
|
||||
pass # probably just mentioning user
|
||||
except Exception as e:
|
||||
Logger.exception(e)
|
||||
|
||||
|
28
src/bot/handler/dynamicMentionHandler.py
Normal file
28
src/bot/handler/dynamicMentionHandler.py
Normal file
@@ -0,0 +1,28 @@
|
||||
import re
|
||||
|
||||
from telegram.ext import Filters, MessageHandler
|
||||
from telegram.ext.callbackcontext import CallbackContext
|
||||
from telegram.update import Update
|
||||
|
||||
from bot.handler.abstractHandler import AbstractHandler
|
||||
from bot.message.replier import Replier
|
||||
from repository.chatRepository import ChatRepository
|
||||
from utils.messageBuilder import MessageBuilder
|
||||
|
||||
|
||||
class DynamicMentionHandler(AbstractHandler):
|
||||
bot_handler: MessageHandler
|
||||
chat_repository: ChatRepository
|
||||
action: str = 'dynamic-mention'
|
||||
|
||||
def __init__(self) -> None:
|
||||
self.bot_handler = MessageHandler(
|
||||
Filters.regex(re.compile(r'@[^ ]')),
|
||||
self.wrap
|
||||
)
|
||||
self.chat_repository = ChatRepository()
|
||||
|
||||
def handle(self, update: Update, context: CallbackContext) -> None:
|
||||
users = self.chat_repository.get_users_for_group(self.inbound)
|
||||
|
||||
Replier.markdown(update, MessageBuilder.mention_message(users))
|
@@ -8,24 +8,21 @@ from config.contents import mention_failed
|
||||
from exception.invalidActionException import InvalidActionException
|
||||
from exception.notFoundException import NotFoundException
|
||||
from repository.chatRepository import ChatRepository
|
||||
from repository.userRepository import UserRepository
|
||||
from utils.messageBuilder import MessageBuilder
|
||||
|
||||
|
||||
class EveryoneHandler(AbstractHandler):
|
||||
bot_handler: CommandHandler
|
||||
chat_repository: ChatRepository
|
||||
user_repository: UserRepository
|
||||
action: str = 'everyone'
|
||||
|
||||
def __init__(self) -> None:
|
||||
self.bot_handler = CommandHandler(self.action, self.wrap)
|
||||
self.chat_repository = ChatRepository()
|
||||
self.user_repository = UserRepository()
|
||||
|
||||
def handle(self, update: Update, context: CallbackContext) -> None:
|
||||
try:
|
||||
users = self.chat_repository.get_users_for_group(self.inbound.chat_id, self.inbound.group_name)
|
||||
users = self.chat_repository.get_users_for_group(self.inbound)
|
||||
|
||||
Replier.markdown(update, MessageBuilder.mention_message(users))
|
||||
except NotFoundException as e:
|
||||
|
@@ -3,9 +3,11 @@ from __future__ import annotations
|
||||
from dataclasses import dataclass
|
||||
|
||||
import names
|
||||
import re
|
||||
from telegram.ext.callbackcontext import CallbackContext
|
||||
from telegram.update import Update
|
||||
|
||||
from exception.invalidActionException import InvalidActionException
|
||||
from validator.accessValidator import AccessValidator
|
||||
from validator.groupNameValidator import GroupNameValidator
|
||||
|
||||
@@ -24,14 +26,25 @@ class InboundMessage:
|
||||
user_id = str(update.effective_user.id)
|
||||
AccessValidator.validate(user_id)
|
||||
|
||||
message_content = update.edited_message.text if update.edited_message else update.message.text
|
||||
message_content = message_content.replace("\n", " ")
|
||||
chat_id = str(update.effective_chat.id)
|
||||
group_name = InboundMessage.default_group
|
||||
|
||||
# done upon resolving a command action
|
||||
if context.args and context.args[0] and group_specific:
|
||||
group_name = str(context.args[0]).lower()
|
||||
|
||||
GroupNameValidator.validate(group_name)
|
||||
|
||||
# done upon resolving a message handler action
|
||||
if '@' in message_content and '@everyone_mention_bot' not in message_content:
|
||||
searched_message_part = [part for part in message_content.split(' ') if '@' in part][0]
|
||||
group_name = re.sub(r'\W+', '', searched_message_part).lower()
|
||||
|
||||
if group_name in GroupNameValidator.FORBIDDEN_GROUP_NAMES:
|
||||
group_name = InboundMessage.default_group
|
||||
|
||||
username = update.effective_user.username or update.effective_user.first_name
|
||||
|
||||
if not username:
|
||||
|
@@ -8,24 +8,30 @@ from logger import Logger
|
||||
|
||||
|
||||
class Replier:
|
||||
|
||||
@staticmethod
|
||||
def interpolate(content: str, inbound_message: InboundMessage):
|
||||
return content.format(
|
||||
formatted = content.format(
|
||||
mention_markdown(inbound_message.user_id, inbound_message.username),
|
||||
inbound_message.group_name
|
||||
)
|
||||
|
||||
telegramRestrictionCharacters = ['_', '*', '[', ']', '(', ')', '~', '`', '>', '#', '+', '-', '=', '|', '{', '}', '.', '!']
|
||||
|
||||
for character in telegramRestrictionCharacters:
|
||||
formatted.replace(character, "\\" + character)
|
||||
|
||||
return formatted
|
||||
|
||||
@staticmethod
|
||||
def markdown(update: Update, message: str, reply_markup: Optional[InlineKeyboardMarkup] = None) -> None:
|
||||
try:
|
||||
update.effective_message.reply_markdown_v2(message, reply_markup=reply_markup)
|
||||
except Exception as err:
|
||||
Logger.error(str(err))
|
||||
Logger.error("replier.markdown error: " + str(err))
|
||||
|
||||
@staticmethod
|
||||
def html(update: Update, html: str, reply_markup: Optional[InlineKeyboardMarkup] = None) -> None:
|
||||
try:
|
||||
update.effective_message.reply_html(html, reply_markup=reply_markup, disable_web_page_preview=True)
|
||||
except Exception as err:
|
||||
Logger.error(str(err))
|
||||
Logger.error("replier.html error: " + str(err))
|
||||
|
@@ -10,13 +10,16 @@ no_groups = 'There are no groups for this chat'
|
||||
start_text = """
|
||||
Hello!
|
||||
@everyone_mention_bot here.
|
||||
|
||||
<b>Description</b>:
|
||||
I <b>do not</b> have access to your messages!
|
||||
I am here to help you with multiple user mentions.
|
||||
|
||||
<b>Usage</b>:
|
||||
Users that joined the group by <code>/join</code> command, can be mentioned after calling <code>/everyone</code> command.
|
||||
Users that joined the group by <code>/join</code> command,
|
||||
can be mentioned after typing one of those in your message:
|
||||
<code>@all</code>, <code>@channel</code>, <code>@chat</code>, <code>@everyone</code>, <code>@group</code> or <code>@here</code>.
|
||||
|
||||
If you did create a group named <code>gaming</code>, simply use <code>@gaming</code> to call users from that group.
|
||||
|
||||
You can also use <code>/everyone</code> command.
|
||||
|
||||
<b>Commands</b>:
|
||||
<pre>/join {group-name}</pre>
|
||||
@@ -36,6 +39,4 @@ Show start & help text
|
||||
|
||||
<b>Please note</b>
|
||||
<code>{group-name}</code> is not required, <code>default</code> if not given.
|
||||
|
||||
If your chat does have multiple bots <b>I might not receive your command</b> according to <a href="https://core.telegram.org/bots/faq#what-messages-will-my-bot-get">policy of bots with privacy mode enabled</a> - use <code>Inline Mode</code> to avoid this.
|
||||
"""
|
||||
|
@@ -39,12 +39,17 @@ class ChatRepository(AbstractRepository):
|
||||
|
||||
return Chat.from_mongo_document(chat)
|
||||
|
||||
def get_users_for_group(self, chat_id: str, group: str) -> Iterable[User]:
|
||||
chat = self.get(chat_id)
|
||||
if not chat.groups.get(group):
|
||||
def get_users_for_group(self, inbound: InboundMessage) -> Iterable[User]:
|
||||
chat = self.get(inbound.chat_id)
|
||||
if not chat.groups.get(inbound.group_name):
|
||||
raise NotFoundException
|
||||
|
||||
return [self.user_repository.get(user_id) for user_id in chat.groups.get(group)]
|
||||
users = [self.user_repository.get(user_id) for user_id in chat.groups.get(inbound.group_name) if user_id != inbound.user_id]
|
||||
|
||||
if not users:
|
||||
raise NotFoundException
|
||||
|
||||
return users
|
||||
|
||||
def save(self, chat: Chat) -> None:
|
||||
self.database_client.save(
|
||||
|
@@ -5,6 +5,7 @@ from exception.invalidArgumentException import InvalidArgumentException
|
||||
|
||||
class GroupNameValidator:
|
||||
MAX_GROUP_NAME_LENGTH: int = 40
|
||||
FORBIDDEN_GROUP_NAMES = ['all', 'channel', 'chat', 'everyone', 'group', 'here']
|
||||
|
||||
@staticmethod
|
||||
def validate(group: str) -> None:
|
||||
@@ -15,3 +16,6 @@ class GroupNameValidator:
|
||||
|
||||
if len(group) > GroupNameValidator.MAX_GROUP_NAME_LENGTH:
|
||||
raise InvalidArgumentException(re.escape(f'Group name length can not be greater than {GroupNameValidator.MAX_GROUP_NAME_LENGTH}.'))
|
||||
|
||||
if group in GroupNameValidator.FORBIDDEN_GROUP_NAMES:
|
||||
raise InvalidArgumentException(re.escape(f'This group name is forbidden, please try with other name.'))
|
||||
|
Reference in New Issue
Block a user