/[sudobot]/branches/6.x/src/commands/moderation/InfractionCommand.ts
ViewVC logotype

Contents of /branches/6.x/src/commands/moderation/InfractionCommand.ts

Parent Directory Parent Directory | Revision Log Revision Log


Revision 577 - (show annotations)
Mon Jul 29 18:52:37 2024 UTC (8 months ago) by rakinar2
File MIME type: application/typescript
File size: 8608 byte(s)
chore: add old version archive branches (2.x to 9.x-dev)
1 /**
2 * This file is part of SudoBot.
3 *
4 * Copyright (C) 2021-2023 OSN Developers.
5 *
6 * SudoBot is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU Affero General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
10 *
11 * SudoBot is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU Affero General Public License for more details.
15 *
16 * You should have received a copy of the GNU Affero General Public License
17 * along with SudoBot. If not, see <https://www.gnu.org/licenses/>.
18 */
19
20 import { InfractionType } from "@prisma/client";
21 import { MessageMentions, PermissionsBitField, SlashCommandBuilder } from "discord.js";
22 import Command, { ArgumentType, BasicCommandContext, CommandMessage, CommandReturn, ValidationRule } from "../../core/Command";
23 import { isSnowflake } from "../../utils/utils";
24
25 const infractionTypes = [
26 {
27 name: "Ban",
28 value: InfractionType.BAN
29 },
30 {
31 name: "Kick",
32 value: InfractionType.KICK
33 },
34 {
35 name: "Mute",
36 value: InfractionType.MUTE
37 },
38 {
39 name: "Warning",
40 value: InfractionType.WARNING
41 },
42 {
43 name: "Unmute",
44 value: InfractionType.UNMUTE
45 },
46 {
47 name: "Unban",
48 value: InfractionType.UNBAN
49 },
50 {
51 name: "Bulk message delete",
52 value: InfractionType.BULK_DELETE_MESSAGE
53 },
54 {
55 name: "Temporary Ban",
56 value: InfractionType.TEMPBAN
57 },
58 {
59 name: "Softban",
60 value: InfractionType.SOFTBAN
61 },
62 {
63 name: "Timeout",
64 value: InfractionType.TIMEOUT
65 },
66 {
67 name: "Timeout remove",
68 value: InfractionType.TIMEOUT_REMOVE
69 }
70 ].map(o => ({ ...o, value: o.value.toLowerCase() }));
71
72 export default class InfractionCommand extends Command {
73 public readonly name = "infraction";
74 public readonly subcommands = ["view", "create", "edit", "delete", "list", "clear"];
75 public readonly subCommandCheck = true;
76 public readonly validationRules: ValidationRule[] = [
77 {
78 types: [ArgumentType.String],
79 name: "subcommand",
80 errors: {
81 required: `Please provide a valid subcommand! The available subcommands are: \`${this.subcommands.join(
82 "`, `"
83 )}\`.`
84 }
85 }
86 ];
87 public readonly permissions = [PermissionsBitField.Flags.ModerateMembers, PermissionsBitField.Flags.ViewAuditLog];
88 public readonly permissionMode = "or";
89 public readonly description = "Manage infractions.";
90 public readonly detailedDescription = "Use this command to manage infractions.";
91 public readonly argumentSyntaxes = ["<subcommand> [...args]"];
92
93 public readonly slashCommandBuilder = new SlashCommandBuilder()
94 .addSubcommand(subcommand =>
95 subcommand
96 .setName("view")
97 .setDescription("View information about an infraction")
98 .addIntegerOption(option => option.setName("id").setDescription("The infraction ID").setRequired(true))
99 )
100 .addSubcommand(subcommand =>
101 subcommand
102 .setName("edit")
103 .setDescription("Update reason/duration of an infraction")
104 .addIntegerOption(option => option.setName("id").setDescription("The infraction ID").setRequired(true))
105 .addStringOption(option => option.setName("new_reason").setDescription("New reason to set"))
106 .addStringOption(option => option.setName("new_duration").setDescription("New duration to set"))
107 .addBooleanOption(option =>
108 option
109 .setName("silent")
110 .setDescription("Specify if the bot should not let the user know about this, defaults to true")
111 )
112 )
113 .addSubcommand(subcommand =>
114 subcommand
115 .setName("delete")
116 .setDescription("Delete an infraction")
117 .addIntegerOption(option => option.setName("id").setDescription("The infraction ID").setRequired(true))
118 )
119 .addSubcommand(subcommand =>
120 subcommand
121 .setName("clear")
122 .setDescription("Clear infractions for a user")
123 .addUserOption(option => option.setName("user").setDescription("The target user").setRequired(true))
124 .addStringOption(option =>
125 option
126 .setName("type")
127 .setDescription("Specify infraction type")
128 .setChoices(...infractionTypes)
129 )
130 )
131 .addSubcommand(subcommand =>
132 subcommand
133 .setName("list")
134 .setDescription("List infractions for a user")
135 .addUserOption(option => option.setName("user").setDescription("The target user").setRequired(true))
136 )
137 .addSubcommand(subcommand =>
138 subcommand
139 .setName("create")
140 .setDescription("Add infractions to a user")
141 .addUserOption(option => option.setName("user").setDescription("The target user").setRequired(true))
142 .addStringOption(option =>
143 option
144 .setName("type")
145 .setDescription("Specify infraction type")
146 .setChoices(...infractionTypes)
147 .setRequired(true)
148 )
149 .addStringOption(option => option.setName("reason").setDescription("The reason for giving this infraction"))
150 .addStringOption(option =>
151 option.setName("duration").setDescription("The duration of this infraction (e.g. 45, 1h30m)")
152 )
153 );
154
155 async execute(message: CommandMessage, context: BasicCommandContext): Promise<CommandReturn> {
156 const subcommand = context.isLegacy ? context.parsedNamedArgs.subcommand : context.options.getSubcommand(true);
157
158 if ((subcommand === "edit" || subcommand === "create") && context.isLegacy) {
159 await message.reply(
160 `${this.emoji("error")} Please use the slash command \`/infraction ${subcommand}\` to perform this action.`
161 );
162 return;
163 }
164
165 if (["clear", "list"].includes(subcommand)) {
166 if (context.isLegacy && !context.args[1]) {
167 await message.reply(`${this.emoji("error")} Please provide a user to perform this action!`);
168 return;
169 }
170
171 if (context.isLegacy) {
172 let userId = "";
173
174 if (isSnowflake(context.args[1])) {
175 userId = context.args[1];
176 } else if (MessageMentions.UsersPattern.test(context.args[1])) {
177 userId = context.args[1].substring(context.args[1].includes("!") ? 3 : 2, context.args[1].length - 1);
178 } else {
179 await message.reply(
180 `${this.emoji("error")} Please provide a valid user mention or ID to perform this action!`
181 );
182 return;
183 }
184
185 const user = await this.client.fetchUserSafe(userId);
186
187 if (!user) {
188 await message.reply(`${this.emoji("error")} That user does not exist!`);
189 return;
190 }
191
192 context.parsedNamedArgs.user = user;
193 }
194 } else {
195 if (context.isLegacy && !context.args[1]) {
196 await message.reply(`${this.emoji("error")} Please provide an infraction ID to perform this action!`);
197 return;
198 }
199
200 if (context.isLegacy && isNaN(parseInt(context.args[1]))) {
201 await message.reply(`${this.emoji("error")} Please provide a __valid__ infraction ID to perform this action!`);
202 return;
203 }
204
205 if (context.isLegacy) context.parsedNamedArgs.id = parseInt(context.args[1]);
206 }
207
208 await this.deferIfInteraction(message);
209
210 const command = this.client.commands.get(`infraction__${subcommand}`);
211
212 if (command) {
213 return await command.execute(message, {
214 ...context
215 });
216 }
217 }
218 }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26