/[sudobot]/branches/5.x/src/commands/moderation/InfractionEditCommand.ts
ViewVC logotype

Contents of /branches/5.x/src/commands/moderation/InfractionEditCommand.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: 8085 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 { formatDistanceToNowStrict } from "date-fns";
21 import { ChatInputCommandInteraction, EmbedBuilder, PermissionsBitField } from "discord.js";
22 import Command, { CommandReturn, ValidationRule } from "../../core/Command";
23 import { ChatInputCommandContext } from "../../services/CommandManager";
24 import { stringToTimeInterval } from "../../utils/datetime";
25
26 export default class InfractionEditCommand extends Command {
27 public readonly name = "infraction__edit";
28 public readonly validationRules: ValidationRule[] = [];
29 public readonly permissions = [PermissionsBitField.Flags.ModerateMembers, PermissionsBitField.Flags.ViewAuditLog];
30 public readonly permissionMode = "or";
31 public readonly supportsLegacy: boolean = false;
32
33 public readonly description = "Edit infractions.";
34 public readonly detailedDescription = "Update an infraction with a new reason or duration or both.";
35 public readonly argumentSyntaxes = ["<infraction_id> [new_reason] [new_duration]"];
36
37 async execute(interaction: ChatInputCommandInteraction, context: ChatInputCommandContext): Promise<CommandReturn> {
38 const id = interaction.options.getInteger("id", true);
39 const newReason = interaction.options.getString("new_reason");
40 const newDuration = interaction.options.getString("new_duration");
41
42 if (!newReason && !newDuration) {
43 await interaction.editReply(
44 `${this.emoji("error")} Either provide a reason or duration or both to update this infraction, if it exists!`
45 );
46 return;
47 }
48
49 const newDurationSeconds = newDuration ? stringToTimeInterval(newDuration) : undefined;
50 const silent = interaction.options.getBoolean("silent") ?? true;
51
52 if (newDurationSeconds && newDurationSeconds.error) {
53 await interaction.editReply(
54 `${this.emoji("error")} ${newDurationSeconds.error} provided in the \`new_duration\` field`
55 );
56 return;
57 }
58
59 const infraction = await this.client.prisma.infraction.findFirst({
60 where: { id, guildId: interaction.guildId! }
61 });
62
63 if (!infraction) {
64 await interaction.editReply(`${this.emoji("error")} Could not find an infraction with that ID!`);
65 return;
66 }
67
68 if (newDurationSeconds?.result && infraction.expiresAt && infraction.expiresAt.getTime() <= Date.now()) {
69 await interaction.editReply(`${this.emoji("error")} That infraction is expired, so you can't change it's duration!`);
70 return;
71 }
72
73 if (newDurationSeconds?.result && infraction.createdAt.getTime() + newDurationSeconds?.result * 1000 <= Date.now()) {
74 await interaction.editReply(
75 `${this.emoji(
76 "error"
77 )} That duration makes the infraction expire in the past, which is not possible! Please make sure the time of infraction creation plus the new duration is greater than the current time!`
78 );
79 return;
80 }
81
82 if (newDurationSeconds?.result && infraction.expiresAt === null) {
83 await interaction.editReply(
84 `${this.emoji("error")} This infraction did not have a duration in the first place, so you can't set one now.`
85 );
86
87 return;
88 }
89
90 const user = await this.client.fetchUserSafe(infraction.userId);
91
92 if (newDurationSeconds && infraction.queueId) {
93 const queue = this.client.queueManager.queues.get(`${infraction.queueId}`);
94
95 if (queue) {
96 await queue.updateTime(new Date(infraction.createdAt.getTime() + newDurationSeconds.result * 1000));
97 }
98 }
99
100 if (!silent) {
101 await user?.send({
102 embeds: [
103 new EmbedBuilder({
104 author: {
105 icon_url: interaction.guild?.iconURL() ?? undefined,
106 name: "Your infraction has been updated in " + interaction.guild!.name
107 },
108 color: 0xf14a60,
109 fields: [
110 ...(newReason
111 ? [
112 {
113 name: "Reason",
114 value: newReason
115 }
116 ]
117 : []),
118 ...(newDurationSeconds?.result
119 ? [
120 {
121 name: "Duration",
122 value: formatDistanceToNowStrict(
123 new Date(infraction.createdAt.getTime() + newDurationSeconds.result * 1000)
124 )
125 }
126 ]
127 : []),
128 {
129 name: "Infraction ID",
130 value: id.toString()
131 }
132 ]
133 }).setTimestamp()
134 ]
135 });
136 }
137
138 await this.client.prisma.infraction.update({
139 data: {
140 reason: newReason ?? undefined,
141 expiresAt: newDurationSeconds
142 ? new Date(infraction.createdAt.getTime() + newDurationSeconds.result * 1000)
143 : undefined
144 },
145 where: { id }
146 });
147
148 await interaction.editReply({
149 embeds: [
150 new EmbedBuilder({
151 title: "Infraction update",
152 description: "Updated successfully!",
153 color: 0x007bff,
154 fields: [
155 ...(newReason
156 ? [
157 {
158 name: "New Reason",
159 value: newReason
160 }
161 ]
162 : []),
163 ...(newDurationSeconds?.result
164 ? [
165 {
166 name: "New Duration",
167 value: formatDistanceToNowStrict(
168 new Date(infraction.createdAt.getTime() + newDurationSeconds.result * 1000)
169 )
170 }
171 ]
172 : []),
173 {
174 name: "ID",
175 value: id.toString()
176 },
177 {
178 name: "Notifying User?",
179 value: silent ? "No" : "Yes",
180 inline: true
181 }
182 ],
183 footer: {
184 text: "Updated"
185 }
186 }).setTimestamp()
187 ]
188 });
189 }
190 }

[email protected]
ViewVC Help
Powered by ViewVC 1.1.26