Refactor hard word mutes

This commit is contained in:
naskya 2023-05-04 13:17:37 +09:00
parent f6428ca16b
commit d09fc8be2b
10 changed files with 72 additions and 55 deletions

View file

@ -12,67 +12,63 @@ type UserLike = {
id: User["id"]; id: User["id"];
}; };
export type Muted = { function escapeRegExp(x: string): string {
muted: boolean;
matched: string[];
};
const NotMuted = { muted: false, matched: [] };
function escapeRegExp(x: string) {
return x.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string return x.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string
} }
function checkWordMute(note: NoteLike): boolean {
if (note == null) return false;
const text = ((note.cw ?? "") + " " + (note.text ?? "")).trim();
if (text === "") return false;
for (const mutePattern of mutedWords) {
let mute: RE2;
let matched: string[];
if (Array.isArray(mutePattern)) {
matched = mutePattern.filter((keyword) => keyword !== "");
if (matched.length === 0) {
continue;
}
mute = new RE2(
`\\b${matched.map(escapeRegExp).join("\\b.*\\b")}\\b`,
"g",
);
} else {
const regexp = mutePattern.match(/^\/(.+)\/(.*)$/);
// This should never happen due to input sanitisation.
if (!regexp) {
console.warn(`Found invalid regex in word mutes: ${mutePattern}`);
continue;
}
mute = new RE2(regexp[1], regexp[2]);
matched = [mutePattern];
}
try {
if (mute.test(text)) return true;
} catch (err) {
// This should never happen due to input sanitisation.
}
}
return notMuted;
}
export async function getWordMute( export async function getWordMute(
note: NoteLike, note: NoteLike,
me: UserLike | null | undefined, me: UserLike | null | undefined,
mutedWords: Array<string | string[]>, mutedWords: Array<string | string[]>,
): Promise<Muted> { ): Promise<boolean> {
// 自分自身 // 自分自身
if (me && note.userId === me.id) { if (me && note.userId === me.id) {
return NotMuted; return false;
} }
if (mutedWords.length > 0) { if (mutedWords.length > 0) {
const text = ((note.cw ?? "") + "\n" + (note.text ?? "")).trim(); return checkWordMute(note) || checkWordMute(reply) || checkWordMute(renote)
if (text === "") {
return NotMuted;
}
for (const mutePattern of mutedWords) {
let mute: RE2;
let matched: string[];
if (Array.isArray(mutePattern)) {
matched = mutePattern.filter((keyword) => keyword !== "");
if (matched.length === 0) {
continue;
}
mute = new RE2(
`\\b${matched.map(escapeRegExp).join("\\b.*\\b")}\\b`,
"g",
);
} else {
const regexp = mutePattern.match(/^\/(.+)\/(.*)$/);
// This should never happen due to input sanitisation.
if (!regexp) {
console.warn(`Found invalid regex in word mutes: ${mutePattern}`);
continue;
}
mute = new RE2(regexp[1], regexp[2]);
matched = [mutePattern];
}
try {
if (mute.test(text)) {
return { muted: true, matched };
}
} catch (err) {
// This should never happen due to input sanitisation.
}
}
} }
return NotMuted; return false;
} }

View file

@ -1,6 +1,7 @@
import Channel from "../channel.js"; import Channel from "../channel.js";
import { Notes } from "@/models/index.js"; import { Notes } from "@/models/index.js";
import { isUserRelated } from "@/misc/is-user-related.js"; import { isUserRelated } from "@/misc/is-user-related.js";
import { getWordMute } from "@/misc/check-word-mute.js";
import type { StreamMessages } from "../types.js"; import type { StreamMessages } from "../types.js";
import { IdentifiableError } from "@/misc/identifiable-error.js"; import { IdentifiableError } from "@/misc/identifiable-error.js";
@ -37,6 +38,12 @@ export default class extends Channel {
if (note.renote && !note.text && isUserRelated(note, this.renoteMuting)) if (note.renote && !note.text && isUserRelated(note, this.renoteMuting))
return; return;
if (
this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords))
)
return;
this.connection.cacheNote(note); this.connection.cacheNote(note);
this.send("note", note); this.send("note", note);

