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 { User } from "@prisma/client"; |
21 |
import bcrypt from "bcrypt"; |
22 |
import { add } from "date-fns"; |
23 |
import { ActionRowBuilder, ButtonBuilder, ButtonStyle, EmbedBuilder, Snowflake, escapeMarkdown } from "discord.js"; |
24 |
import jwt from "jsonwebtoken"; |
25 |
import { randomInt } from "node:crypto"; |
26 |
import { request as undiciRequest } from "undici"; |
27 |
import { z } from "zod"; |
28 |
import { Action } from "../../decorators/Action"; |
29 |
import { Validate } from "../../decorators/Validate"; |
30 |
import { safeUserFetch } from "../../utils/fetch"; |
31 |
import { logError } from "../../utils/logger"; |
32 |
import Controller from "../Controller"; |
33 |
import Request from "../Request"; |
34 |
import Response from "../Response"; |
35 |
|
36 |
export default class AuthController extends Controller { |
37 |
private isBannedUser(discordId?: Snowflake | null) { |
38 |
if (!discordId) { |
39 |
return true; |
40 |
} |
41 |
|
42 |
if (this.client.commandManager.isBanned(discordId)) { |
43 |
return true; |
44 |
} |
45 |
|
46 |
return false; |
47 |
} |
48 |
|
49 |
private async genToken(user: User) { |
50 |
if (!user.token || (user.token && user.tokenExpiresAt && user.tokenExpiresAt.getTime() <= Date.now())) { |
51 |
user.token = jwt.sign( |
52 |
{ |
53 |
userId: user.id, |
54 |
random: Math.round(Math.random() * 2000) |
55 |
}, |
56 |
process.env.JWT_SECRET!, |
57 |
{ |
58 |
expiresIn: "2 days", |
59 |
issuer: process.env.JWT_ISSUER ?? "SudoBot", |
60 |
subject: "Temporary API token for authenticated user" |
61 |
} |
62 |
); |
63 |
|
64 |
user.tokenExpiresAt = new Date(Date.now() + 1000 * 60 * 60 * 24 * 2); |
65 |
|
66 |
await this.client.prisma.user.updateMany({ |
67 |
where: { |
68 |
id: user.id |
69 |
}, |
70 |
data: { |
71 |
token: user.token, |
72 |
tokenExpiresAt: user.tokenExpiresAt |
73 |
} |
74 |
}); |
75 |
} |
76 |
} |
77 |
|
78 |
@Action("POST", "/auth/recovery_token") |
79 |
@Validate( |
80 |
z.object({ |
81 |
username: z.string(), |
82 |
code: z.number() |
83 |
}) |
84 |
) |
85 |
public async recoveryToken(request: Request) { |
86 |
const { username, code } = request.parsedBody ?? {}; |
87 |
|
88 |
const user = await this.client.prisma.user.findFirst({ |
89 |
where: { |
90 |
username: username.trim(), |
91 |
recoveryCode: code.toString().trim() |
92 |
} |
93 |
}); |
94 |
|
95 |
if (!user) { |
96 |
return new Response({ |
97 |
status: 403, |
98 |
body: { |
99 |
error: "Invalid username or code provided" |
100 |
} |
101 |
}); |
102 |
} |
103 |
|
104 |
if (this.isBannedUser(user.discordId)) { |
105 |
return new Response({ |
106 |
status: 403, |
107 |
body: { |
108 |
error: "Forbidden" |
109 |
} |
110 |
}); |
111 |
} |
112 |
|
113 |
try { |
114 |
jwt.verify(user.recoveryToken!, process.env.JWT_SECRET!, { |
115 |
issuer: process.env.JWT_ISSUER ?? "SudoBot", |
116 |
subject: "Temporary recovery token", |
117 |
complete: true |
118 |
}); |
119 |
} catch (e) { |
120 |
logError(e); |
121 |
|
122 |
return new Response({ |
123 |
status: 403, |
124 |
body: { |
125 |
error: "Invalid username or code provided" |
126 |
} |
127 |
}); |
128 |
} |
129 |
|
130 |
if ((user.recoveryTokenExpiresAt?.getTime() ?? Date.now()) <= Date.now()) { |
131 |
return new Response({ |
132 |
status: 400, |
133 |
body: { |
134 |
error: "This account recovery request has expired" |
135 |
} |
136 |
}); |
137 |
} |
138 |
|
139 |
await this.client.prisma.user.updateMany({ |
140 |
where: { |
141 |
id: user.id |
142 |
}, |
143 |
data: { |
144 |
recoveryCode: null |
145 |
} |
146 |
}); |
147 |
|
148 |
return { success: true, token: user.recoveryToken }; |
149 |
} |
150 |
|
151 |
@Action("POST", "/auth/reset") |
152 |
@Validate( |
153 |
z.object({ |
154 |
username: z.string(), |
155 |
token: z.string(), |
156 |
new_password: z.string() |
157 |
}) |
158 |
) |
159 |
public async reset(request: Request) { |
160 |
const { username, token, new_password } = request.parsedBody ?? {}; |
161 |
|
162 |
const user = await this.client.prisma.user.findFirst({ |
163 |
where: { |
164 |
username: username.trim(), |
165 |
recoveryToken: token.trim() |
166 |
} |
167 |
}); |
168 |
|
169 |
if (!user) { |
170 |
return new Response({ |
171 |
status: 403, |
172 |
body: { |
173 |
error: "Invalid username or token provided" |
174 |
} |
175 |
}); |
176 |
} |
177 |
|
178 |
if (this.isBannedUser(user.discordId)) { |
179 |
return new Response({ |
180 |
status: 403, |
181 |
body: { |
182 |
error: "Forbidden" |
183 |
} |
184 |
}); |
185 |
} |
186 |
|
187 |
try { |
188 |
jwt.verify(user.recoveryToken!, process.env.JWT_SECRET!, { |
189 |
issuer: process.env.JWT_ISSUER ?? "SudoBot", |
190 |
subject: "Temporary recovery token", |
191 |
complete: true |
192 |
}); |
193 |
} catch (e) { |
194 |
logError(e); |
195 |
|
196 |
return new Response({ |
197 |
status: 403, |
198 |
body: { |
199 |
error: "Invalid username or token provided" |
200 |
} |
201 |
}); |
202 |
} |
203 |
|
204 |
if ((user.recoveryTokenExpiresAt?.getTime() ?? Date.now()) <= Date.now()) { |
205 |
return new Response({ |
206 |
status: 400, |
207 |
body: { |
208 |
error: "This account recovery request has expired" |
209 |
} |
210 |
}); |
211 |
} |
212 |
|
213 |
await this.client.prisma.user.updateMany({ |
214 |
where: { |
215 |
id: user.id |
216 |
}, |
217 |
data: { |
218 |
recoveryAttempts: 0, |
219 |
recoveryToken: null, |
220 |
recoveryTokenExpiresAt: null, |
221 |
password: bcrypt.hashSync(new_password, bcrypt.genSaltSync(10)), |
222 |
token: null, |
223 |
tokenExpiresAt: null, |
224 |
recoveryCode: null |
225 |
} |
226 |
}); |
227 |
|
228 |
const discordUser = await safeUserFetch(this.client, user.discordId); |
229 |
|
230 |
await discordUser |
231 |
?.send({ |
232 |
embeds: [ |
233 |
new EmbedBuilder({ |
234 |
author: { |
235 |
name: "Account Successfully Recovered", |
236 |
icon_url: this.client.user?.displayAvatarURL() |
237 |
}, |
238 |
color: 0x007bff, |
239 |
description: `Hey ${escapeMarkdown( |
240 |
discordUser.username |
241 |
)},\n\nYour SudoBot Account was recovered and the password was reset. You've been automatically logged out everywhere you were logged in before.\n\nCheers,\nSudoBot Developers`, |
242 |
footer: { |
243 |
text: "Recovery succeeded" |
244 |
} |
245 |
}).setTimestamp() |
246 |
], |
247 |
components: [ |
248 |
new ActionRowBuilder<ButtonBuilder>().addComponents( |
249 |
new ButtonBuilder() |
250 |
.setStyle(ButtonStyle.Link) |
251 |
.setLabel("Log into your account") |
252 |
.setURL(`${process.env.FRONTEND_URL}/login`) |
253 |
) |
254 |
] |
255 |
}) |
256 |
.catch(logError); |
257 |
|
258 |
return { |
259 |
success: true, |
260 |
message: "Successfully completed account recovery" |
261 |
}; |
262 |
} |
263 |
|
264 |
@Action("POST", "/auth/recovery") |
265 |
@Validate( |
266 |
z.object({ |
267 |
username: z.string() |
268 |
}) |
269 |
) |
270 |
public async recovery(request: Request) { |
271 |
const { username } = request.parsedBody ?? {}; |
272 |
|
273 |
const user = await this.client.prisma.user.findFirst({ |
274 |
where: { |
275 |
username: username.trim() |
276 |
} |
277 |
}); |
278 |
|
279 |
if (!user) { |
280 |
return new Response({ |
281 |
status: 403, |
282 |
body: { |
283 |
error: "Invalid username provided" |
284 |
} |
285 |
}); |
286 |
} |
287 |
|
288 |
if (this.isBannedUser(user.discordId)) { |
289 |
return new Response({ |
290 |
status: 403, |
291 |
body: { |
292 |
error: "Forbidden" |
293 |
} |
294 |
}); |
295 |
} |
296 |
|
297 |
if ( |
298 |
user.recoveryToken && |
299 |
user.recoveryTokenExpiresAt && |
300 |
user.recoveryTokenExpiresAt.getTime() > Date.now() && |
301 |
user.recoveryAttempts >= 2 |
302 |
) { |
303 |
return new Response({ |
304 |
status: 429, |
305 |
body: { |
306 |
error: "This account is already awaiting for recovery" |
307 |
} |
308 |
}); |
309 |
} |
310 |
|
311 |
const recoveryToken = jwt.sign( |
312 |
{ |
313 |
type: "pwdreset", |
314 |
userId: user.id |
315 |
}, |
316 |
process.env.JWT_SECRET!, |
317 |
{ |
318 |
expiresIn: "2 days", |
319 |
issuer: process.env.JWT_ISSUER ?? "SudoBot", |
320 |
subject: "Temporary recovery token" |
321 |
} |
322 |
); |
323 |
|
324 |
const recoveryTokenExpiresAt = add(new Date(), { |
325 |
days: 2 |
326 |
}); |
327 |
|
328 |
const recoveryCode = randomInt(10000, 99999).toString(); |
329 |
|
330 |
await this.client.prisma.user.updateMany({ |
331 |
where: { |
332 |
id: user.id |
333 |
}, |
334 |
data: { |
335 |
recoveryAttempts: { |
336 |
increment: 1 |
337 |
}, |
338 |
recoveryToken, |
339 |
recoveryTokenExpiresAt, |
340 |
recoveryCode |
341 |
} |
342 |
}); |
343 |
|
344 |
const discordUser = await safeUserFetch(this.client, user.discordId); |
345 |
|
346 |
await discordUser |
347 |
?.send({ |
348 |
embeds: [ |
349 |
new EmbedBuilder({ |
350 |
author: { |
351 |
name: "Account Recovery Request", |
352 |
icon_url: this.client.user?.displayAvatarURL() |
353 |
}, |
354 |
color: 0x007bff, |
355 |
description: `Hey ${escapeMarkdown( |
356 |
discordUser.username |
357 |
)},\n\nWe've received a recovery request for your SudoBot Account. Your recovery code is:\n\n# ${recoveryCode}\n\nAlternatively, click the button at the bottom to reset your account's password.\nIf you haven't requested this, feel free to ignore this DM.\n\nCheers,\nSudoBot Developers`, |
358 |
footer: { |
359 |
text: "This account recovery request will be valid for the next 48 hours" |
360 |
} |
361 |
}).setTimestamp() |
362 |
], |
363 |
components: [ |
364 |
new ActionRowBuilder<ButtonBuilder>().addComponents( |
365 |
new ButtonBuilder() |
366 |
.setStyle(ButtonStyle.Link) |
367 |
.setLabel("Reset your password") |
368 |
.setURL( |
369 |
`${process.env.FRONTEND_URL}/account/reset?u=${encodeURIComponent( |
370 |
user.username |
371 |
)}&t=${encodeURIComponent(recoveryToken)}` |
372 |
) |
373 |
) |
374 |
] |
375 |
}) |
376 |
.catch(logError); |
377 |
|
378 |
return { |
379 |
success: true, |
380 |
message: "Successfully initiated account recovery" |
381 |
}; |
382 |
} |
383 |
|
384 |
@Action("POST", "/auth/login") |
385 |
@Validate( |
386 |
z.object({ |
387 |
username: z.string(), |
388 |
password: z.string() |
389 |
}) |
390 |
) |
391 |
public async login(request: Request) { |
392 |
const { username, password } = request.parsedBody ?? {}; |
393 |
|
394 |
const user = await this.client.prisma.user.findFirst({ |
395 |
where: { |
396 |
username: username.trim() |
397 |
} |
398 |
}); |
399 |
|
400 |
if (!user || !bcrypt.compareSync(password.trim(), user.password)) { |
401 |
return new Response({ |
402 |
status: 403, |
403 |
body: { |
404 |
error: "Invalid login credentials" |
405 |
} |
406 |
}); |
407 |
} |
408 |
|
409 |
if (this.isBannedUser(user.discordId)) { |
410 |
return new Response({ |
411 |
status: 403, |
412 |
body: { |
413 |
error: "Unable to log in" |
414 |
} |
415 |
}); |
416 |
} |
417 |
|
418 |
await this.genToken(user); |
419 |
|
420 |
const guilds = []; |
421 |
const discordUser = await safeUserFetch(this.client, user.discordId); |
422 |
|
423 |
for (const id of user.guilds) { |
424 |
const guild = this.client.guilds.cache.get(id); |
425 |
|
426 |
if (guild) { |
427 |
guilds.push({ |
428 |
id: guild.id, |
429 |
name: guild.name, |
430 |
iconURL: guild.iconURL() ?? undefined |
431 |
}); |
432 |
} |
433 |
} |
434 |
|
435 |
return { |
436 |
message: "Login successful", |
437 |
user: { |
438 |
id: user.id, |
439 |
avatarURL: discordUser?.displayAvatarURL(), |
440 |
username: user.username, |
441 |
name: user.name, |
442 |
discordId: user.discordId, |
443 |
guilds, |
444 |
token: user.token, |
445 |
tokenExpiresAt: user.tokenExpiresAt, |
446 |
createdAt: user.createdAt |
447 |
} |
448 |
}; |
449 |
} |
450 |
|
451 |
@Action("POST", "/auth/discord") |
452 |
@Validate( |
453 |
z.object({ |
454 |
code: z.string() |
455 |
}) |
456 |
) |
457 |
async discord(request: Request) { |
458 |
const { parsedBody } = request; |
459 |
|
460 |
try { |
461 |
const tokenResponseData = await undiciRequest("https://discord.com/api/oauth2/token", { |
462 |
method: "POST", |
463 |
body: new URLSearchParams({ |
464 |
client_id: process.env.CLIENT_ID!, |
465 |
client_secret: process.env.CLIENT_SECRET!, |
466 |
code: parsedBody.code, |
467 |
grant_type: "authorization_code", |
468 |
redirect_uri: `${process.env.DISCORD_OAUTH2_REDIRECT_URI}`, |
469 |
scope: "identify" |
470 |
}).toString(), |
471 |
headers: { |
472 |
"Content-Type": "application/x-www-form-urlencoded" |
473 |
} |
474 |
}); |
475 |
|
476 |
const oauthData = <any>await tokenResponseData.body.json(); |
477 |
console.log(oauthData); |
478 |
|
479 |
if (oauthData?.error) { |
480 |
throw new Error(`${oauthData?.error}: ${oauthData?.error_description}`); |
481 |
} |
482 |
|
483 |
const userResponse = await undiciRequest("https://discord.com/api/users/@me", { |
484 |
headers: { |
485 |
authorization: `${oauthData.token_type} ${oauthData.access_token}` |
486 |
} |
487 |
}); |
488 |
|
489 |
const userData = <any>await userResponse.body.json(); |
490 |
|
491 |
if (userData?.error) { |
492 |
throw new Error(`${userData?.error}: ${userData?.error_description}`); |
493 |
} |
494 |
|
495 |
console.log(userData); |
496 |
|
497 |
const avatarURL = `https://cdn.discordapp.com/avatars/${encodeURIComponent(userData.id)}/${encodeURIComponent( |
498 |
userData.avatar |
499 |
)}.${userData.avatar.startsWith("a_") ? "gif" : "webp"}?size=512`; |
500 |
|
501 |
const user = await this.client.prisma.user.findFirst({ |
502 |
where: { |
503 |
discordId: userData.id |
504 |
} |
505 |
}); |
506 |
|
507 |
if (!user) { |
508 |
return new Response({ status: 400, body: "Access denied, no such user found" }); |
509 |
} |
510 |
|
511 |
if (this.isBannedUser(user.discordId)) { |
512 |
return new Response({ |
513 |
status: 403, |
514 |
body: { |
515 |
error: "Unable to log in" |
516 |
} |
517 |
}); |
518 |
} |
519 |
|
520 |
await this.genToken(user); |
521 |
|
522 |
const guilds = []; |
523 |
|
524 |
for (const id of user.guilds) { |
525 |
const guild = this.client.guilds.cache.get(id); |
526 |
|
527 |
if (guild) { |
528 |
guilds.push({ |
529 |
id: guild.id, |
530 |
name: guild.name, |
531 |
iconURL: guild.iconURL() ?? undefined |
532 |
}); |
533 |
} |
534 |
} |
535 |
|
536 |
return { |
537 |
message: "Login successful", |
538 |
user: { |
539 |
id: user.id, |
540 |
avatarURL, |
541 |
username: user.username, |
542 |
name: user.name, |
543 |
discordId: user.discordId, |
544 |
guilds, |
545 |
token: user.token, |
546 |
tokenExpiresAt: user.tokenExpiresAt, |
547 |
createdAt: user.createdAt |
548 |
} |
549 |
}; |
550 |
} catch (error) { |
551 |
console.error(error); |
552 |
return new Response({ status: 400, body: "Invalid oauth2 grant code" }); |
553 |
} |
554 |
} |
555 |
} |