1 |
rakinar2 |
577 |
/** |
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 { ChannelType, Guild, GuildBasedChannel, PermissionFlagsBits, Snowflake, TextChannel, User } from "discord.js"; |
21 |
|
|
import Service from "../core/Service"; |
22 |
|
|
import { log, logError } from "../utils/logger"; |
23 |
|
|
import { getChannelPermissionOverride } from "../utils/utils"; |
24 |
|
|
|
25 |
|
|
export const name = "channelLockManager"; |
26 |
|
|
|
27 |
|
|
type ChannelLockOptions = { |
28 |
|
|
channels?: Snowflake[]; |
29 |
|
|
channelMode?: "exclude" | "include"; |
30 |
|
|
ignorePrivateChannels?: boolean; |
31 |
|
|
moderator: User; |
32 |
|
|
reason?: string; |
33 |
|
|
}; |
34 |
|
|
|
35 |
|
|
export default class ChannelLockManager extends Service { |
36 |
|
|
async shouldLock(channels: Snowflake[], channel: GuildBasedChannel, channelMode: "exclude" | "include") { |
37 |
|
|
return ( |
38 |
|
|
channels.length === 0 || |
39 |
|
|
(channelMode === "include" && channels.includes(channel.id)) || |
40 |
|
|
(channelMode === "exclude" && !channels.includes(channel.id)) |
41 |
|
|
); |
42 |
|
|
} |
43 |
|
|
|
44 |
|
|
async lockGuild(guild: Guild, { reason, moderator, channels = [], channelMode = "include", ignorePrivateChannels = true }: ChannelLockOptions) { |
45 |
|
|
let countSuccess = 0, |
46 |
|
|
countFailed = 0, |
47 |
|
|
countSkipped = 0, |
48 |
|
|
countInvalidChannel = 0; |
49 |
|
|
|
50 |
|
|
const originalPermissions = []; |
51 |
|
|
|
52 |
|
|
for (const [, channel] of guild.channels.cache) { |
53 |
|
|
log(channel.name); |
54 |
|
|
|
55 |
|
|
if ( |
56 |
|
|
!this.shouldLock(channels, channel, channelMode) || |
57 |
|
|
![ |
58 |
|
|
ChannelType.GuildAnnouncement, |
59 |
|
|
ChannelType.GuildCategory, |
60 |
|
|
ChannelType.GuildText, |
61 |
|
|
ChannelType.GuildVoice, |
62 |
|
|
ChannelType.GuildStageVoice, |
63 |
|
|
ChannelType.GuildForum |
64 |
|
|
].includes(channel.type) |
65 |
|
|
) { |
66 |
|
|
countInvalidChannel++; |
67 |
|
|
continue; |
68 |
|
|
} |
69 |
|
|
|
70 |
|
|
try { |
71 |
|
|
const options = { |
72 |
|
|
Connect: false, |
73 |
|
|
SendMessages: false, |
74 |
|
|
SendMessagesInThreads: false, |
75 |
|
|
AddReactions: false |
76 |
|
|
}; |
77 |
|
|
|
78 |
|
|
const permissionOverwrites = (channel as TextChannel).permissionOverwrites?.cache.get(guild.id); |
79 |
|
|
|
80 |
|
|
if (ignorePrivateChannels) { |
81 |
|
|
if (permissionOverwrites?.deny.has(PermissionFlagsBits.ViewChannel, true)) { |
82 |
|
|
log("Private channel, skipping lock"); |
83 |
|
|
countSkipped++; |
84 |
|
|
continue; |
85 |
|
|
} |
86 |
|
|
} |
87 |
|
|
|
88 |
|
|
const permissionJson = { |
89 |
|
|
Connect: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.Connect, permissionOverwrites) : null, |
90 |
|
|
SendMessages: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.SendMessages, permissionOverwrites) : null, |
91 |
|
|
SendMessagesInThreads: permissionOverwrites |
92 |
|
|
? getChannelPermissionOverride(PermissionFlagsBits.SendMessagesInThreads, permissionOverwrites) |
93 |
|
|
: null, |
94 |
|
|
AddReactions: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.AddReactions, permissionOverwrites) : null |
95 |
|
|
}; |
96 |
|
|
|
97 |
|
|
originalPermissions.push({ |
98 |
|
|
channel_id: channel.id, |
99 |
|
|
guild_id: guild.id, |
100 |
|
|
permissions: permissionJson |
101 |
|
|
}); |
102 |
|
|
|
103 |
|
|
await (channel as TextChannel).permissionOverwrites?.edit(guild.id, options); |
104 |
|
|
|
105 |
|
|
countSuccess++; |
106 |
|
|
} catch (e) { |
107 |
|
|
logError(e); |
108 |
|
|
countFailed++; |
109 |
|
|
} |
110 |
|
|
} |
111 |
|
|
|
112 |
|
|
await this.client.prisma.channelLock.createMany({ |
113 |
|
|
data: originalPermissions |
114 |
|
|
}); |
115 |
|
|
|
116 |
|
|
log(originalPermissions); |
117 |
|
|
|
118 |
|
|
await this.client.logger.logServerLockOrUnlock({ |
119 |
|
|
guild, |
120 |
|
|
action: "Locked", |
121 |
|
|
moderator, |
122 |
|
|
countSuccess, |
123 |
|
|
countFailed, |
124 |
|
|
countInvalidChannel, |
125 |
|
|
countSkipped, |
126 |
|
|
reason |
127 |
|
|
}); |
128 |
|
|
|
129 |
|
|
return { |
130 |
|
|
countSuccess, |
131 |
|
|
countFailed, |
132 |
|
|
countInvalidChannel, |
133 |
|
|
countSkipped |
134 |
|
|
}; |
135 |
|
|
} |
136 |
|
|
|
137 |
|
|
async unlockGuild( |
138 |
|
|
guild: Guild, |
139 |
|
|
{ channels = [], channelMode = "include", ignorePrivateChannels = true, force, moderator, reason }: ChannelLockOptions & { force?: boolean } |
140 |
|
|
) { |
141 |
|
|
let countSuccess = 0, |
142 |
|
|
countFailed = 0, |
143 |
|
|
countSkipped = 0, |
144 |
|
|
countInvalidChannel = 0; |
145 |
|
|
|
146 |
|
|
const originalPermissions = force |
147 |
|
|
? guild.channels.cache.map(c => ({ channel_id: c.id, permissions: {}, id: 0 })) |
148 |
|
|
: await this.client.prisma.channelLock.findMany({ |
149 |
|
|
where: { |
150 |
|
|
guild_id: guild.id |
151 |
|
|
} |
152 |
|
|
}); |
153 |
|
|
|
154 |
|
|
for (const originalPermission of originalPermissions) { |
155 |
|
|
const channel = guild.channels.cache.get(originalPermission.channel_id); |
156 |
|
|
|
157 |
|
|
if (!channel) continue; |
158 |
|
|
|
159 |
|
|
if ( |
160 |
|
|
!this.shouldLock(channels, channel, channelMode) || |
161 |
|
|
![ |
162 |
|
|
ChannelType.GuildAnnouncement, |
163 |
|
|
ChannelType.GuildCategory, |
164 |
|
|
ChannelType.GuildText, |
165 |
|
|
ChannelType.GuildVoice, |
166 |
|
|
ChannelType.GuildStageVoice, |
167 |
|
|
ChannelType.GuildForum |
168 |
|
|
].includes(channel.type) |
169 |
|
|
) { |
170 |
|
|
countInvalidChannel++; |
171 |
|
|
continue; |
172 |
|
|
} |
173 |
|
|
|
174 |
|
|
try { |
175 |
|
|
const options = { |
176 |
|
|
Connect: force ? true : (originalPermission.permissions! as any).Connect, |
177 |
|
|
SendMessages: force ? true : (originalPermission.permissions! as any).SendMessages, |
178 |
|
|
SendMessagesInThreads: force ? true : (originalPermission.permissions! as any).SendMessagesInThreads, |
179 |
|
|
AddReactions: force ? true : (originalPermission.permissions! as any).AddReactions |
180 |
|
|
}; |
181 |
|
|
|
182 |
|
|
const permissionOverwrites = (channel as TextChannel).permissionOverwrites?.cache.get(guild.id); |
183 |
|
|
|
184 |
|
|
if (ignorePrivateChannels && !force) { |
185 |
|
|
if (permissionOverwrites?.deny.has(PermissionFlagsBits.ViewChannel, true)) { |
186 |
|
|
log("Private channel, skipping lock"); |
187 |
|
|
countSkipped++; |
188 |
|
|
continue; |
189 |
|
|
} |
190 |
|
|
} |
191 |
|
|
|
192 |
|
|
if (permissionOverwrites) await (channel as TextChannel).permissionOverwrites?.edit(guild.id, options); |
193 |
|
|
else await (channel as TextChannel).permissionOverwrites?.create(guild.id, options); |
194 |
|
|
|
195 |
|
|
countSuccess++; |
196 |
|
|
} catch (e) { |
197 |
|
|
logError(e); |
198 |
|
|
countFailed++; |
199 |
|
|
} |
200 |
|
|
} |
201 |
|
|
|
202 |
|
|
log(originalPermissions); |
203 |
|
|
|
204 |
|
|
if (!force) |
205 |
|
|
await this.client.prisma.channelLock.deleteMany({ |
206 |
|
|
where: { |
207 |
|
|
id: { |
208 |
|
|
in: originalPermissions.map(permission => permission.id) |
209 |
|
|
} |
210 |
|
|
} |
211 |
|
|
}); |
212 |
|
|
|
213 |
|
|
await this.client.logger.logServerLockOrUnlock({ |
214 |
|
|
guild, |
215 |
|
|
action: "Unlocked", |
216 |
|
|
moderator, |
217 |
|
|
countSuccess, |
218 |
|
|
countFailed, |
219 |
|
|
countInvalidChannel, |
220 |
|
|
countSkipped, |
221 |
|
|
reason |
222 |
|
|
}); |
223 |
|
|
|
224 |
|
|
return { |
225 |
|
|
countSuccess, |
226 |
|
|
countFailed, |
227 |
|
|
countInvalidChannel, |
228 |
|
|
countSkipped |
229 |
|
|
}; |
230 |
|
|
} |
231 |
|
|
|
232 |
|
|
async lock(channel: TextChannel, moderator: User, reason?: string) { |
233 |
|
|
try { |
234 |
|
|
const options = { |
235 |
|
|
Connect: false, |
236 |
|
|
SendMessages: false, |
237 |
|
|
SendMessagesInThreads: false, |
238 |
|
|
AddReactions: false |
239 |
|
|
}; |
240 |
|
|
|
241 |
|
|
const permissionOverwrites = channel.permissionOverwrites?.cache.get(channel.guild.id); |
242 |
|
|
|
243 |
|
|
const permissionJson = { |
244 |
|
|
Connect: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.Connect, permissionOverwrites) : null, |
245 |
|
|
SendMessages: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.SendMessages, permissionOverwrites) : null, |
246 |
|
|
SendMessagesInThreads: permissionOverwrites |
247 |
|
|
? getChannelPermissionOverride(PermissionFlagsBits.SendMessagesInThreads, permissionOverwrites) |
248 |
|
|
: null, |
249 |
|
|
AddReactions: permissionOverwrites ? getChannelPermissionOverride(PermissionFlagsBits.AddReactions, permissionOverwrites) : null |
250 |
|
|
}; |
251 |
|
|
|
252 |
|
|
if (permissionOverwrites) await channel.permissionOverwrites?.edit(channel.guild.id, options); |
253 |
|
|
else await channel.permissionOverwrites?.create(channel.guild.id, options); |
254 |
|
|
|
255 |
|
|
this.client.logger |
256 |
|
|
.logChannelLockOrUnlock({ |
257 |
|
|
guild: channel.guild, |
258 |
|
|
action: "Locked", |
259 |
|
|
channel, |
260 |
|
|
moderator, |
261 |
|
|
reason |
262 |
|
|
}) |
263 |
|
|
.catch(logError); |
264 |
|
|
|
265 |
|
|
return await this.client.prisma.channelLock.create({ |
266 |
|
|
data: { |
267 |
|
|
channel_id: channel.id, |
268 |
|
|
guild_id: channel.guild.id, |
269 |
|
|
permissions: permissionJson |
270 |
|
|
} |
271 |
|
|
}); |
272 |
|
|
} catch (e) { |
273 |
|
|
logError(e); |
274 |
|
|
return null; |
275 |
|
|
} |
276 |
|
|
} |
277 |
|
|
|
278 |
|
|
async unlock(channel: TextChannel, moderator: User, reason?: string, force?: boolean) { |
279 |
|
|
try { |
280 |
|
|
const channelLock = await this.client.prisma.channelLock.findFirst({ |
281 |
|
|
where: { |
282 |
|
|
channel_id: channel.id, |
283 |
|
|
guild_id: channel.guild.id |
284 |
|
|
} |
285 |
|
|
}); |
286 |
|
|
|
287 |
|
|
if (!channelLock) return null; |
288 |
|
|
|
289 |
|
|
const options = { |
290 |
|
|
Connect: force ? true : (channelLock.permissions as any).Connect, |
291 |
|
|
SendMessages: force ? true : (channelLock.permissions as any).SendMessages, |
292 |
|
|
SendMessagesInThreads: force ? true : (channelLock.permissions as any).SendMessagesInThreads, |
293 |
|
|
AddReactions: force ? true : (channelLock.permissions as any).AddReactions |
294 |
|
|
}; |
295 |
|
|
|
296 |
|
|
const permissionOverwrites = channel.permissionOverwrites?.cache.get(channel.guild.id); |
297 |
|
|
|
298 |
|
|
if (permissionOverwrites) await channel.permissionOverwrites?.edit(channel.guild.id, options); |
299 |
|
|
else await channel.permissionOverwrites?.create(channel.guild.id, options); |
300 |
|
|
|
301 |
|
|
this.client.logger |
302 |
|
|
.logChannelLockOrUnlock({ |
303 |
|
|
guild: channel.guild, |
304 |
|
|
action: "Unlocked", |
305 |
|
|
channel, |
306 |
|
|
moderator, |
307 |
|
|
reason |
308 |
|
|
}) |
309 |
|
|
.catch(logError); |
310 |
|
|
|
311 |
|
|
return true; |
312 |
|
|
} catch (e) { |
313 |
|
|
logError(e); |
314 |
|
|
return false; |
315 |
|
|
} |
316 |
|
|
} |
317 |
|
|
} |