2023-01-13 05:40:33 +01:00
|
|
|
import type Bull from "bull";
|
|
|
|
import { queueLogger } from "../../logger.js";
|
|
|
|
import { DriveFiles, Notes, UserProfiles, Users } from "@/models/index.js";
|
|
|
|
import type { DbUserDeleteJobData } from "@/queue/types.js";
|
|
|
|
import type { Note } from "@/models/entities/note.js";
|
|
|
|
import type { DriveFile } from "@/models/entities/drive-file.js";
|
|
|
|
import { MoreThan } from "typeorm";
|
|
|
|
import { deleteFileSync } from "@/services/drive/delete-file.js";
|
|
|
|
import { sendEmail } from "@/services/send-email.js";
|
|
|
|
|
|
|
|
const logger = queueLogger.createSubLogger("delete-account");
|
|
|
|
|
|
|
|
export async function deleteAccount(
|
|
|
|
job: Bull.Job<DbUserDeleteJobData>,
|
|
|
|
): Promise<string | void> {
|
2021-08-21 05:41:56 +02:00
|
|
|
logger.info(`Deleting account of ${job.data.user.id} ...`);
|
|
|
|
|
2022-03-26 07:34:00 +01:00
|
|
|
const user = await Users.findOneBy({ id: job.data.user.id });
|
2021-08-21 05:41:56 +02:00
|
|
|
if (user == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
{
|
|
|
|
// Delete notes
|
|
|
|
let cursor: Note["id"] | null = null;
|
2021-08-21 05:41:56 +02:00
|
|
|
|
|
|
|
while (true) {
|
2023-01-13 05:40:33 +01:00
|
|
|
const notes = (await Notes.find({
|
2021-08-21 05:41:56 +02:00
|
|
|
where: {
|
|
|
|
userId: user.id,
|
2021-12-09 15:58:30 +01:00
|
|
|
...(cursor ? { id: MoreThan(cursor) } : {}),
|
2021-08-21 05:41:56 +02:00
|
|
|
},
|
|
|
|
take: 100,
|
|
|
|
order: {
|
2021-12-09 15:58:30 +01:00
|
|
|
id: 1,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
})) as Note[];
|
2021-08-21 05:41:56 +02:00
|
|
|
|
|
|
|
if (notes.length === 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor = notes[notes.length - 1].id;
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
await Notes.delete(notes.map((note) => note.id));
|
2021-08-21 05:41:56 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
logger.succ("All of notes deleted");
|
2021-08-21 05:41:56 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
{
|
|
|
|
// Delete files
|
|
|
|
let cursor: DriveFile["id"] | null = null;
|
2021-08-21 05:41:56 +02:00
|
|
|
|
|
|
|
while (true) {
|
2023-01-13 05:40:33 +01:00
|
|
|
const files = (await DriveFiles.find({
|
2021-08-21 05:41:56 +02:00
|
|
|
where: {
|
|
|
|
userId: user.id,
|
2021-12-09 15:58:30 +01:00
|
|
|
...(cursor ? { id: MoreThan(cursor) } : {}),
|
2021-08-21 05:41:56 +02:00
|
|
|
},
|
|
|
|
take: 10,
|
|
|
|
order: {
|
2021-12-09 15:58:30 +01:00
|
|
|
id: 1,
|
|
|
|
},
|
2023-01-13 05:40:33 +01:00
|
|
|
})) as DriveFile[];
|
2021-08-21 05:41:56 +02:00
|
|
|
|
|
|
|
if (files.length === 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
cursor = files[files.length - 1].id;
|
|
|
|
|
|
|
|
for (const file of files) {
|
|
|
|
await deleteFileSync(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
logger.succ("All of files deleted");
|
2021-08-21 05:41:56 +02:00
|
|
|
}
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
{
|
|
|
|
// Send email notification
|
2022-03-26 07:34:00 +01:00
|
|
|
const profile = await UserProfiles.findOneByOrFail({ userId: user.id });
|
2021-08-21 05:48:50 +02:00
|
|
|
if (profile.email && profile.emailVerified) {
|
2023-01-13 05:40:33 +01:00
|
|
|
sendEmail(
|
|
|
|
profile.email,
|
|
|
|
"Account deleted",
|
|
|
|
"Your account has been deleted.",
|
|
|
|
"Your account has been deleted.",
|
|
|
|
);
|
2021-08-21 05:48:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-22 10:34:48 +02:00
|
|
|
// soft指定されている場合は物理削除しない
|
|
|
|
if (job.data.soft) {
|
|
|
|
// nop
|
|
|
|
} else {
|
|
|
|
await Users.delete(job.data.user.id);
|
|
|
|
}
|
2021-08-21 05:41:56 +02:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
return "Account deleted";
|
2021-08-21 05:41:56 +02:00
|
|
|
}
|