2023-07-27 05:31:52 +00:00
|
|
|
/*
|
2024-02-12 02:37:45 +00:00
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 05:31:52 +00:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-05-05 10:49:34 +00:00
|
|
|
import { AhoCorasick } from 'slacc';
|
2022-02-27 02:07:39 +00:00
|
|
|
import RE2 from 're2';
|
2023-09-20 02:33:36 +00:00
|
|
|
import type { MiNote } from '@/models/Note.js';
|
|
|
|
import type { MiUser } from '@/models/User.js';
|
2020-07-27 04:34:20 +00:00
|
|
|
|
|
|
|
type NoteLike = {
|
2023-08-16 08:51:28 +00:00
|
|
|
userId: MiNote['userId'];
|
|
|
|
text: MiNote['text'];
|
|
|
|
cw?: MiNote['cw'];
|
2020-07-27 04:34:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
type UserLike = {
|
2023-08-16 08:51:28 +00:00
|
|
|
id: MiUser['id'];
|
2020-07-27 04:34:20 +00:00
|
|
|
};
|
|
|
|
|
2023-05-05 10:49:34 +00:00
|
|
|
const acCache = new Map<string, AhoCorasick>();
|
|
|
|
|
2022-02-10 10:47:46 +00:00
|
|
|
export async function checkWordMute(note: NoteLike, me: UserLike | null | undefined, mutedWords: Array<string | string[]>): Promise<boolean> {
|
2020-07-27 04:34:20 +00:00
|
|
|
// 自分自身
|
|
|
|
if (me && (note.userId === me.id)) return false;
|
|
|
|
|
2022-02-10 10:47:46 +00:00
|
|
|
if (mutedWords.length > 0) {
|
2022-06-23 11:26:47 +00:00
|
|
|
const text = ((note.cw ?? '') + '\n' + (note.text ?? '')).trim();
|
|
|
|
|
2022-06-24 10:44:22 +00:00
|
|
|
if (text === '') return false;
|
2020-07-27 04:34:20 +00:00
|
|
|
|
2023-05-05 10:49:34 +00:00
|
|
|
const acable = mutedWords.filter(filter => Array.isArray(filter) && filter.length === 1).map(filter => filter[0]).sort();
|
|
|
|
const unacable = mutedWords.filter(filter => !Array.isArray(filter) || filter.length !== 1);
|
|
|
|
const acCacheKey = acable.join('\n');
|
|
|
|
const ac = acCache.get(acCacheKey) ?? AhoCorasick.withPatterns(acable);
|
|
|
|
acCache.delete(acCacheKey);
|
|
|
|
for (const obsoleteKeys of acCache.keys()) {
|
|
|
|
if (acCache.size > 1000) {
|
|
|
|
acCache.delete(obsoleteKeys);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
acCache.set(acCacheKey, ac);
|
|
|
|
if (ac.isMatch(text)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const matched = unacable.some(filter => {
|
2022-02-10 10:47:46 +00:00
|
|
|
if (Array.isArray(filter)) {
|
2022-06-23 11:26:47 +00:00
|
|
|
return filter.every(keyword => text.includes(keyword));
|
2022-02-10 10:47:46 +00:00
|
|
|
} else {
|
|
|
|
// represents RegExp
|
|
|
|
const regexp = filter.match(/^\/(.+)\/(.*)$/);
|
|
|
|
|
|
|
|
// This should never happen due to input sanitisation.
|
|
|
|
if (!regexp) return false;
|
|
|
|
|
|
|
|
try {
|
2022-06-23 11:26:47 +00:00
|
|
|
return new RE2(regexp[1], regexp[2]).test(text);
|
2022-02-10 10:47:46 +00:00
|
|
|
} catch (err) {
|
|
|
|
// This should never happen due to input sanitisation.
|
|
|
|
return false;
|
2020-07-27 04:34:20 +00:00
|
|
|
}
|
2022-02-10 10:47:46 +00:00
|
|
|
}
|
|
|
|
});
|
2020-07-27 04:34:20 +00:00
|
|
|
|
|
|
|
if (matched) return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|