mirror of
https://codeberg.org/yeentown/barkey.git
synced 2025-10-23 17:54:52 +00:00
217 lines
6.4 KiB
TypeScript
217 lines
6.4 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
import { DataSource, EntityManager } from 'typeorm';
|
|
import * as Redis from 'ioredis';
|
|
import { DI } from '@/di-symbols.js';
|
|
import { MiMeta } from '@/models/Meta.js';
|
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
import type { GlobalEvents } from '@/core/GlobalEventService.js';
|
|
import { FeaturedService } from '@/core/FeaturedService.js';
|
|
import { MiInstance } from '@/models/Instance.js';
|
|
import { diffArrays } from '@/misc/diff-arrays.js';
|
|
import type { MetasRepository } from '@/models/_.js';
|
|
import type { OnApplicationShutdown } from '@nestjs/common';
|
|
|
|
@Injectable()
|
|
export class MetaService implements OnApplicationShutdown {
|
|
private cache: MiMeta | undefined;
|
|
private intervalId: NodeJS.Timeout;
|
|
|
|
constructor(
|
|
@Inject(DI.redisForSub)
|
|
private redisForSub: Redis.Redis,
|
|
|
|
@Inject(DI.db)
|
|
private db: DataSource,
|
|
|
|
@Inject(DI.metasRepository)
|
|
private readonly metasRepository: MetasRepository,
|
|
|
|
private featuredService: FeaturedService,
|
|
private globalEventService: GlobalEventService,
|
|
) {
|
|
//this.onMessage = this.onMessage.bind(this);
|
|
|
|
if (process.env.NODE_ENV !== 'test') {
|
|
this.intervalId = setInterval(() => {
|
|
this.fetch(true).then(meta => {
|
|
// fetch内でもセットしてるけど仕様変更の可能性もあるため一応
|
|
this.cache = meta;
|
|
});
|
|
}, 1000 * 60 * 5);
|
|
}
|
|
|
|
this.redisForSub.on('message', this.onMessage);
|
|
}
|
|
|
|
@bindThis
|
|
private async onMessage(_: string, data: string): Promise<void> {
|
|
const obj = JSON.parse(data);
|
|
|
|
if (obj.channel === 'internal') {
|
|
const { type, body } = obj.message as GlobalEvents['internal']['payload'];
|
|
switch (type) {
|
|
case 'metaUpdated': {
|
|
this.cache = { // TODO: このあたりのデシリアライズ処理は各modelファイル内に関数としてexportしたい
|
|
...(body.after),
|
|
rootUser: null, // joinなカラムは通常取ってこないので
|
|
};
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
@bindThis
|
|
public async fetch(noCache = false): Promise<MiMeta> {
|
|
if (!noCache && this.cache) return this.cache;
|
|
|
|
// 過去のバグでレコードが複数出来てしまっている可能性があるので新しいIDを優先する
|
|
let meta = await this.metasRepository.createQueryBuilder('meta')
|
|
.select()
|
|
.orderBy({
|
|
id: 'DESC',
|
|
})
|
|
.limit(1)
|
|
.getOne();
|
|
|
|
if (!meta) {
|
|
await this.metasRepository.createQueryBuilder('meta')
|
|
.insert()
|
|
.values({
|
|
id: 'x',
|
|
})
|
|
.orIgnore()
|
|
.execute();
|
|
|
|
meta = await this.metasRepository.createQueryBuilder('meta')
|
|
.select()
|
|
.orderBy({
|
|
id: 'DESC',
|
|
})
|
|
.limit(1)
|
|
.getOneOrFail();
|
|
}
|
|
|
|
this.cache = meta;
|
|
return meta;
|
|
}
|
|
|
|
@bindThis
|
|
public async update(data: Partial<MiMeta>): Promise<MiMeta> {
|
|
let before: MiMeta | undefined;
|
|
|
|
const updated = await this.db.transaction(async transactionalEntityManager => {
|
|
const metas: (MiMeta | undefined)[] = await transactionalEntityManager.find(MiMeta, {
|
|
order: {
|
|
id: 'DESC',
|
|
},
|
|
});
|
|
|
|
before = metas[0];
|
|
|
|
if (before) {
|
|
await transactionalEntityManager.update(MiMeta, before.id, data);
|
|
} else {
|
|
await transactionalEntityManager.save(MiMeta, {
|
|
...data,
|
|
id: 'x',
|
|
});
|
|
}
|
|
|
|
const afters = await transactionalEntityManager.find(MiMeta, {
|
|
order: {
|
|
id: 'DESC',
|
|
},
|
|
});
|
|
|
|
// Propagate changes to blockedHosts, silencedHosts, mediaSilencedHosts, federationInstances, and bubbleInstances to the relevant instance rows
|
|
// Do this inside the transaction to avoid potential race condition (when an instance gets registered while we're updating).
|
|
await this.persistBlocks(transactionalEntityManager, before ?? {}, afters[0]);
|
|
|
|
return afters[0];
|
|
});
|
|
|
|
if (data.hiddenTags) {
|
|
process.nextTick(() => {
|
|
const hiddenTags = new Set<string>(data.hiddenTags);
|
|
if (before) {
|
|
for (const previousHiddenTag of before.hiddenTags) {
|
|
hiddenTags.delete(previousHiddenTag);
|
|
}
|
|
}
|
|
|
|
for (const hiddenTag of hiddenTags) {
|
|
this.featuredService.removeHashtagsFromRanking(hiddenTag);
|
|
}
|
|
});
|
|
}
|
|
|
|
this.globalEventService.publishInternalEvent('metaUpdated', { before, after: updated });
|
|
|
|
return updated;
|
|
}
|
|
|
|
@bindThis
|
|
public dispose(): void {
|
|
clearInterval(this.intervalId);
|
|
this.redisForSub.off('message', this.onMessage);
|
|
}
|
|
|
|
@bindThis
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
this.dispose();
|
|
}
|
|
|
|
private async persistBlocks(tem: EntityManager, before: Partial<MiMeta>, after: Partial<MiMeta>): Promise<void> {
|
|
await this.persistBlock(tem, before.blockedHosts, after.blockedHosts, 'isBlocked');
|
|
await this.persistBlock(tem, before.silencedHosts, after.silencedHosts, 'isSilenced');
|
|
await this.persistBlock(tem, before.mediaSilencedHosts, after.mediaSilencedHosts, 'isMediaSilenced');
|
|
await this.persistBlock(tem, before.federationHosts, after.federationHosts, 'isAllowListed');
|
|
await this.persistBlock(tem, before.bubbleInstances, after.bubbleInstances, 'isBubbled');
|
|
}
|
|
|
|
private async persistBlock(tem: EntityManager, before: string[] | undefined, after: string[] | undefined, field: keyof MiInstance): Promise<void> {
|
|
const { added, removed } = diffArrays(before, after);
|
|
|
|
if (removed.length > 0) {
|
|
await this.updateInstancesByHost(tem, field, false, removed);
|
|
}
|
|
|
|
if (added.length > 0) {
|
|
await this.updateInstancesByHost(tem, field, true, added);
|
|
}
|
|
}
|
|
|
|
private async updateInstancesByHost(tem: EntityManager, field: keyof MiInstance, value: boolean, hosts: string[]): Promise<void> {
|
|
// Use non-array queries when possible, as they are indexed and can be much faster.
|
|
if (hosts.length === 1) {
|
|
const pattern = genHostPattern(hosts[0]);
|
|
await tem
|
|
.createQueryBuilder(MiInstance, 'instance')
|
|
.update()
|
|
.set({ [field]: value })
|
|
.where('(lower(reverse("host")) || \'.\') LIKE :pattern', { pattern })
|
|
.execute();
|
|
} else if (hosts.length > 1) {
|
|
const patterns = hosts.map(host => genHostPattern(host));
|
|
await tem
|
|
.createQueryBuilder(MiInstance, 'instance')
|
|
.update()
|
|
.set({ [field]: value })
|
|
.where('(lower(reverse("host")) || \'.\') LIKE ANY (:patterns)', { patterns })
|
|
.execute();
|
|
}
|
|
}
|
|
}
|
|
|
|
function genHostPattern(host: string): string {
|
|
return host.toLowerCase().split('').reverse().join('') + '.%';
|
|
}
|