2023-01-13 05:40:33 +01:00
|
|
|
import RE2 from "re2";
|
|
|
|
import type { Note } from "@/models/entities/note.js";
|
|
|
|
import type { User } from "@/models/entities/user.js";
|
2020-07-27 06:34:20 +02:00
|
|
|
|
|
|
|
type NoteLike = {
|
2023-01-13 05:40:33 +01:00
|
|
|
userId: Note["userId"];
|
|
|
|
text: Note["text"];
|
2023-05-18 13:38:33 +02:00
|
|
|
files: Note["files"];
|
2023-01-20 03:11:27 +01:00
|
|
|
cw?: Note["cw"];
|
2020-07-27 06:34:20 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
type UserLike = {
|
2023-01-13 05:40:33 +01:00
|
|
|
id: User["id"];
|
2020-07-27 06:34:20 +02:00
|
|
|
};
|
|
|
|
|
2023-05-04 22:17:16 +02:00
|
|
|
function checkWordMute(
|
|
|
|
note: NoteLike,
|
|
|
|
mutedWords: Array<string | string[]>,
|
|
|
|
): boolean {
|
2023-05-04 06:17:37 +02:00
|
|
|
if (note == null) return false;
|
2023-01-20 03:11:27 +01:00
|
|
|
|
2023-05-18 13:38:33 +02:00
|
|
|
const text = `${note.cw ?? ""} ${note.text ?? ""} ${note.files.map((f) => f.comment ?? "").join(" ")}`.trim();
|
2023-05-04 06:17:37 +02:00
|
|
|
if (text === "") return false;
|
|
|
|
|
2023-05-05 05:36:17 +02:00
|
|
|
for (const mutePattern of mutedWords) {
|
|
|
|
if (Array.isArray(mutePattern)) {
|
|
|
|
// Clean up
|
|
|
|
const keywords = mutePattern.filter((keyword) => keyword !== "");
|
|
|
|
|
|
|
|
if (
|
|
|
|
keywords.length > 0 &&
|
|
|
|
keywords.every((keyword) => text.includes(keyword))
|
|
|
|
)
|
|
|
|
return true;
|
2023-05-04 06:17:37 +02:00
|
|
|
} else {
|
2023-05-04 22:48:31 +02:00
|
|
|
// represents RegExp
|
2023-05-05 05:36:17 +02:00
|
|
|
const regexp = mutePattern.match(/^\/(.+)\/(.*)$/);
|
2023-05-04 22:48:31 +02:00
|
|
|
|
2023-05-04 06:17:37 +02:00
|
|
|
// This should never happen due to input sanitisation.
|
|
|
|
if (!regexp) {
|
2023-05-05 05:36:17 +02:00
|
|
|
console.warn(`Found invalid regex in word mutes: ${mutePattern}`);
|
|
|
|
continue;
|
2023-05-04 06:17:37 +02:00
|
|
|
}
|
|
|
|
|
2023-05-04 22:48:31 +02:00
|
|
|
try {
|
2023-05-05 05:49:34 +02:00
|
|
|
if (new RE2(regexp[1], regexp[2]).test(text)) return true;
|
2023-05-04 22:48:31 +02:00
|
|
|
} catch (err) {
|
|
|
|
// This should never happen due to input sanitisation.
|
|
|
|
}
|
2023-05-04 06:17:37 +02:00
|
|
|
}
|
2023-05-05 05:36:17 +02:00
|
|
|
}
|
2023-05-04 06:17:37 +02:00
|
|
|
|
2023-05-05 05:36:17 +02:00
|
|
|
return false;
|
2023-01-20 03:11:27 +01:00
|
|
|
}
|
|
|
|
|
2023-05-04 07:20:06 +02:00
|
|
|
export async function getWordHardMute(
|
2023-01-13 05:40:33 +01:00
|
|
|
note: NoteLike,
|
|
|
|
me: UserLike | null | undefined,
|
|
|
|
mutedWords: Array<string | string[]>,
|
2023-05-04 06:17:37 +02:00
|
|
|
): Promise<boolean> {
|
2020-07-27 06:34:20 +02:00
|
|
|
// 自分自身
|
2023-01-20 03:11:27 +01:00
|
|
|
if (me && note.userId === me.id) {
|
2023-05-04 06:17:37 +02:00
|
|
|
return false;
|
2023-01-20 03:11:27 +01:00
|
|
|
}
|
2020-07-27 06:34:20 +02:00
|
|
|
|
2022-02-10 11:47:46 +01:00
|
|
|
if (mutedWords.length > 0) {
|
2023-05-04 22:17:16 +02:00
|
|
|
return (
|
|
|
|
checkWordMute(note, mutedWords) ||
|
2023-05-04 23:16:23 +02:00
|
|
|
checkWordMute(note.reply, mutedWords) ||
|
|
|
|
checkWordMute(note.renote, mutedWords)
|
2023-05-04 22:17:16 +02:00
|
|
|
);
|
2020-07-27 06:34:20 +02:00
|
|
|
}
|
|
|
|
|
2023-05-04 06:17:37 +02:00
|
|
|
return false;
|
2020-07-27 06:34:20 +02:00
|
|
|
}
|