mirror of
https://codeberg.org/yeentown/barkey.git
synced 2025-10-23 17:54:52 +00:00
213 lines
6.5 KiB
TypeScript
213 lines
6.5 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 { 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,
|
|
|
|
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;
|
|
|
|
return await this.db.transaction(async transactionalEntityManager => {
|
|
// 過去のバグでレコードが複数出来てしまっている可能性があるので新しいIDを優先する
|
|
const metas = await transactionalEntityManager.find(MiMeta, {
|
|
order: {
|
|
id: 'DESC',
|
|
},
|
|
});
|
|
|
|
const meta = metas[0];
|
|
|
|
if (meta) {
|
|
this.cache = meta;
|
|
return meta;
|
|
} else {
|
|
// metaが空のときfetchMetaが同時に呼ばれるとここが同時に呼ばれてしまうことがあるのでフェイルセーフなupsertを使う
|
|
const saved = await transactionalEntityManager
|
|
.upsert(
|
|
MiMeta,
|
|
{
|
|
id: 'x',
|
|
},
|
|
['id'],
|
|
)
|
|
.then((x) => transactionalEntityManager.findOneByOrFail(MiMeta, x.identifiers[0]));
|
|
|
|
this.cache = saved;
|
|
return saved;
|
|
}
|
|
});
|
|
}
|
|
|
|
@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('') + '.%';
|
|
}
|