2023-07-27 00:31:52 -05:00
|
|
|
/*
|
2024-02-13 09:59:27 -06:00
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 00:31:52 -05:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
2023-04-13 23:50:05 -05:00
|
|
|
import * as Redis from 'ioredis';
|
2023-09-19 21:33:36 -05:00
|
|
|
import type { MiAntenna } from '@/models/Antenna.js';
|
|
|
|
import type { MiNote } from '@/models/Note.js';
|
|
|
|
import type { MiUser } from '@/models/User.js';
|
2022-09-17 13:27:08 -05:00
|
|
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
|
|
|
import * as Acct from '@/misc/acct.js';
|
2023-03-09 23:22:37 -06:00
|
|
|
import type { Packed } from '@/misc/json-schema.js';
|
2022-09-17 13:27:08 -05:00
|
|
|
import { DI } from '@/di-symbols.js';
|
2023-10-03 06:26:11 -05:00
|
|
|
import type { AntennasRepository, UserListMembershipsRepository } from '@/models/_.js';
|
2022-12-03 19:16:03 -06:00
|
|
|
import { UtilityService } from '@/core/UtilityService.js';
|
2022-12-04 00:03:09 -06:00
|
|
|
import { bindThis } from '@/decorators.js';
|
2023-09-28 21:29:54 -05:00
|
|
|
import type { GlobalEvents } from '@/core/GlobalEventService.js';
|
2023-11-25 19:02:22 -06:00
|
|
|
import { FanoutTimelineService } from '@/core/FanoutTimelineService.js';
|
2023-01-08 23:12:42 -06:00
|
|
|
import type { OnApplicationShutdown } from '@nestjs/common';
|
2022-09-17 13:27:08 -05:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class AntennaService implements OnApplicationShutdown {
|
2022-09-18 13:11:50 -05:00
|
|
|
private antennasFetched: boolean;
|
2023-08-16 03:51:28 -05:00
|
|
|
private antennas: MiAntenna[];
|
2022-09-17 13:27:08 -05:00
|
|
|
|
|
|
|
constructor(
|
2023-10-03 06:26:11 -05:00
|
|
|
@Inject(DI.redisForTimelines)
|
|
|
|
private redisForTimelines: Redis.Redis,
|
2023-04-02 22:11:16 -05:00
|
|
|
|
2023-04-09 03:09:27 -05:00
|
|
|
@Inject(DI.redisForSub)
|
|
|
|
private redisForSub: Redis.Redis,
|
2022-09-17 13:27:08 -05:00
|
|
|
|
|
|
|
@Inject(DI.antennasRepository)
|
|
|
|
private antennasRepository: AntennasRepository,
|
|
|
|
|
2023-10-03 06:26:11 -05:00
|
|
|
@Inject(DI.userListMembershipsRepository)
|
|
|
|
private userListMembershipsRepository: UserListMembershipsRepository,
|
2022-09-17 13:27:08 -05:00
|
|
|
|
|
|
|
private utilityService: UtilityService,
|
2023-02-03 19:02:03 -06:00
|
|
|
private globalEventService: GlobalEventService,
|
2023-11-25 19:02:22 -06:00
|
|
|
private fanoutTimelineService: FanoutTimelineService,
|
2022-09-17 13:27:08 -05:00
|
|
|
) {
|
2022-09-18 13:11:50 -05:00
|
|
|
this.antennasFetched = false;
|
|
|
|
this.antennas = [];
|
2022-09-17 13:27:08 -05:00
|
|
|
|
2023-04-09 03:09:27 -05:00
|
|
|
this.redisForSub.on('message', this.onRedisMessage);
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|
|
|
|
|
2022-12-04 00:03:09 -06:00
|
|
|
@bindThis
|
2022-09-23 16:45:44 -05:00
|
|
|
private async onRedisMessage(_: string, data: string): Promise<void> {
|
2022-09-17 13:27:08 -05:00
|
|
|
const obj = JSON.parse(data);
|
|
|
|
|
|
|
|
if (obj.channel === 'internal') {
|
2023-09-28 21:29:54 -05:00
|
|
|
const { type, body } = obj.message as GlobalEvents['internal']['payload'];
|
2022-09-17 13:27:08 -05:00
|
|
|
switch (type) {
|
|
|
|
case 'antennaCreated':
|
2024-01-22 02:44:03 -06:00
|
|
|
this.antennas.push({ // TODO: このあたりのデシリアライズ処理は各modelファイル内に関数としてexportしたい
|
2023-01-24 20:18:16 -06:00
|
|
|
...body,
|
2023-03-20 06:12:38 -05:00
|
|
|
lastUsedAt: new Date(body.lastUsedAt),
|
2024-01-22 02:44:03 -06:00
|
|
|
user: null, // joinなカラムは通常取ってこないので
|
|
|
|
userList: null, // joinなカラムは通常取ってこないので
|
2023-01-24 20:18:16 -06:00
|
|
|
});
|
2022-09-17 13:27:08 -05:00
|
|
|
break;
|
2023-11-21 00:32:34 -06:00
|
|
|
case 'antennaUpdated': {
|
|
|
|
const idx = this.antennas.findIndex(a => a.id === body.id);
|
|
|
|
if (idx >= 0) {
|
2024-01-22 02:44:03 -06:00
|
|
|
this.antennas[idx] = { // TODO: このあたりのデシリアライズ処理は各modelファイル内に関数としてexportしたい
|
2023-11-21 00:32:34 -06:00
|
|
|
...body,
|
|
|
|
lastUsedAt: new Date(body.lastUsedAt),
|
2024-01-22 02:44:03 -06:00
|
|
|
user: null, // joinなカラムは通常取ってこないので
|
|
|
|
userList: null, // joinなカラムは通常取ってこないので
|
2023-11-21 00:32:34 -06:00
|
|
|
};
|
|
|
|
} else {
|
|
|
|
// サーバ起動時にactiveじゃなかった場合、リストに持っていないので追加する必要あり
|
2024-01-22 02:44:03 -06:00
|
|
|
this.antennas.push({ // TODO: このあたりのデシリアライズ処理は各modelファイル内に関数としてexportしたい
|
2023-11-21 00:32:34 -06:00
|
|
|
...body,
|
|
|
|
lastUsedAt: new Date(body.lastUsedAt),
|
2024-01-22 02:44:03 -06:00
|
|
|
user: null, // joinなカラムは通常取ってこないので
|
|
|
|
userList: null, // joinなカラムは通常取ってこないので
|
2023-11-21 00:32:34 -06:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2022-09-17 13:27:08 -05:00
|
|
|
break;
|
|
|
|
case 'antennaDeleted':
|
2022-09-18 13:11:50 -05:00
|
|
|
this.antennas = this.antennas.filter(a => a.id !== body.id);
|
2022-09-17 13:27:08 -05:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-04 00:03:09 -06:00
|
|
|
@bindThis
|
2024-03-20 17:51:01 -05:00
|
|
|
public async addNoteToAntennas(note: MiNote, noteUser: { id: MiUser['id']; username: string; host: string | null; isBot: boolean; }): Promise<void> {
|
2023-04-11 20:07:14 -05:00
|
|
|
const antennas = await this.getAntennas();
|
|
|
|
const antennasWithMatchResult = await Promise.all(antennas.map(antenna => this.checkHitAntenna(antenna, note, noteUser).then(hit => [antenna, hit] as const)));
|
|
|
|
const matchedAntennas = antennasWithMatchResult.filter(([, hit]) => hit).map(([antenna]) => antenna);
|
|
|
|
|
2023-10-03 06:26:11 -05:00
|
|
|
const redisPipeline = this.redisForTimelines.pipeline();
|
2023-04-11 20:07:14 -05:00
|
|
|
|
|
|
|
for (const antenna of matchedAntennas) {
|
2023-11-25 19:02:22 -06:00
|
|
|
this.fanoutTimelineService.push(`antennaTimeline:${antenna.id}`, note.id, 200, redisPipeline);
|
2023-04-11 20:07:14 -05:00
|
|
|
this.globalEventService.publishAntennaStream(antenna.id, 'note', note);
|
|
|
|
}
|
|
|
|
|
|
|
|
redisPipeline.exec();
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: フォローしているユーザーのノート、リストのユーザーのノート、グループのユーザーのノート指定はパフォーマンス上の理由で無効になっている
|
|
|
|
|
2022-12-04 00:03:09 -06:00
|
|
|
@bindThis
|
2024-03-20 17:51:01 -05:00
|
|
|
public async checkHitAntenna(antenna: MiAntenna, note: (MiNote | Packed<'Note'>), noteUser: { id: MiUser['id']; username: string; host: string | null; isBot: boolean; }): Promise<boolean> {
|
2022-09-17 13:27:08 -05:00
|
|
|
if (note.visibility === 'specified') return false;
|
2023-01-08 23:12:42 -06:00
|
|
|
if (note.visibility === 'followers') return false;
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2024-03-20 17:51:01 -05:00
|
|
|
if (antenna.excludeBots && noteUser.isBot) return false;
|
|
|
|
|
2023-10-16 01:06:00 -05:00
|
|
|
if (antenna.localOnly && noteUser.host != null) return false;
|
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (!antenna.withReplies && note.replyId != null) return false;
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (antenna.src === 'home') {
|
2023-01-08 23:12:42 -06:00
|
|
|
// TODO
|
2022-09-17 13:27:08 -05:00
|
|
|
} else if (antenna.src === 'list') {
|
2024-09-17 08:02:34 -05:00
|
|
|
if (antenna.userListId == null) return false;
|
|
|
|
const exists = await this.userListMembershipsRepository.exists({
|
|
|
|
where: {
|
|
|
|
userListId: antenna.userListId,
|
|
|
|
userId: note.userId,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (!exists) return false;
|
2022-09-17 13:27:08 -05:00
|
|
|
} else if (antenna.src === 'users') {
|
|
|
|
const accts = antenna.users.map(x => {
|
|
|
|
const { username, host } = Acct.parse(x);
|
|
|
|
return this.utilityService.getFullApAccount(username, host).toLowerCase();
|
|
|
|
});
|
2024-11-20 09:37:16 -06:00
|
|
|
const matchUser = this.utilityService.getFullApAccount(noteUser.username, noteUser.host).toLowerCase();
|
|
|
|
const matchWildcard = this.utilityService.getFullApAccount('*', noteUser.host).toLowerCase();
|
|
|
|
if (!accts.includes(matchUser) && !accts.includes(matchWildcard)) return false;
|
2023-09-22 02:52:43 -05:00
|
|
|
} else if (antenna.src === 'users_blacklist') {
|
|
|
|
const accts = antenna.users.map(x => {
|
|
|
|
const { username, host } = Acct.parse(x);
|
|
|
|
return this.utilityService.getFullApAccount(username, host).toLowerCase();
|
|
|
|
});
|
2024-11-20 09:37:16 -06:00
|
|
|
const matchUser = this.utilityService.getFullApAccount(noteUser.username, noteUser.host).toLowerCase();
|
|
|
|
const matchWildcard = this.utilityService.getFullApAccount('*', noteUser.host).toLowerCase();
|
|
|
|
if (accts.includes(matchUser) || accts.includes(matchWildcard)) return false;
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
const keywords = antenna.keywords
|
|
|
|
// Clean up
|
|
|
|
.map(xs => xs.filter(x => x !== ''))
|
|
|
|
.filter(xs => xs.length > 0);
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (keywords.length > 0) {
|
2023-02-28 05:20:23 -06:00
|
|
|
if (note.text == null && note.cw == null) return false;
|
|
|
|
|
|
|
|
const _text = (note.text ?? '') + '\n' + (note.cw ?? '');
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
const matched = keywords.some(and =>
|
|
|
|
and.every(keyword =>
|
|
|
|
antenna.caseSensitive
|
2023-02-28 05:20:23 -06:00
|
|
|
? _text.includes(keyword)
|
|
|
|
: _text.toLowerCase().includes(keyword.toLowerCase()),
|
2022-09-17 13:27:08 -05:00
|
|
|
));
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (!matched) return false;
|
|
|
|
}
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
const excludeKeywords = antenna.excludeKeywords
|
|
|
|
// Clean up
|
|
|
|
.map(xs => xs.filter(x => x !== ''))
|
|
|
|
.filter(xs => xs.length > 0);
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (excludeKeywords.length > 0) {
|
2023-02-28 05:20:23 -06:00
|
|
|
if (note.text == null && note.cw == null) return false;
|
|
|
|
|
|
|
|
const _text = (note.text ?? '') + '\n' + (note.cw ?? '');
|
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
const matched = excludeKeywords.some(and =>
|
|
|
|
and.every(keyword =>
|
|
|
|
antenna.caseSensitive
|
2023-02-28 05:20:23 -06:00
|
|
|
? _text.includes(keyword)
|
|
|
|
: _text.toLowerCase().includes(keyword.toLowerCase()),
|
2022-09-17 13:27:08 -05:00
|
|
|
));
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (matched) return false;
|
|
|
|
}
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
if (antenna.withFile) {
|
|
|
|
if (note.fileIds && note.fileIds.length === 0) return false;
|
|
|
|
}
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
// TODO: eval expression
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-17 13:27:08 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-12-04 00:03:09 -06:00
|
|
|
@bindThis
|
2022-09-17 13:27:08 -05:00
|
|
|
public async getAntennas() {
|
2022-09-18 13:11:50 -05:00
|
|
|
if (!this.antennasFetched) {
|
2023-03-20 06:12:38 -05:00
|
|
|
this.antennas = await this.antennasRepository.findBy({
|
|
|
|
isActive: true,
|
|
|
|
});
|
2022-09-18 13:11:50 -05:00
|
|
|
this.antennasFetched = true;
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|
2023-07-07 17:08:16 -05:00
|
|
|
|
2022-09-18 13:11:50 -05:00
|
|
|
return this.antennas;
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|
2023-05-28 23:21:26 -05:00
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public dispose(): void {
|
|
|
|
this.redisForSub.off('message', this.onRedisMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
|
|
this.dispose();
|
|
|
|
}
|
2022-09-17 13:27:08 -05:00
|
|
|
}
|