Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat: warn and mute command #142

Draft
wants to merge 28 commits into
base: main
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from 13 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
f91bdc3
pre-refactoring warn command
alepiaz Nov 25, 2023
3d8ded5
merged to main
alepiaz Nov 25, 2023
e83c83a
added warn and mute to user
alepiaz Nov 25, 2023
1bea2f0
added docstrings
alepiaz Nov 26, 2023
7c6d00f
added warn to command handlers
alepiaz Nov 26, 2023
5ad7ca1
feat: added warned_users to db and better handling of ban in warn con…
alepiaz Nov 26, 2023
003656e
chore: runned isort linting
alepiaz Nov 26, 2023
95ea13b
feat: added daily job to clean warned users if their warn time has ex…
alepiaz Nov 27, 2023
ed9b959
chore: linted code for job_handlers
alepiaz Nov 27, 2023
5f5a31a
fix: updated setting.yaml with new warn settings
alepiaz Nov 27, 2023
f3c2c9a
fix: updated settings.yaml.types
alepiaz Nov 27, 2023
f09d904
chore: removed comments
alepiaz Nov 27, 2023
c7732d1
chore: Update settings.yaml.types
alepiaz Nov 27, 2023
e7f417b
chore: removed unneeded default settings
alepiaz Nov 27, 2023
f0d3625
chore: adjusting docstrings for user.py
alepiaz Nov 27, 2023
76022db
chore: removed unneeded comment
alepiaz Nov 27, 2023
4bc2e84
fix: clean_warned_users is now deleting the rows
alepiaz Nov 27, 2023
1871ffc
removed user name from group message
alepiaz Nov 27, 2023
6fa7dae
Merge branch 'feat/warn_command' of github.com:UNICT-DMI/Telegram-Spo…
alepiaz Nov 27, 2023
7d1c699
feat: added mute command handler
alepiaz Nov 30, 2023
40c342d
feat: unmuting user with job_queue
alepiaz Nov 30, 2023
ca26e18
feat: added function to unrestrict the user from a mute
alepiaz Nov 30, 2023
6620ffb
feat: included mute cmd and is_admin filter in app_handler
alepiaz Nov 30, 2023
94a0844
feat: added function to unban from callback
alepiaz Nov 30, 2023
7380b9b
feat: added a custom filter to check if sender is admin
alepiaz Nov 30, 2023
af0f98d
feat: added warned and muted users table to sql files
alepiaz Nov 30, 2023
f724895
chore: isorted new files
alepiaz Nov 30, 2023
f6b48e7
test: edit get_administrators usage
alepiaz Dec 3, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 7 additions & 0 deletions src/spotted/config/db/post_db_init.sql
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,13 @@ CREATE TABLE IF NOT EXISTS banned_users
PRIMARY KEY (user_id)
);
-----
CREATE TABLE IF NOT EXISTS warned_users
(
user_id BIGINT NOT NULL,
warn_date TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (user_id, warn_date)
);
-----
CREATE TABLE IF NOT EXISTS spot_report
(
user_id BIGINT NOT NULL,
Expand Down
4 changes: 3 additions & 1 deletion src/spotted/config/yaml/settings.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -18,5 +18,7 @@ post:
delete_anonymous_comments: true
reject_after_autoreply: true
autoreplies_per_page: 6
warn_after_days: 7
max_n_warns : 3
token: ""
bot_tag: "@bot_tag"
bot_tag: "@bot_tag"
24 changes: 24 additions & 0 deletions src/spotted/config/yaml/settings.yaml.default
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
debug:
local_log: false
reset_on_load: false
log_file: "logs/spotted.log"
log_error_file: "logs/spotted_error.log"
db_file: "spotted.sqlite3"
post:
community_group_id: -1
channel_id: -2
channel_tag: "@channel_tag"
comments: true
admin_group_id: -3
n_votes: 2
remove_after_h: 12
report: true
report_wait_mins: 30
replace_anonymous_comments: true
delete_anonymous_comments: true
reject_after_autoreply: true
autoreplies_per_page: 6
warn_after_days: 10
max_n_warns : 3
token: ""
bot_tag: "@bot_tag"
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
2 changes: 2 additions & 0 deletions src/spotted/config/yaml/settings.yaml.types
Original file line number Diff line number Diff line change
Expand Up @@ -17,5 +17,7 @@ post:
delete_anonymous_comments: bool
reject_after_autoreply: bool
autoreplies_per_page: int
warn_after_days: int
max_n_warns: int
token: str
bot_tag: str
47 changes: 46 additions & 1 deletion src/spotted/data/user.py
Original file line number Diff line number Diff line change
@@ -1,10 +1,11 @@
"""Users management"""
from dataclasses import dataclass
from datetime import datetime
from datetime import datetime, timedelta
from random import choice

from telegram import Bot

from .config import Config
from .data_reader import read_md
from .db_manager import DbManager
from .pending_post import PendingPost
Expand Down Expand Up @@ -41,6 +42,15 @@ def is_credited(self) -> bool:
"""If the user is in the credited list"""
return DbManager.count_from(table_name="credited_users", where="user_id = %s", where_args=(self.user_id,)) == 1

def get_n_warns(self) -> int:
"""_summary_

Returns:
int: _description_
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
"""
count = DbManager.count_from(table_name="warned_users", where="user_id = %s", where_args=(self.user_id,))
return count if count else 0

@classmethod
def banned_users(cls) -> "list[User]":
"""Returns a list of all the banned users"""
Expand Down Expand Up @@ -83,6 +93,7 @@ def ban(self):

if not self.is_banned:
DbManager.insert_into(table_name="banned_users", columns=("user_id",), values=(self.user_id,))
DbManager.delete_from(table_name="warned_users", where="user_id = %s", where_args=(self.user_id,))

def sban(self) -> bool:
"""Removes the user from the banned list
Expand All @@ -95,6 +106,40 @@ def sban(self) -> bool:
return True
return False

def mute(self, bot: Bot, days: int = 0):
"""_summary_

Args:
bot (Bot): the telegram bot
days (int, optional): The number of days the user should be muted for.
Defaults to 0.
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
"""
bot.restrict_chat_member(
chat_id=Config.post_get("channel_id"),
user_id=self.user_id,
can_send_messages=False,
can_send_media_messages=False,
can_send_other_messages=False,
can_add_web_page_previews=False,
)
current_date_time = datetime.now()
expiration_date_time = (current_date_time + timedelta(days=days)).strftime("%Y-%m-%d %H:%M:%S")
DbManager.insert_into(
table_name="muted_users",
columns=("user_id", "expiration_date"),
values=(self.user_id, expiration_date_time),
)
alepiaz marked this conversation as resolved.
Show resolved Hide resolved

def warn(self):
"""Increase the number of warns of a user
If this is number would reach 3 the user is banned
alepiaz marked this conversation as resolved.
Show resolved Hide resolved

Args:
bot (Bot): the telegram bot
alepiaz marked this conversation as resolved.
Show resolved Hide resolved

"""
DbManager.insert_into(table_name="warned_users", columns=("user_id",), values=(self.user_id,))

def become_anonym(self) -> bool:
"""Removes the user from the credited list, if he was present

Expand Down
28 changes: 16 additions & 12 deletions src/spotted/handlers/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
"""Modules that handle the events the bot recognizes and reacts to"""
from datetime import time
from datetime import time, timedelta
from warnings import filterwarnings

from pytz import utc
Expand Down Expand Up @@ -27,7 +27,7 @@
from .follow_spot import follow_spot_callback
from .forwarded_post import forwarded_post_msg
from .help import help_cmd
from .job_handlers import clean_pending_job, db_backup_job
from .job_handlers import clean_pending_job, clean_warned_users, db_backup_job
from .purge import purge_cmd
from .reload import reload_cmd
from .reply import reply_cmd
Expand All @@ -38,6 +38,7 @@
from .settings import settings_callback, settings_cmd
from .spot import spot_conv_handler
from .start import start_cmd
from .warn import warn_cmd


async def add_commands(app: Application):
Expand Down Expand Up @@ -84,7 +85,8 @@ def add_handlers(app: Application):
if Config.settings_get("debug", "local_log"): # add MessageHandler only if log_message is enabled
app.add_handler(MessageHandler(filters.ALL, log_message), 1)

admin_filter = filters.Chat(chat_id=Config.post_get("admin_group_id"))
# admin_message_filter = filters.MessageFilter
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
admin_group_filter = filters.Chat(chat_id=Config.post_get("admin_group_id"))
community_filter = filters.Chat(chat_id=Config.post_get("community_group_id"))

# Error handler
Expand All @@ -97,23 +99,24 @@ def add_handlers(app: Application):

# Command handlers
app.add_handler(CommandHandler("start", start_cmd, filters=filters.ChatType.PRIVATE))
app.add_handler(CommandHandler("help", help_cmd, filters=filters.ChatType.PRIVATE | admin_filter))
app.add_handler(CommandHandler("help", help_cmd, filters=filters.ChatType.PRIVATE | admin_group_filter))
app.add_handler(CommandHandler("rules", rules_cmd, filters=filters.ChatType.PRIVATE))
app.add_handler(CommandHandler("settings", settings_cmd, filters=filters.ChatType.PRIVATE))
# it must be after the conversation handler's 'cancel'
app.add_handler(CommandHandler("cancel", cancel_cmd, filters=filters.ChatType.PRIVATE))

# Command handlers: Admin commands
app.add_handler(CommandHandler("sban", sban_cmd, filters=admin_filter))
app.add_handler(CommandHandler("clean_pending", clean_pending_cmd, filters=admin_filter))
app.add_handler(CommandHandler("db_backup", db_backup_cmd, filters=admin_filter))
app.add_handler(CommandHandler("purge", purge_cmd, filters=admin_filter))
app.add_handler(CommandHandler("reload", reload_cmd, filters=admin_filter))
app.add_handler(CommandHandler("sban", sban_cmd, filters=admin_group_filter))
app.add_handler(CommandHandler("clean_pending", clean_pending_cmd, filters=admin_group_filter))
app.add_handler(CommandHandler("db_backup", db_backup_cmd, filters=admin_group_filter))
app.add_handler(CommandHandler("purge", purge_cmd, filters=admin_group_filter))
app.add_handler(CommandHandler("reload", reload_cmd, filters=admin_group_filter))
app.add_handler(CommandHandler("warn", warn_cmd, filters=community_filter))

# MessageHandler
app.add_handler(MessageHandler(filters.REPLY & admin_filter & filters.Regex(r"^/ban$"), ban_cmd))
app.add_handler(MessageHandler(filters.REPLY & admin_filter & filters.Regex(r"^/reply"), reply_cmd))
app.add_handler(MessageHandler(filters.REPLY & admin_filter & filters.Regex(r"^/autoreply"), autoreply_cmd))
app.add_handler(MessageHandler(filters.REPLY & admin_group_filter & filters.Regex(r"^/ban$"), ban_cmd))
app.add_handler(MessageHandler(filters.REPLY & admin_group_filter & filters.Regex(r"^/reply"), reply_cmd))
app.add_handler(MessageHandler(filters.REPLY & admin_group_filter & filters.Regex(r"^/autoreply"), autoreply_cmd))

# Callback handlers
app.add_handler(CallbackQueryHandler(settings_callback, pattern=r"^settings\.*"))
Expand Down Expand Up @@ -144,3 +147,4 @@ def add_jobs(app: Application):
"""
app.job_queue.run_daily(clean_pending_job, time=time(hour=5, tzinfo=utc)) # run each day at 05:00 utc
app.job_queue.run_daily(db_backup_job, time=time(hour=5, tzinfo=utc)) # run each day at 05:00 utc
app.job_queue.run_daily(clean_warned_users, time=time(hour=5, tzinfo=utc)) # run each day at 05:00 utc
18 changes: 17 additions & 1 deletion src/spotted/handlers/job_handlers.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
from telegram.error import BadRequest, Forbidden
from telegram.ext import CallbackContext

from spotted.data import Config, PendingPost
from spotted.data import Config, DbManager, PendingPost
from spotted.debug import logger
from spotted.utils import EventInfo

Expand Down Expand Up @@ -62,3 +62,19 @@ async def db_backup_job(context: CallbackContext):
)
except Exception as ex: # pylint: disable=broad-except
await context.bot.send_message(chat_id=admin_group_id, text=f"✖️ Impossibile effettuare il backup\n\n{ex}")


async def clean_warned_users():
"""Job called each day at 05:00 utc.
Removed users who have been warned for longer than setting duration
Args:
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
context (CallbackContext): _description_
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
"""
warn_expiration = datetime.now() + timedelta(days=Config.post_get("warn_after_days"))
DbManager.select_from(
table_name="warned_users",
select=("user_id", "MAX(warn_time)"),
where="warn_time > %s",
where_args=(warn_expiration,),
group_by="user_id",
)
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
36 changes: 36 additions & 0 deletions src/spotted/handlers/warn.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
"""/warn command"""
from telegram import Update
from telegram.ext import CallbackContext

from spotted.data import Config, User
from spotted.utils import EventInfo


async def warn_cmd(update: Update, context: CallbackContext):
"""Handles the /warn command.
Warn a user by replying to one of his message in the comment group with /warn

Args:
update: update event
context: context passed by the handler
"""
info = EventInfo.from_message(update, context)
g_message = update.message.reply_to_message
if g_message is None:
await info.bot.send_message(
chat_id=info.chat_id,
text="Per warnare qualcuno, rispondi al suo commento con /warn",
)
return
user = User(g_message.from_user.id)
n_warns = user.get_n_warns()
text = f"L'utente {g_message.from_user.name} "
alepiaz marked this conversation as resolved.
Show resolved Hide resolved
max_warns = Config.post_get("max_n_warns")
if n_warns < max_warns:
user.warn()
text += f"ha ricevuto {n_warns + 1} warn(s)"
else:
text += "è stato bannato."
await info.bot.ban_chat_member(Config.post_get("community_group_id"), user.user_id)
user.ban()
await info.bot.send_message(chat_id=info.chat_id, text=text)