summaryrefslogtreecommitdiff
path: root/packages/backend/src/misc/check-word-mute.ts
blob: a7a60c20e9f248af91adfee327d698530797ffd7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
/*
 * SPDX-FileCopyrightText: syuilo and other misskey contributors
 * SPDX-License-Identifier: AGPL-3.0-only
 */

import { AhoCorasick } from 'slacc';
import RE2 from 're2';
import type { MiNote } from '@/models/entities/Note.js';
import type { MiUser } from '@/models/entities/User.js';

type NoteLike = {
	userId: MiNote['userId'];
	text: MiNote['text'];
	cw?: MiNote['cw'];
};

type UserLike = {
	id: MiUser['id'];
};

const acCache = new Map<string, AhoCorasick>();

export async function checkWordMute(note: NoteLike, me: UserLike | null | undefined, mutedWords: Array<string | string[]>): Promise<boolean> {
	// 自分自身
	if (me && (note.userId === me.id)) return false;

	if (mutedWords.length > 0) {
		const text = ((note.cw ?? '') + '\n' + (note.text ?? '')).trim();

		if (text === '') return false;

		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 => {
			if (Array.isArray(filter)) {
				return filter.every(keyword => text.includes(keyword));
			} else {
				// represents RegExp
				const regexp = filter.match(/^\/(.+)\/(.*)$/);

				// This should never happen due to input sanitisation.
				if (!regexp) return false;

				try {
					return new RE2(regexp[1], regexp[2]).test(text);
				} catch (err) {
					// This should never happen due to input sanitisation.
					return false;
				}
			}
		});

		if (matched) return true;
	}

	return false;
}