mirror of
https://github.com/skylanix/MamieHenriette.git
synced 2026-02-06 06:40:35 +01:00
Ajout de la gestion des rôles de modération dans le panneau d'administration. Mise à jour de la logique de configuration pour permettre la sélection de plusieurs rôles. Amélioration de l'interface utilisateur pour la gestion des rôles et des commandes de modération. Ajout de la mise à jour automatique du cache anti-cheat et de nouvelles fonctionnalités pour récupérer et stocker les données anti-cheat.
This commit is contained in:
@@ -8,7 +8,17 @@ from database.helpers import ConfigurationHelper
|
||||
from database.models import Configuration, Humeur, Commande
|
||||
from discord import Message, TextChannel, Member
|
||||
from discordbot.humblebundle import checkHumbleBundleAndNotify
|
||||
from discordbot.command import handle_warning_command, handle_remove_warning_command, handle_list_warnings_command, handle_ban_command, handle_kick_command, handle_unban_command
|
||||
from discordbot.moderation import (
|
||||
handle_warning_command,
|
||||
handle_remove_warning_command,
|
||||
handle_list_warnings_command,
|
||||
handle_ban_command,
|
||||
handle_kick_command,
|
||||
handle_unban_command,
|
||||
handle_inspect_command,
|
||||
handle_ban_list_command,
|
||||
handle_staff_help_command
|
||||
)
|
||||
from discordbot.welcome import sendWelcomeMessage, sendLeaveMessage, updateInviteCache
|
||||
from protondb import searhProtonDb
|
||||
|
||||
@@ -47,11 +57,26 @@ class DiscordBot(discord.Client):
|
||||
if isinstance(channel, TextChannel):
|
||||
channels.append(channel)
|
||||
return channels
|
||||
|
||||
def getAllRoles(self):
|
||||
guilds_roles = []
|
||||
for guild in self.guilds:
|
||||
roles = []
|
||||
for role in guild.roles:
|
||||
if role.name != "@everyone":
|
||||
roles.append(role)
|
||||
if roles:
|
||||
guilds_roles.append({
|
||||
'guild_name': guild.name,
|
||||
'guild_id': guild.id,
|
||||
'roles': roles
|
||||
})
|
||||
return guilds_roles
|
||||
|
||||
|
||||
def begin(self) :
|
||||
token = Configuration.query.filter_by(key='discord_token').first()
|
||||
if token :
|
||||
if token and token.value and token.value.strip():
|
||||
self.run(token.value)
|
||||
else :
|
||||
logging.error('Aucun token Discord configuré. Le bot ne peut pas être démarré')
|
||||
@@ -92,12 +117,23 @@ async def on_message(message: Message):
|
||||
if command_name == '!unban':
|
||||
await handle_unban_command(message, bot)
|
||||
return
|
||||
if command_name == '!banlist':
|
||||
await handle_ban_list_command(message, bot)
|
||||
return
|
||||
|
||||
if ConfigurationHelper().getValue('moderation_kick_enable'):
|
||||
if command_name == '!kick':
|
||||
await handle_kick_command(message, bot)
|
||||
return
|
||||
|
||||
if ConfigurationHelper().getValue('moderation_enable'):
|
||||
if command_name == '!inspect':
|
||||
await handle_inspect_command(message, bot)
|
||||
return
|
||||
if command_name in ['!aide', '!help']:
|
||||
await handle_staff_help_command(message, bot)
|
||||
return
|
||||
|
||||
commande = Commande.query.filter_by(discord_enable=True, trigger=command_name).first()
|
||||
if commande:
|
||||
try:
|
||||
@@ -106,27 +142,80 @@ async def on_message(message: Message):
|
||||
except Exception as e:
|
||||
logging.error(f'Échec de l\'exécution de la commande Discord : {e}')
|
||||
|
||||
if(ConfigurationHelper().getValue('proton_db_enable_enable') and message.content.find('!protondb')==0) :
|
||||
# Commande !protondb ou !pdb avec embed
|
||||
if (ConfigurationHelper().getValue('proton_db_enable_enable') and (message.content.startswith('!protondb') or message.content.startswith('!pdb'))):
|
||||
if (message.content.find('<@')>0) :
|
||||
mention = message.content[message.content.find('<@'):]
|
||||
else :
|
||||
mention = message.author.mention
|
||||
name = message.content.replace('!protondb', '').replace(f'{mention}', '').strip();
|
||||
# Nettoyer le nom en enlevant la commande (!protondb ou !pdb)
|
||||
name = message.content
|
||||
if name.startswith('!protondb'):
|
||||
name = name.replace('!protondb', '', 1)
|
||||
elif name.startswith('!pdb'):
|
||||
name = name.replace('!pdb', '', 1)
|
||||
name = name.replace(f'{mention}', '').strip();
|
||||
games = searhProtonDb(name)
|
||||
if (len(games)==0) :
|
||||
msg = f'{mention} Je n\'ai pas trouvé de jeux correspondant à **{name}**. Es-tu sûr que le jeu est disponible sur Steam ?'
|
||||
else :
|
||||
msg = f'{mention} J\'ai trouvé {len(games)} jeux :\n'
|
||||
ite = iter(games)
|
||||
while (game := next(ite, None)) is not None and len(msg) < 1850 :
|
||||
msg += f'- [{game.get('name')}](https://www.protondb.com/app/{game.get('id')}) classé **{game.get('tier')}**\n'
|
||||
rest = sum(1 for _ in ite)
|
||||
if (rest > 0):
|
||||
msg += f'- et encore {rest} autres jeux'
|
||||
try:
|
||||
await message.channel.send(msg, suppress_embeds=True)
|
||||
except Exception as e:
|
||||
logging.error(f"Échec de l'envoi du message ProtonDB : {e}")
|
||||
return
|
||||
|
||||
# Construire un bel embed
|
||||
embed = discord.Embed(
|
||||
title=f"🔎 Résultats ProtonDB pour {name}",
|
||||
color=discord.Color.blurple()
|
||||
)
|
||||
embed.set_footer(text=f"Demandé par {message.author.name}")
|
||||
|
||||
max_fields = 10
|
||||
count = 0
|
||||
for game in games:
|
||||
if count >= max_fields:
|
||||
break
|
||||
g_name = str(game.get('name'))
|
||||
g_id = str(game.get('id'))
|
||||
tier = str(game.get('tier') or 'N/A')
|
||||
# Anti-cheat info si disponible
|
||||
ac_status = game.get('anticheat_status')
|
||||
ac_emoji = ''
|
||||
ac_text = ''
|
||||
if ac_status:
|
||||
status_lower = str(ac_status).lower()
|
||||
if status_lower == 'supported':
|
||||
ac_emoji, ac_text = '✅', 'Supporté'
|
||||
elif status_lower == 'running':
|
||||
ac_emoji, ac_text = '⚠️', 'Fonctionne'
|
||||
elif status_lower == 'broken':
|
||||
ac_emoji, ac_text = '❌', 'Cassé'
|
||||
elif status_lower == 'denied':
|
||||
ac_emoji, ac_text = '🚫', 'Refusé'
|
||||
elif status_lower == 'planned':
|
||||
ac_emoji, ac_text = '📅', 'Planifié'
|
||||
else:
|
||||
ac_emoji, ac_text = '❔', str(ac_status)
|
||||
acs = game.get('anticheats') or []
|
||||
ac_list = ', '.join([str(ac) for ac in acs if ac])
|
||||
ac_line = f" | Anti-cheat: {ac_emoji} **{ac_text}**"
|
||||
if ac_list:
|
||||
ac_line += f" ({ac_list})"
|
||||
else:
|
||||
ac_line = ''
|
||||
value = f"Tier: **{tier}**{ac_line}\nLien: https://www.protondb.com/app/{g_id}"
|
||||
embed.add_field(name=g_name, value=value[:1024], inline=False)
|
||||
count += 1
|
||||
|
||||
rest = max(0, len(games) - count)
|
||||
if rest > 0:
|
||||
embed.add_field(name="…", value=f"et encore {rest} autres jeux", inline=False)
|
||||
|
||||
try :
|
||||
await message.channel.send(msg, suppress_embeds=True)
|
||||
await message.channel.send(content=mention, embed=embed)
|
||||
except Exception as e:
|
||||
logging.error(f'Échec de l\'envoi du message ProtonDB : {e}')
|
||||
logging.error(f"Échec de l'envoi de l'embed ProtonDB : {e}")
|
||||
|
||||
@bot.event
|
||||
async def on_member_join(member: Member):
|
||||
|
||||
@@ -1,533 +0,0 @@
|
||||
import discord
|
||||
import asyncio
|
||||
from datetime import datetime
|
||||
from database import db
|
||||
from database.helpers import ConfigurationHelper
|
||||
from database.models import ModerationEvent
|
||||
from discord import Message
|
||||
|
||||
def get_staff_role_id():
|
||||
staff_role = ConfigurationHelper().getValue('moderation_staff_role_id')
|
||||
return int(staff_role) if staff_role else 581990740431732738
|
||||
|
||||
def get_embed_delete_delay():
|
||||
delay = ConfigurationHelper().getValue('moderation_embed_delete_delay')
|
||||
return int(delay) if delay else 0
|
||||
|
||||
async def delete_after_delay(message):
|
||||
delay = get_embed_delete_delay()
|
||||
if delay > 0:
|
||||
await asyncio.sleep(delay)
|
||||
try:
|
||||
await message.delete()
|
||||
except:
|
||||
pass
|
||||
|
||||
async def handle_warning_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=2)
|
||||
|
||||
if len(parts) < 2:
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!averto @utilisateur [raison]` ou `!averto <id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!averto @User Spam dans le chat`\n• `!warn 123456789012345678 Comportement inapproprié`\n• `!av @User`", inline=False)
|
||||
embed.add_field(name="Aliases", value="`!averto`, `!av`, `!avertissement`, `!warn`", inline=False)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
target_user = None
|
||||
if message.mentions:
|
||||
target_user = message.mentions[0]
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
else:
|
||||
try:
|
||||
user_id = int(parts[1])
|
||||
target_user = await bot.fetch_user(user_id)
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
except (ValueError, discord.NotFound):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Utilisateur introuvable. Vérifiez la mention ou l'ID Discord.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
if not target_user:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Impossible de trouver l'utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
event = ModerationEvent(
|
||||
type='warning',
|
||||
username=target_user.name,
|
||||
discord_id=str(target_user.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(event)
|
||||
db.session.commit()
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_user.name}** (`@{target_user.name}`) a été **averti**.",
|
||||
color=discord.Color.orange()
|
||||
)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await message.delete()
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
async def handle_remove_warning_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=1)
|
||||
|
||||
if len(parts) < 2:
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!delaverto <id>`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!delaverto 5`\n• `!removewarn 12`", inline=False)
|
||||
embed.add_field(name="Aliases", value="`!delaverto`, `!removewarn`, `!delwarn`", inline=False)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
try:
|
||||
event_id = int(parts[1])
|
||||
except ValueError:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="L'ID doit être un nombre entier.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
event = ModerationEvent.query.filter_by(id=event_id).first()
|
||||
|
||||
if not event:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description=f"Aucun événement de modération trouvé avec l'ID `{event_id}`.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
username = event.username
|
||||
event_type = event.type
|
||||
|
||||
db.session.delete(event)
|
||||
db.session.commit()
|
||||
|
||||
embed = discord.Embed(
|
||||
title="✅ Événement supprimé",
|
||||
description=f"L'événement de type **{event_type}** pour **{username}** (ID: {event_id}) a été supprimé.",
|
||||
color=discord.Color.green(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
embed.add_field(name="🛡️ Modérateur", value=f"{message.author.name}\n`{message.author.id}`", inline=True)
|
||||
embed.set_footer(text="Mamie Henriette")
|
||||
|
||||
await message.channel.send(embed=embed)
|
||||
await message.delete()
|
||||
|
||||
async def handle_list_warnings_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=1)
|
||||
user_filter = None
|
||||
|
||||
if len(parts) > 1 and message.mentions:
|
||||
user_filter = str(message.mentions[0].id)
|
||||
|
||||
if user_filter:
|
||||
events = ModerationEvent.query.filter_by(discord_id=user_filter).order_by(ModerationEvent.created_at.desc()).all()
|
||||
else:
|
||||
events = ModerationEvent.query.order_by(ModerationEvent.created_at.desc()).all()
|
||||
|
||||
if not events:
|
||||
embed = discord.Embed(
|
||||
title="📋 Liste des événements",
|
||||
description="Aucun événement de modération trouvé.",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
page = 0
|
||||
per_page = 5
|
||||
max_page = (len(events) - 1) // per_page
|
||||
|
||||
def create_embed(page_num):
|
||||
start = page_num * per_page
|
||||
end = start + per_page
|
||||
page_events = events[start:end]
|
||||
|
||||
embed = discord.Embed(
|
||||
title="📋 Liste des événements de modération",
|
||||
description=f"Total : {len(events)} événement(s)",
|
||||
color=discord.Color.blue(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
|
||||
for event in page_events:
|
||||
date_str = event.created_at.strftime('%d/%m/%Y %H:%M') if event.created_at else 'N/A'
|
||||
embed.add_field(
|
||||
name=f"ID {event.id} - {event.type.upper()} - {event.username}",
|
||||
value=f"**Discord ID:** `{event.discord_id}`\n**Date:** {date_str}\n**Raison:** {event.reason}\n**Staff:** {event.staff_name}",
|
||||
inline=False
|
||||
)
|
||||
|
||||
embed.set_footer(text=f"Page {page_num + 1}/{max_page + 1}")
|
||||
return embed
|
||||
|
||||
msg = await message.channel.send(embed=create_embed(page))
|
||||
|
||||
if max_page > 0:
|
||||
await msg.add_reaction('⬅️')
|
||||
await msg.add_reaction('➡️')
|
||||
await msg.add_reaction('❌')
|
||||
|
||||
def check(reaction, user):
|
||||
return user == message.author and str(reaction.emoji) in ['⬅️', '➡️', '❌'] and reaction.message.id == msg.id
|
||||
|
||||
while True:
|
||||
try:
|
||||
reaction, user = await bot.wait_for('reaction_add', timeout=60.0, check=check)
|
||||
|
||||
if str(reaction.emoji) == '❌':
|
||||
await msg.delete()
|
||||
break
|
||||
elif str(reaction.emoji) == '➡️' and page < max_page:
|
||||
page += 1
|
||||
await msg.edit(embed=create_embed(page))
|
||||
elif str(reaction.emoji) == '⬅️' and page > 0:
|
||||
page -= 1
|
||||
await msg.edit(embed=create_embed(page))
|
||||
|
||||
await msg.remove_reaction(reaction, user)
|
||||
except:
|
||||
break
|
||||
|
||||
if msg:
|
||||
try:
|
||||
await msg.clear_reactions()
|
||||
except:
|
||||
pass
|
||||
|
||||
await message.delete()
|
||||
|
||||
async def handle_ban_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=2)
|
||||
|
||||
if len(parts) < 2:
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!ban @utilisateur [raison]` ou `!ban <id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!ban @User Spam répété`\n• `!ban 123456789012345678 Comportement toxique`", inline=False)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
target_user = None
|
||||
if message.mentions:
|
||||
target_user = message.mentions[0]
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
else:
|
||||
try:
|
||||
user_id = int(parts[1])
|
||||
target_user = await bot.fetch_user(user_id)
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
except (ValueError, discord.NotFound):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Utilisateur introuvable. Vérifiez la mention ou l'ID Discord.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
if not target_user:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Impossible de trouver l'utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
member = message.guild.get_member(target_user.id)
|
||||
joined_days = None
|
||||
if member and member.joined_at:
|
||||
delta = datetime.utcnow() - member.joined_at.replace(tzinfo=None)
|
||||
joined_days = delta.days
|
||||
|
||||
try:
|
||||
await message.guild.ban(target_user, reason=reason)
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour bannir cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
event = ModerationEvent(
|
||||
type='ban',
|
||||
username=target_user.name,
|
||||
discord_id=str(target_user.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(event)
|
||||
db.session.commit()
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_user.name}** (`@{target_user.name}`) a été **banni**.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
embed.add_field(name="ID Discord", value=f"`{target_user.id}`", inline=False)
|
||||
if joined_days is not None:
|
||||
embed.add_field(name="Membre depuis", value=f"{joined_days} jour{'s' if joined_days > 1 else ''}", inline=False)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await message.delete()
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
async def handle_kick_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=2)
|
||||
|
||||
if len(parts) < 2 or not message.mentions:
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!kick @utilisateur [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!kick @User Spam dans le chat`\n• `!kick @User Comportement inapproprié`", inline=False)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
target_member = message.mentions[0]
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
|
||||
member_obj = message.guild.get_member(target_member.id)
|
||||
if not member_obj:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="L'utilisateur n'est pas membre du serveur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
joined_days = None
|
||||
if member_obj.joined_at:
|
||||
delta = datetime.utcnow() - member_obj.joined_at.replace(tzinfo=None)
|
||||
joined_days = delta.days
|
||||
|
||||
try:
|
||||
await message.guild.kick(member_obj, reason=reason)
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour expulser cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
event = ModerationEvent(
|
||||
type='kick',
|
||||
username=target_member.name,
|
||||
discord_id=str(target_member.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(event)
|
||||
db.session.commit()
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_member.name}** (`@{target_member.name}`) a été **expulsé**.",
|
||||
color=discord.Color.orange()
|
||||
)
|
||||
if joined_days is not None:
|
||||
embed.add_field(name="Membre depuis", value=f"{joined_days} jour{'s' if joined_days > 1 else ''}", inline=False)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await message.delete()
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
async def handle_unban_command(message: Message, bot):
|
||||
if not any(role.id == get_staff_role_id() for role in message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=2)
|
||||
|
||||
if len(parts) < 2:
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!unban <discord_id>` ou `!unban #<sanction_id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!unban 123456789012345678`\n• `!unban #5 Appel accepté`", inline=False)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
target_user = None
|
||||
discord_id = None
|
||||
|
||||
if parts[1].startswith('#'):
|
||||
try:
|
||||
sanction_id = int(parts[1][1:])
|
||||
event = ModerationEvent.query.filter_by(id=sanction_id, type='ban').first()
|
||||
if not event:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description=f"Aucune sanction de ban trouvée avec l'ID #{sanction_id}.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
discord_id = event.discord_id
|
||||
try:
|
||||
target_user = await bot.fetch_user(int(discord_id))
|
||||
except discord.NotFound:
|
||||
pass
|
||||
except ValueError:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="ID de sanction invalide.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
else:
|
||||
try:
|
||||
discord_id = parts[1]
|
||||
target_user = await bot.fetch_user(int(discord_id))
|
||||
except (ValueError, discord.NotFound):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="ID Discord invalide ou utilisateur introuvable.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
try:
|
||||
await message.guild.unban(discord.Object(id=int(discord_id)), reason=reason)
|
||||
except discord.NotFound:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Cet utilisateur n'est pas banni.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour débannir cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
username = target_user.name if target_user else f"ID: {discord_id}"
|
||||
|
||||
event = ModerationEvent(
|
||||
type='unban',
|
||||
username=username,
|
||||
discord_id=discord_id,
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(event)
|
||||
db.session.commit()
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{username}** (`@{username}`) a été **débanni**.",
|
||||
color=discord.Color.green()
|
||||
)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await message.delete()
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
925
discordbot/moderation.py
Normal file
925
discordbot/moderation.py
Normal file
@@ -0,0 +1,925 @@
|
||||
import asyncio
|
||||
import logging
|
||||
import time
|
||||
import discord
|
||||
from datetime import datetime, timezone
|
||||
from database import db
|
||||
from database.helpers import ConfigurationHelper
|
||||
from database.models import ModerationEvent
|
||||
from discord import Message
|
||||
|
||||
def get_staff_role_ids():
|
||||
staff_roles = ConfigurationHelper().getValue('moderation_staff_role_ids')
|
||||
if staff_roles:
|
||||
return [int(role_id.strip()) for role_id in staff_roles.split(',') if role_id.strip()]
|
||||
staff_role_old = ConfigurationHelper().getValue('moderation_staff_role_id')
|
||||
if staff_role_old:
|
||||
return [int(staff_role_old)]
|
||||
return []
|
||||
|
||||
def has_staff_role(user_roles):
|
||||
staff_role_ids = get_staff_role_ids()
|
||||
if not staff_role_ids:
|
||||
return False
|
||||
return any(role.id in staff_role_ids for role in user_roles)
|
||||
|
||||
def get_embed_delete_delay():
|
||||
delay = ConfigurationHelper().getValue('moderation_embed_delete_delay')
|
||||
return int(delay) if delay else 0
|
||||
|
||||
async def delete_after_delay(message):
|
||||
delay = get_embed_delete_delay()
|
||||
if delay > 0:
|
||||
await asyncio.sleep(delay)
|
||||
try:
|
||||
await message.delete()
|
||||
except:
|
||||
pass
|
||||
|
||||
async def safe_delete_message(message: Message):
|
||||
try:
|
||||
await message.delete()
|
||||
except:
|
||||
pass
|
||||
|
||||
async def send_access_denied(channel):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def send_user_not_found(channel):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Utilisateur introuvable. Vérifiez la mention ou l'ID Discord.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def parse_target_user_and_reason(message, bot, parts: list):
|
||||
if message.mentions:
|
||||
target_user = message.mentions[0]
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
return target_user, reason
|
||||
|
||||
try:
|
||||
user_id = int(parts[1])
|
||||
target_user = await bot.fetch_user(user_id)
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
return target_user, reason
|
||||
except (ValueError, discord.NotFound):
|
||||
return None, None
|
||||
|
||||
async def send_warning_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!averto @utilisateur [raison]` ou `!averto <id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!averto @User Spam dans le chat`\n• `!warn 123456789012345678 Comportement inapproprié`\n• `!av @User`", inline=False)
|
||||
embed.add_field(name="Aliases", value="`!averto`, `!av`, `!avertissement`, `!warn`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
def create_warning_event(target_user, reason: str, staff_member):
|
||||
event = ModerationEvent(
|
||||
type='warning',
|
||||
username=target_user.name,
|
||||
discord_id=str(target_user.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(staff_member.id),
|
||||
staff_name=staff_member.name
|
||||
)
|
||||
db.session.add(event)
|
||||
_commit_with_retry()
|
||||
|
||||
def _commit_with_retry(max_retries: int = 5, base_delay: float = 0.1):
|
||||
attempt = 0
|
||||
while True:
|
||||
try:
|
||||
db.session.commit()
|
||||
return
|
||||
except Exception as e:
|
||||
msg = str(e)
|
||||
if 'database is locked' in msg.lower() and attempt < max_retries:
|
||||
db.session.rollback()
|
||||
delay = base_delay * (2 ** attempt)
|
||||
time.sleep(delay)
|
||||
attempt += 1
|
||||
continue
|
||||
db.session.rollback()
|
||||
raise
|
||||
|
||||
async def send_warning_confirmation(channel, target_user, reason: str, original_message: Message):
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_user.name}** (`@{target_user.name}`) a été **averti**.",
|
||||
color=discord.Color.orange()
|
||||
)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await channel.send(embed=embed)
|
||||
await safe_delete_message(original_message)
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
async def handle_warning_command(message: Message, bot):
|
||||
parts = message.content.split(maxsplit=2)
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
elif len(parts) < 2:
|
||||
await send_warning_usage(message.channel)
|
||||
else:
|
||||
target_user, reason = await parse_target_user_and_reason(message, bot, parts)
|
||||
if not target_user:
|
||||
await send_user_not_found(message.channel)
|
||||
else:
|
||||
await _process_warning_success(message, target_user, reason)
|
||||
|
||||
async def _process_warning_success(message: Message, target_user, reason: str):
|
||||
create_warning_event(target_user, reason, message.author)
|
||||
await send_warning_confirmation(message.channel, target_user, reason, message)
|
||||
|
||||
async def send_remove_warning_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!delaverto <id>`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!delaverto 5`\n• `!removewarn 12`", inline=False)
|
||||
embed.add_field(name="Aliases", value="`!delaverto`, `!removewarn`, `!delwarn`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def send_invalid_event_id(channel):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="L'ID doit être un nombre entier.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def send_event_not_found(channel, event_id: int):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description=f"Aucun événement de modération trouvé avec l'ID `{event_id}`.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
def delete_moderation_event(event: ModerationEvent):
|
||||
db.session.delete(event)
|
||||
db.session.commit()
|
||||
|
||||
async def send_event_deleted_confirmation(channel, event: ModerationEvent, moderator, original_message: Message):
|
||||
embed = discord.Embed(
|
||||
title="✅ Événement supprimé",
|
||||
description=f"L'événement de type **{event.type}** pour **{event.username}** (ID: {event.id}) a été supprimé.",
|
||||
color=discord.Color.green(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
embed.add_field(name="🛡️ Modérateur", value=f"{moderator.name}\n`{moderator.id}`", inline=True)
|
||||
embed.set_footer(text="Mamie Henriette")
|
||||
|
||||
await channel.send(embed=embed)
|
||||
await safe_delete_message(original_message)
|
||||
|
||||
async def handle_remove_warning_command(message: Message, bot):
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=1)
|
||||
|
||||
if len(parts) < 2:
|
||||
await send_remove_warning_usage(message.channel)
|
||||
return
|
||||
|
||||
try:
|
||||
event_id = int(parts[1])
|
||||
except ValueError:
|
||||
await send_invalid_event_id(message.channel)
|
||||
return
|
||||
|
||||
event = ModerationEvent.query.filter_by(id=event_id).first()
|
||||
|
||||
if not event:
|
||||
await send_event_not_found(message.channel, event_id)
|
||||
return
|
||||
|
||||
delete_moderation_event(event)
|
||||
await send_event_deleted_confirmation(message.channel, event, message.author, message)
|
||||
|
||||
def get_moderation_events(user_filter: str = None):
|
||||
if user_filter:
|
||||
return ModerationEvent.query.filter_by(discord_id=user_filter).order_by(ModerationEvent.created_at.desc()).all()
|
||||
return ModerationEvent.query.order_by(ModerationEvent.created_at.desc()).all()
|
||||
|
||||
async def send_no_events_found(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Liste des événements",
|
||||
description="Aucun événement de modération trouvé.",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
def create_events_list_embed(events: list, page_num: int, per_page: int):
|
||||
start = page_num * per_page
|
||||
end = start + per_page
|
||||
page_events = events[start:end]
|
||||
max_page = (len(events) - 1) // per_page
|
||||
|
||||
embed = discord.Embed(
|
||||
title="📋 Liste des événements de modération",
|
||||
description=f"Total : {len(events)} événement(s)",
|
||||
color=discord.Color.blue(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
|
||||
for event in page_events:
|
||||
date_str = event.created_at.strftime('%d/%m/%Y %H:%M') if event.created_at else 'N/A'
|
||||
embed.add_field(
|
||||
name=f"ID {event.id} - {event.type.upper()} - {event.username}",
|
||||
value=f"**Discord ID:** `{event.discord_id}`\n**Date:** {date_str}\n**Raison:** {event.reason}\n**Staff:** {event.staff_name}",
|
||||
inline=False
|
||||
)
|
||||
|
||||
embed.set_footer(text=f"Page {page_num + 1}/{max_page + 1}")
|
||||
return embed
|
||||
|
||||
async def add_pagination_reactions(msg, max_page: int):
|
||||
if max_page > 0:
|
||||
await msg.add_reaction('⬅️')
|
||||
await msg.add_reaction('➡️')
|
||||
await msg.add_reaction('❌')
|
||||
|
||||
async def handle_pagination_loop(msg, bot, message_author, events: list, per_page: int):
|
||||
page = 0
|
||||
max_page = (len(events) - 1) // per_page
|
||||
|
||||
def check(reaction, user):
|
||||
return user == message_author and str(reaction.emoji) in ['⬅️', '➡️', '❌'] and reaction.message.id == msg.id
|
||||
|
||||
while True:
|
||||
try:
|
||||
reaction, user = await bot.wait_for('reaction_add', timeout=60.0, check=check)
|
||||
|
||||
if str(reaction.emoji) == '❌':
|
||||
await msg.delete()
|
||||
break
|
||||
elif str(reaction.emoji) == '➡️' and page < max_page:
|
||||
page += 1
|
||||
await msg.edit(embed=create_events_list_embed(events, page, per_page))
|
||||
elif str(reaction.emoji) == '⬅️' and page > 0:
|
||||
page -= 1
|
||||
await msg.edit(embed=create_events_list_embed(events, page, per_page))
|
||||
|
||||
await msg.remove_reaction(reaction, user)
|
||||
except:
|
||||
break
|
||||
|
||||
try:
|
||||
await msg.clear_reactions()
|
||||
except:
|
||||
pass
|
||||
|
||||
async def handle_list_warnings_command(message: Message, bot):
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=1)
|
||||
user_filter = str(message.mentions[0].id) if len(parts) > 1 and message.mentions else None
|
||||
|
||||
events = get_moderation_events(user_filter)
|
||||
|
||||
if not events:
|
||||
await send_no_events_found(message.channel)
|
||||
return
|
||||
|
||||
per_page = 5
|
||||
max_page = (len(events) - 1) // per_page
|
||||
|
||||
msg = await message.channel.send(embed=create_events_list_embed(events, 0, per_page))
|
||||
await add_pagination_reactions(msg, max_page)
|
||||
await handle_pagination_loop(msg, bot, message.author, events, per_page)
|
||||
await safe_delete_message(message)
|
||||
|
||||
async def handle_ban_command(message: Message, bot):
|
||||
parts = message.content.split(maxsplit=2)
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
elif len(parts) < 2:
|
||||
await _send_ban_usage(message.channel)
|
||||
else:
|
||||
target_user, reason = await _parse_ban_target_and_reason(message, bot, parts)
|
||||
if not target_user:
|
||||
await _send_user_not_found_for_ban(message.channel)
|
||||
else:
|
||||
await _process_ban_success(message, target_user, reason)
|
||||
|
||||
async def _send_ban_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!ban @utilisateur [raison]` ou `!ban <id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!ban @User Spam répété`\n• `!ban 123456789012345678 Comportement toxique`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def _parse_ban_target_and_reason(message: Message, bot, parts: list):
|
||||
if message.mentions:
|
||||
return message.mentions[0], (parts[2] if len(parts) > 2 else "Sans raison")
|
||||
try:
|
||||
user_id = int(parts[1])
|
||||
user = await bot.fetch_user(user_id)
|
||||
return user, (parts[2] if len(parts) > 2 else "Sans raison")
|
||||
except (ValueError, discord.NotFound):
|
||||
return None, None
|
||||
|
||||
async def _send_user_not_found_for_ban(channel):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Utilisateur introuvable. Vérifiez la mention ou l'ID Discord.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
def _create_ban_event(target_user, reason: str, staff_member):
|
||||
event = ModerationEvent(
|
||||
type='ban',
|
||||
username=target_user.name,
|
||||
discord_id=str(target_user.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(staff_member.id),
|
||||
staff_name=staff_member.name
|
||||
)
|
||||
db.session.add(event)
|
||||
_commit_with_retry()
|
||||
return event
|
||||
|
||||
async def _process_ban_success(message: Message, target_user, reason: str):
|
||||
member = message.guild.get_member(target_user.id)
|
||||
joined_days = None
|
||||
if member and member.joined_at:
|
||||
delta = datetime.utcnow() - member.joined_at.replace(tzinfo=None)
|
||||
joined_days = delta.days
|
||||
try:
|
||||
await message.guild.ban(target_user, reason=reason)
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour bannir cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
event = _create_ban_event(target_user, reason, message.author)
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_user.name}** (`@{target_user.name}`) a été **banni**.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
embed.add_field(name="ID Discord", value=f"`{target_user.id}`", inline=False)
|
||||
if joined_days is not None:
|
||||
embed.add_field(name="Membre depuis", value=format_days_to_age(joined_days), inline=False)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await safe_delete_message(message)
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
async def handle_unban_command(message: Message, bot):
|
||||
parts = message.content.split(maxsplit=2)
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
elif len(parts) < 2:
|
||||
await _send_unban_usage(message.channel)
|
||||
else:
|
||||
target_user, discord_id, reason = await _parse_unban_target_and_reason(message, bot, parts)
|
||||
if not discord_id:
|
||||
await _send_unban_invalid_id(message.channel)
|
||||
else:
|
||||
await _process_unban_success(message, bot, target_user, discord_id, reason)
|
||||
|
||||
async def _send_unban_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!unban <discord_id>` ou `!unban #<sanction_id> [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!unban 123456789012345678`\n• `!unban #5 Appel accepté`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def _parse_unban_target_and_reason(message: Message, bot, parts: list):
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
target_user = None
|
||||
discord_id = None
|
||||
if parts[1].startswith('#'):
|
||||
try:
|
||||
sanction_id = int(parts[1][1:])
|
||||
evt = ModerationEvent.query.filter_by(id=sanction_id, type='ban').first()
|
||||
if not evt:
|
||||
return None, None, reason
|
||||
discord_id = evt.discord_id
|
||||
try:
|
||||
target_user = await bot.fetch_user(int(discord_id))
|
||||
except discord.NotFound:
|
||||
pass
|
||||
except ValueError:
|
||||
return None, None, reason
|
||||
else:
|
||||
try:
|
||||
discord_id = parts[1]
|
||||
target_user = await bot.fetch_user(int(discord_id))
|
||||
except (ValueError, discord.NotFound):
|
||||
return None, None, reason
|
||||
return target_user, discord_id, reason
|
||||
|
||||
async def _send_unban_invalid_id(channel):
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="ID Discord invalide ou utilisateur introuvable.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def _process_unban_success(message: Message, bot, target_user, discord_id: str, reason: str):
|
||||
try:
|
||||
await message.guild.unban(discord.Object(id=int(discord_id)), reason=reason)
|
||||
except discord.NotFound:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Cet utilisateur n'est pas banni.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour débannir cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
username = target_user.name if target_user else f"ID: {discord_id}"
|
||||
create = ModerationEvent(
|
||||
type='unban',
|
||||
username=username,
|
||||
discord_id=discord_id,
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(create)
|
||||
_commit_with_retry()
|
||||
|
||||
try:
|
||||
asyncio.create_task(_send_unban_invite(message, bot, target_user, discord_id))
|
||||
except:
|
||||
pass
|
||||
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{username}** (`@{username}`) a été **débanni**.",
|
||||
color=discord.Color.green()
|
||||
)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await safe_delete_message(message)
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
async def _send_unban_invite(message: Message, bot, target_user, discord_id: str):
|
||||
try:
|
||||
user_obj = target_user or await bot.fetch_user(int(discord_id))
|
||||
channel = None
|
||||
try:
|
||||
channel_id = ConfigurationHelper().getIntValue('welcome_channel_id')
|
||||
if channel_id:
|
||||
channel = bot.get_channel(channel_id) or message.guild.get_channel(channel_id)
|
||||
except:
|
||||
pass
|
||||
if not channel:
|
||||
me = message.guild.me or message.guild.get_member(bot.user.id)
|
||||
for ch in message.guild.text_channels:
|
||||
try:
|
||||
perms = ch.permissions_for(me) if me else None
|
||||
if not perms or not perms.create_instant_invite:
|
||||
continue
|
||||
channel = ch
|
||||
break
|
||||
except:
|
||||
continue
|
||||
if not channel:
|
||||
channel = message.guild.system_channel or message.channel
|
||||
invite = None
|
||||
try:
|
||||
invite = await channel.create_invite(max_age=86400, max_uses=1, unique=True, reason='Invitation automatique après débannissement')
|
||||
except Exception as e:
|
||||
logging.warning(f"[UNBAN] Échec création d'invitation sur #{channel and channel.name}: {e}")
|
||||
return
|
||||
if user_obj and invite:
|
||||
try:
|
||||
msg = f"Tu as été débanni de {message.guild.name}. Voici une invitation pour revenir : {invite.url}"
|
||||
await user_obj.send(msg)
|
||||
except Exception as e:
|
||||
logging.warning(f"[UNBAN] Impossible d'envoyer un MP à {user_obj} ({user_obj.id}): {e}")
|
||||
try:
|
||||
await message.author.send(f"Impossible d'envoyer un MP à {user_obj} pour l'unban. Voici l'invitation à lui transmettre : {invite.url}")
|
||||
except:
|
||||
pass
|
||||
except:
|
||||
pass
|
||||
|
||||
async def handle_ban_list_command(message: Message, bot):
|
||||
if not has_staff_role(message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Vous n'avez pas les permissions nécessaires pour utiliser cette commande.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
# Récupérer la liste des bannis
|
||||
bans = []
|
||||
try:
|
||||
async for entry in message.guild.bans(limit=None):
|
||||
bans.append(entry)
|
||||
except TypeError:
|
||||
try:
|
||||
bans = await message.guild.bans()
|
||||
except Exception:
|
||||
bans = []
|
||||
except Exception:
|
||||
bans = []
|
||||
|
||||
if not bans:
|
||||
embed = discord.Embed(
|
||||
title="🔨 Utilisateurs bannis",
|
||||
description="Aucun utilisateur banni sur ce serveur.",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
msg = await message.channel.send(embed=embed)
|
||||
asyncio.create_task(delete_after_delay(msg))
|
||||
await safe_delete_message(message)
|
||||
return
|
||||
|
||||
page = 0
|
||||
per_page = 10
|
||||
max_page = (len(bans) - 1) // per_page
|
||||
|
||||
def create_banlist_embed(page_num: int):
|
||||
start = page_num * per_page
|
||||
end = start + per_page
|
||||
page_bans = bans[start:end]
|
||||
embed = discord.Embed(
|
||||
title="🔨 Utilisateurs bannis",
|
||||
description=f"Total : {len(bans)} utilisateur(s) banni(s)",
|
||||
color=discord.Color.red(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
for entry in page_bans:
|
||||
user = entry.user
|
||||
reason = entry.reason or 'Sans raison'
|
||||
embed.add_field(
|
||||
name=f"{user.name} ({user.id})",
|
||||
value=f"Raison: {reason}",
|
||||
inline=False
|
||||
)
|
||||
embed.set_footer(text=f"Page {page_num + 1}/{max_page + 1}")
|
||||
return embed
|
||||
|
||||
msg = await message.channel.send(embed=create_banlist_embed(page))
|
||||
if max_page > 0:
|
||||
await msg.add_reaction('⬅️')
|
||||
await msg.add_reaction('➡️')
|
||||
await msg.add_reaction('❌')
|
||||
|
||||
def check(reaction, user):
|
||||
return user == message.author and str(reaction.emoji) in ['⬅️', '➡️', '❌'] and reaction.message.id == msg.id
|
||||
|
||||
while True:
|
||||
try:
|
||||
reaction, user = await bot.wait_for('reaction_add', timeout=60.0, check=check)
|
||||
if str(reaction.emoji) == '❌':
|
||||
await msg.delete()
|
||||
break
|
||||
elif str(reaction.emoji) == '➡️' and page < max_page:
|
||||
page += 1
|
||||
await msg.edit(embed=create_banlist_embed(page))
|
||||
elif str(reaction.emoji) == '⬅️' and page > 0:
|
||||
page -= 1
|
||||
await msg.edit(embed=create_banlist_embed(page))
|
||||
await msg.remove_reaction(reaction, user)
|
||||
except Exception:
|
||||
break
|
||||
|
||||
try:
|
||||
await msg.clear_reactions()
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
await safe_delete_message(message)
|
||||
|
||||
async def handle_staff_help_command(message: Message, bot):
|
||||
if not has_staff_role(message.author.roles):
|
||||
embed = discord.Embed(
|
||||
title="❌ Accès refusé",
|
||||
description="Cette commande est réservée au staff.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
|
||||
embed = discord.Embed(
|
||||
title="🛠️ Aide staff",
|
||||
description="Commandes de modération disponibles",
|
||||
color=discord.Color.blurple(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
embed.set_footer(text=f"Demandé par {message.author.name}")
|
||||
|
||||
# Avertissements
|
||||
if ConfigurationHelper().getValue('moderation_enable'):
|
||||
value = (
|
||||
"• `!averto @utilisateur [raison]`\n"
|
||||
"• `!delaverto <id>`\n"
|
||||
"• `!warnings` ou `!warnings @utilisateur`\n"
|
||||
"Exemples:\n"
|
||||
"`!averto @User Spam`\n"
|
||||
"`!delaverto 12`\n"
|
||||
"`!warnings @User`"
|
||||
)
|
||||
embed.add_field(name="⚠️ Avertissements", value=value, inline=False)
|
||||
# Inspect
|
||||
embed.add_field(
|
||||
name="🔎 Inspection",
|
||||
value=("• `!inspect @utilisateur` ou `!inspect <id>`\n"
|
||||
"Ex: `!inspect @User`"),
|
||||
inline=False
|
||||
)
|
||||
|
||||
# Bans / Unban
|
||||
if ConfigurationHelper().getValue('moderation_ban_enable'):
|
||||
value = (
|
||||
"• `!ban @utilisateur [raison]`\n"
|
||||
"• `!unban <discord_id>` ou `!unban #<sanction_id> [raison]`\n"
|
||||
"• `!banlist`\n"
|
||||
"Exemples:\n"
|
||||
"`!ban @User Toxicité`\n"
|
||||
"`!unban 123456789012345678 Erreur`\n"
|
||||
"`!unban #5 Appel accepté`"
|
||||
)
|
||||
embed.add_field(name="🔨 Ban / Unban", value=value, inline=False)
|
||||
|
||||
# Kick
|
||||
if ConfigurationHelper().getValue('moderation_kick_enable'):
|
||||
value = (
|
||||
"• `!kick @utilisateur [raison]`\n"
|
||||
"Exemple: `!kick @User Spam`"
|
||||
)
|
||||
embed.add_field(name="👢 Kick", value=value, inline=False)
|
||||
|
||||
try:
|
||||
sent = await message.channel.send(embed=embed)
|
||||
asyncio.create_task(delete_after_delay(sent))
|
||||
except Exception:
|
||||
pass
|
||||
await safe_delete_message(message)
|
||||
|
||||
async def handle_kick_command(message: Message, bot):
|
||||
parts = message.content.split(maxsplit=2)
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
elif len(parts) < 2 or not message.mentions:
|
||||
await _send_kick_usage(message.channel)
|
||||
else:
|
||||
target_member = message.mentions[0]
|
||||
reason = parts[2] if len(parts) > 2 else "Sans raison"
|
||||
await _process_kick_success(message, target_member, reason)
|
||||
|
||||
async def _send_kick_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!kick @utilisateur [raison]`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!kick @User Spam dans le chat`\n• `!kick @User Comportement inapproprié`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def _process_kick_success(message: Message, target_member, reason: str):
|
||||
member_obj = message.guild.get_member(target_member.id)
|
||||
if not member_obj:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="L'utilisateur n'est pas membre du serveur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
joined_days = None
|
||||
if member_obj.joined_at:
|
||||
delta = datetime.utcnow() - member_obj.joined_at.replace(tzinfo=None)
|
||||
joined_days = delta.days
|
||||
try:
|
||||
await message.guild.kick(member_obj, reason=reason)
|
||||
except discord.Forbidden:
|
||||
embed = discord.Embed(
|
||||
title="❌ Erreur",
|
||||
description="Je n'ai pas les permissions nécessaires pour expulser cet utilisateur.",
|
||||
color=discord.Color.red()
|
||||
)
|
||||
await message.channel.send(embed=embed)
|
||||
return
|
||||
create = ModerationEvent(
|
||||
type='kick',
|
||||
username=target_member.name,
|
||||
discord_id=str(target_member.id),
|
||||
created_at=datetime.utcnow(),
|
||||
reason=reason,
|
||||
staff_id=str(message.author.id),
|
||||
staff_name=message.author.name
|
||||
)
|
||||
db.session.add(create)
|
||||
_commit_with_retry()
|
||||
embed = discord.Embed(
|
||||
title="⚠️ Sanction",
|
||||
description=f"L'utilisateur **{target_member.name}** (`@{target_member.name}`) a été **expulsé**.",
|
||||
color=discord.Color.orange()
|
||||
)
|
||||
if joined_days is not None:
|
||||
embed.add_field(name="Membre depuis", value=format_days_to_age(joined_days), inline=False)
|
||||
if reason != "Sans raison":
|
||||
embed.add_field(name="Raison", value=reason, inline=False)
|
||||
sent_message = await message.channel.send(embed=embed)
|
||||
await safe_delete_message(message)
|
||||
asyncio.create_task(delete_after_delay(sent_message))
|
||||
|
||||
def format_days_to_age(days: int) -> str:
|
||||
if days >= 365:
|
||||
years = days // 365
|
||||
remaining_days = days % 365
|
||||
if remaining_days > 0:
|
||||
return f"{years} an{'s' if years > 1 else ''} et {remaining_days} jour{'s' if remaining_days > 1 else ''}"
|
||||
return f"{years} an{'s' if years > 1 else ''}"
|
||||
return f"{days} jour{'s' if days > 1 else ''}"
|
||||
|
||||
async def get_member_join_info(guild, member_id: int):
|
||||
member = guild.get_member(member_id)
|
||||
if not member or not member.joined_at:
|
||||
return None, None
|
||||
|
||||
join_date = member.joined_at
|
||||
days_on_server = (datetime.now(timezone.utc) - join_date).days
|
||||
return join_date, days_on_server
|
||||
|
||||
def get_account_age(user):
|
||||
if not user.created_at:
|
||||
return None
|
||||
account_age = (datetime.now(timezone.utc) - user.created_at).days
|
||||
return account_age
|
||||
|
||||
def get_user_moderation_history(discord_id: str):
|
||||
events = ModerationEvent.query.filter_by(discord_id=discord_id).order_by(ModerationEvent.created_at.desc()).all()
|
||||
|
||||
warnings = [e for e in events if e.type == 'warning']
|
||||
kicks = [e for e in events if e.type == 'kick']
|
||||
bans = [e for e in events if e.type == 'ban']
|
||||
|
||||
return warnings, kicks, bans
|
||||
|
||||
async def send_inspect_usage(channel):
|
||||
embed = discord.Embed(
|
||||
title="📋 Utilisation de la commande",
|
||||
description="**Syntaxe :** `!inspect @utilisateur` ou `!inspect <id>`",
|
||||
color=discord.Color.blue()
|
||||
)
|
||||
embed.add_field(name="Exemples", value="• `!inspect @User`\n• `!inspect 123456789012345678`", inline=False)
|
||||
await channel.send(embed=embed)
|
||||
|
||||
async def parse_target_user(message: Message, bot, parts: list):
|
||||
if message.mentions:
|
||||
return message.mentions[0]
|
||||
|
||||
try:
|
||||
user_id = int(parts[1])
|
||||
return await bot.fetch_user(user_id)
|
||||
except (ValueError, discord.NotFound):
|
||||
return None
|
||||
|
||||
def create_inspect_embed(user, member, join_date, days_on_server, account_age, warnings, kicks, bans, invite_info):
|
||||
embed = discord.Embed(
|
||||
title=f"🔍 Inspection de {user.name}",
|
||||
color=discord.Color.blue(),
|
||||
timestamp=datetime.utcnow()
|
||||
)
|
||||
|
||||
embed.set_thumbnail(url=user.display_avatar.url)
|
||||
embed.add_field(name="👤 Utilisateur", value=f"{user.mention}\n`{user.id}`", inline=True)
|
||||
|
||||
if account_age is not None:
|
||||
embed.add_field(
|
||||
name="📅 Compte créé",
|
||||
value=f"{user.created_at.strftime('%d/%m/%Y')}\n({format_days_to_age(account_age)})",
|
||||
inline=True
|
||||
)
|
||||
|
||||
if member and join_date:
|
||||
embed.add_field(
|
||||
name="📥 Rejoint le serveur",
|
||||
value=f"{join_date.strftime('%d/%m/%Y à %H:%M')}\n({format_days_to_age(days_on_server)})",
|
||||
inline=True
|
||||
)
|
||||
|
||||
if invite_info:
|
||||
embed.add_field(name="🎫 Invitation", value=invite_info, inline=True)
|
||||
else:
|
||||
embed.add_field(name="🎫 Invitation", value="Inconnue", inline=True)
|
||||
|
||||
warning_text = f"⚠️ **{len(warnings)}** avertissement{'s' if len(warnings) > 1 else ''}"
|
||||
kick_text = f"👢 **{len(kicks)}** expulsion{'s' if len(kicks) > 1 else ''}"
|
||||
ban_text = f"🔨 **{len(bans)}** ban{'s' if len(bans) > 1 else ''}"
|
||||
|
||||
mod_history = f"{warning_text}\n{kick_text}\n{ban_text}"
|
||||
|
||||
if warnings or kicks or bans:
|
||||
embed.add_field(name="📋 Historique de modération", value=mod_history, inline=False)
|
||||
|
||||
if warnings:
|
||||
recent_warnings = warnings[:3]
|
||||
warnings_detail = "\n".join([
|
||||
f"• ID {w.id} - {w.created_at.strftime('%d/%m/%Y')} - {w.reason[:50]}{'...' if len(w.reason) > 50 else ''}"
|
||||
for w in recent_warnings
|
||||
])
|
||||
if len(warnings) > 3:
|
||||
warnings_detail += f"\n*... et {len(warnings) - 3} autre(s)*"
|
||||
embed.add_field(name="⚠️ Derniers avertissements", value=warnings_detail, inline=False)
|
||||
else:
|
||||
embed.add_field(name="✅ Historique de modération", value="Aucun incident", inline=False)
|
||||
|
||||
embed.set_footer(text="Mamie Henriette")
|
||||
return embed
|
||||
|
||||
async def get_invite_info_for_user(bot, guild, user_id: int):
|
||||
try:
|
||||
from discordbot.welcome import invite_cache
|
||||
|
||||
audit_logs = [entry async for entry in guild.audit_logs(limit=100, action=discord.AuditLogAction.member_join)]
|
||||
|
||||
for entry in audit_logs:
|
||||
if entry.target and entry.target.id == user_id:
|
||||
if hasattr(entry, 'extra') and entry.extra:
|
||||
return f"Code: `{entry.extra}`"
|
||||
|
||||
return None
|
||||
except:
|
||||
return None
|
||||
|
||||
async def handle_inspect_command(message: Message, bot):
|
||||
if not has_staff_role(message.author.roles):
|
||||
await send_access_denied(message.channel)
|
||||
return
|
||||
|
||||
parts = message.content.split(maxsplit=1)
|
||||
|
||||
if len(parts) < 2:
|
||||
await send_inspect_usage(message.channel)
|
||||
return
|
||||
|
||||
target_user = await parse_target_user(message, bot, parts)
|
||||
|
||||
if not target_user:
|
||||
await send_user_not_found(message.channel)
|
||||
return
|
||||
|
||||
member = message.guild.get_member(target_user.id)
|
||||
join_date, days_on_server = await get_member_join_info(message.guild, target_user.id)
|
||||
account_age = get_account_age(target_user)
|
||||
warnings, kicks, bans = get_user_moderation_history(str(target_user.id))
|
||||
invite_info = await get_invite_info_for_user(bot, message.guild, target_user.id)
|
||||
|
||||
embed = create_inspect_embed(
|
||||
target_user,
|
||||
member,
|
||||
join_date,
|
||||
days_on_server,
|
||||
account_age,
|
||||
warnings,
|
||||
kicks,
|
||||
bans,
|
||||
invite_info
|
||||
)
|
||||
|
||||
await message.channel.send(embed=embed)
|
||||
await safe_delete_message(message)
|
||||
|
||||
@@ -112,17 +112,19 @@ async def sendLeaveMessage(bot: discord.Client, member: Member):
|
||||
reason = 'Départ volontaire'
|
||||
try:
|
||||
async for entry in member.guild.audit_logs(limit=5):
|
||||
if entry.target and entry.target.id == member.id:
|
||||
if entry.action == discord.AuditLogAction.kick:
|
||||
reason = f'Expulsé par {entry.user.mention}'
|
||||
if entry.reason:
|
||||
reason += f' - Raison: {entry.reason}'
|
||||
break
|
||||
elif entry.action == discord.AuditLogAction.ban:
|
||||
reason = f'Banni par {entry.user.mention}'
|
||||
if entry.reason:
|
||||
reason += f' - Raison: {entry.reason}'
|
||||
break
|
||||
if not (entry.target and entry.target.id == member.id):
|
||||
continue
|
||||
|
||||
if entry.action == discord.AuditLogAction.kick:
|
||||
reason = f'Expulsé par {entry.user.mention}'
|
||||
if entry.reason:
|
||||
reason += f' - Raison: {entry.reason}'
|
||||
break
|
||||
elif entry.action == discord.AuditLogAction.ban:
|
||||
reason = f'Banni par {entry.user.mention}'
|
||||
if entry.reason:
|
||||
reason += f' - Raison: {entry.reason}'
|
||||
break
|
||||
except:
|
||||
pass
|
||||
|
||||
|
||||
Reference in New Issue
Block a user