const { SlashCommandBuilder, PermissionFlagsBits, EmbedBuilder, MessageFlags } = require('discord.js'); const db = require('../../functions/database/db.js'); const { sendLog } = require('../../utils/helpers'); const { colors, emojis } = require('../../utils/constants'); module.exports = { category: 'moderation', data: new SlashCommandBuilder() .setName('revokesanction') .setDescription('Révoquer une sanction (Haut gradés, Co-Fonda/Fonda uniquement).') .addUserOption(option => option.setName('user') .setDescription('L\'utilisateur concerné') .setRequired(true)) .addIntegerOption(option => option.setName('sanction_id') .setDescription('L\'ID DB de la sanction à révoquer (visible dans /casier, ex: ID: 123)') .setRequired(true)) .addStringOption(option => option.setName('reason') .setDescription('Raison de la révocation') .setRequired(false)) .setDefaultMemberPermissions(PermissionFlagsBits.Administrator), async execute(interaction) { const target = interaction.options.getUser('user'); const sanctionId = interaction.options.getInteger('sanction_id'); const reason = interaction.options.getString('reason') || 'Aucune raison fournie'; await interaction.deferReply({ flags: MessageFlags.Ephemeral }); if (!interaction.member.permissions.has(PermissionFlagsBits.Administrator) && interaction.user.id !== interaction.guild.ownerId) { return interaction.editReply({ content: '❌ Seuls les administrateurs et le propriétaire du serveur peuvent révoquer une sanction.' }); } try { const [sanctionRows] = await db.query( 'SELECT * FROM logs WHERE id = ? AND userId = ? AND guildId = ?', [sanctionId, target.id, interaction.guild.id] ); if (!sanctionRows.length) { return interaction.editReply({ content: `❌ Aucune sanction trouvée avec l'ID ${sanctionId} pour ${target.tag}.\n\n💡 **Astuce:** Utilise l'ID DB visible dans \`/casier\` (ex: \`#1 (ID: 123)\` → utilise \`123\`).` }); } const sanction = sanctionRows[0]; if (sanction.action === 'Modification de sanction' || sanction.action === 'Révocation de sanction' || sanction.action === 'Modification de mute') { return interaction.editReply({ content: `❌ Cette entrée est une modification, pas une sanction. Utilise l'ID d'une vraie sanction.` }); } await db.query( 'UPDATE logs SET reason = CONCAT(reason, " [RÉVOQUÉE]") WHERE id = ?', [sanction.id] ); await db.query( `INSERT INTO logs (userId, userTag, modId, modTag, action, reason, type, guildId, timestamp) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`, [target.id, target.tag, interaction.user.id, interaction.user.tag, 'Révocation de sanction', `Sanction #${sanction.id} révoquée. Raison: ${reason}`, null, interaction.guild.id, Date.now()] ); const embed = new EmbedBuilder() .setTitle(`${emojis.success} Sanction Révoquée`) .setColor(colors.success) .setThumbnail(target.displayAvatarURL({ dynamic: true })) .addFields( { name: '👤 Utilisateur', value: `<@${target.id}> (${target.tag})`, inline: true }, { name: '👑 Modérateur', value: `<@${interaction.user.id}> (${interaction.user.tag})`, inline: true }, { name: '📝 Raison de révocation', value: reason, inline: false }, { name: '🆔 ID Sanction révoquée', value: `${sanction.id}`, inline: true }, { name: '📅 Date', value: ``, inline: true } ) .setFooter({ text: `ID: ${target.id}` }) .setTimestamp(); await interaction.editReply({ embeds: [embed] }); const logEmbed = new EmbedBuilder() .setTitle(`${emojis.success} Révocation de Sanction`) .setColor(colors.success) .addFields( { name: '👤 Utilisateur', value: `<@${target.id}> (${target.tag})`, inline: true }, { name: '👑 Modérateur', value: `<@${interaction.user.id}> (${interaction.user.tag})`, inline: true }, { name: '📝 Raison de révocation', value: reason, inline: false }, { name: '🆔 ID Sanction révoquée', value: `${sanction.id}`, inline: true }, { name: '📅 Date', value: ``, inline: true } ) .setFooter({ text: `ID: ${target.id} | Serveur: ${interaction.guild.name}` }) .setTimestamp(); await sendLog(interaction.guild, { embeds: [logEmbed] }); } catch (err) { console.error('Erreur revokesanction:', err); await interaction.editReply({ content: `❌ Échec de la révocation de la sanction: ${err.message}` }); } }, };