View file

@ -1,6 +1,7 @@
import Channel from "../channel.js"; import Channel from "../channel.js";
import { Users } from "@/models/index.js"; import { Users } from "@/models/index.js";
import { isUserRelated } from "@/misc/is-user-related.js"; import { isUserRelated } from "@/misc/is-user-related.js";
import { getWordMute } from "@/misc/check-word-mute.js";
import type { User } from "@/models/entities/user.js"; import type { User } from "@/models/entities/user.js";
import type { StreamMessages } from "../types.js"; import type { StreamMessages } from "../types.js";
import type { Packed } from "@/misc/schema.js"; import type { Packed } from "@/misc/schema.js";
@ -39,6 +40,12 @@ export default class extends Channel {
if (note.renote && !note.text && isUserRelated(note, this.renoteMuting)) if (note.renote && !note.text && isUserRelated(note, this.renoteMuting))
return; return;
if (
this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords))
)
return;
this.connection.cacheNote(note); this.connection.cacheNote(note);
this.send("note", note); this.send("note", note);

View file

@ -66,7 +66,7 @@ export default class extends Channel {
// そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる // そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる
if ( if (
this.userProfile && this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords)).muted (await getWordMute(note, this.user, this.userProfile.mutedWords))
) )
return; return;

View file

@ -64,7 +64,7 @@ export default class extends Channel {
// そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる // そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる
if ( if (
this.userProfile && this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords)).muted (await getWordMute(note, this.user, this.userProfile.mutedWords))
) )
return; return;

View file

@ -81,7 +81,7 @@ export default class extends Channel {
// そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる // そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる
if ( if (
this.userProfile && this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords)).muted (await getWordMute(note, this.user, this.userProfile.mutedWords))
) )
return; return;

View file

@ -58,7 +58,7 @@ export default class extends Channel {
// そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる // そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる
if ( if (
this.userProfile && this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords)).muted (await getWordMute(note, this.user, this.userProfile.mutedWords))
) )
return; return;

View file

@ -79,7 +79,7 @@ export default class extends Channel {
// そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる // そのためレコードが存在するかのチェックでは不十分なので、改めてgetWordMuteを呼んでいる
if ( if (
this.userProfile && this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords)).muted (await getWordMute(note, this.user, this.userProfile.mutedWords))
) )
return; return;

View file

@ -2,6 +2,7 @@ import Channel from "../channel.js";
import { UserListJoinings, UserLists } from "@/models/index.js"; import { UserListJoinings, UserLists } from "@/models/index.js";
import type { User } from "@/models/entities/user.js"; import type { User } from "@/models/entities/user.js";
import { isUserRelated } from "@/misc/is-user-related.js"; import { isUserRelated } from "@/misc/is-user-related.js";
import { getWordMute } from "@/misc/check-word-mute.js";
import type { Packed } from "@/misc/schema.js"; import type { Packed } from "@/misc/schema.js";
export default class extends Channel { export default class extends Channel {
@ -59,6 +60,12 @@ export default class extends Channel {
if (note.renote && !note.text && isUserRelated(note, this.renoteMuting)) if (note.renote && !note.text && isUserRelated(note, this.renoteMuting))
return; return;
if (
this.userProfile &&
(await getWordMute(note, this.user, this.userProfile.mutedWords))
)
return;
this.send("note", note); this.send("note", note);
} }

View file

@ -356,7 +356,7 @@ export default async (
for (const u of us) { for (const u of us) {
getWordMute(note, { id: u.userId }, u.mutedWords).then( getWordMute(note, { id: u.userId }, u.mutedWords).then(
(shouldMute) => { (shouldMute) => {
if (shouldMute.muted) { if (shouldMute) {
MutedNotes.insert({ MutedNotes.insert({
id: genId(), id: genId(),
userId: u.userId, userId: u.userId,