2022-09-17 20:27:08 +02:00
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
|
|
|
import { DataSource, In, IsNull } from 'typeorm';
|
2023-04-06 04:14:43 +02:00
|
|
|
|
import Redis from 'ioredis';
|
2022-09-17 20:27:08 +02:00
|
|
|
|
import { DI } from '@/di-symbols.js';
|
|
|
|
|
import { IdService } from '@/core/IdService.js';
|
2023-01-22 15:53:24 +01:00
|
|
|
|
import { EmojiEntityService } from '@/core/entities/EmojiEntityService.js';
|
|
|
|
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
2022-09-17 20:27:08 +02:00
|
|
|
|
import type { DriveFile } from '@/models/entities/DriveFile.js';
|
|
|
|
|
import type { Emoji } from '@/models/entities/Emoji.js';
|
2023-04-06 04:14:43 +02:00
|
|
|
|
import type { EmojisRepository } from '@/models/index.js';
|
2022-12-04 09:05:32 +01:00
|
|
|
|
import { bindThis } from '@/decorators.js';
|
2023-04-06 04:14:43 +02:00
|
|
|
|
import { MemoryKVCache, RedisSingleCache } from '@/misc/cache.js';
|
2023-01-26 07:48:12 +01:00
|
|
|
|
import { UtilityService } from '@/core/UtilityService.js';
|
|
|
|
|
import type { Config } from '@/config.js';
|
|
|
|
|
import { query } from '@/misc/prelude/url.js';
|
2022-09-17 20:27:08 +02:00
|
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
|
export class CustomEmojiService {
|
2023-04-04 08:56:47 +02:00
|
|
|
|
private cache: MemoryKVCache<Emoji | null>;
|
2023-04-06 04:14:43 +02:00
|
|
|
|
public localEmojisCache: RedisSingleCache<Map<string, Emoji>>;
|
2023-01-26 07:48:12 +01:00
|
|
|
|
|
2022-09-17 20:27:08 +02:00
|
|
|
|
constructor(
|
2023-04-06 04:14:43 +02:00
|
|
|
|
@Inject(DI.redis)
|
|
|
|
|
private redisClient: Redis.Redis,
|
|
|
|
|
|
2023-01-26 07:48:12 +01:00
|
|
|
|
@Inject(DI.config)
|
|
|
|
|
private config: Config,
|
|
|
|
|
|
2022-09-17 20:27:08 +02:00
|
|
|
|
@Inject(DI.db)
|
|
|
|
|
private db: DataSource,
|
|
|
|
|
|
|
|
|
|
@Inject(DI.emojisRepository)
|
|
|
|
|
private emojisRepository: EmojisRepository,
|
|
|
|
|
|
2023-01-26 07:48:12 +01:00
|
|
|
|
private utilityService: UtilityService,
|
2022-09-17 20:27:08 +02:00
|
|
|
|
private idService: IdService,
|
2023-01-22 15:53:24 +01:00
|
|
|
|
private emojiEntityService: EmojiEntityService,
|
|
|
|
|
private globalEventService: GlobalEventService,
|
2022-09-17 20:27:08 +02:00
|
|
|
|
) {
|
2023-04-04 08:56:47 +02:00
|
|
|
|
this.cache = new MemoryKVCache<Emoji | null>(1000 * 60 * 60 * 12);
|
2023-04-06 04:14:43 +02:00
|
|
|
|
|
|
|
|
|
this.localEmojisCache = new RedisSingleCache<Map<string, Emoji>>(this.redisClient, 'localEmojis', {
|
|
|
|
|
lifetime: 1000 * 60 * 30, // 30m
|
|
|
|
|
memoryCacheLifetime: 1000 * 60 * 3, // 3m
|
|
|
|
|
fetcher: () => this.emojisRepository.find({ where: { host: IsNull() } }).then(emojis => new Map(emojis.map(emoji => [emoji.name, emoji]))),
|
|
|
|
|
toRedisConverter: (value) => JSON.stringify(value.values()),
|
2023-04-09 02:52:19 +02:00
|
|
|
|
fromRedisConverter: (value) => {
|
|
|
|
|
if (!Array.isArray(JSON.parse(value))) return undefined;
|
|
|
|
|
return new Map(JSON.parse(value).map((x: Emoji) => [x.name, x]));
|
|
|
|
|
}, // TODO: Date型の変換
|
2023-04-06 04:14:43 +02:00
|
|
|
|
});
|
2022-09-17 20:27:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2022-12-04 07:03:09 +01:00
|
|
|
|
@bindThis
|
2022-09-17 20:27:08 +02:00
|
|
|
|
public async add(data: {
|
|
|
|
|
driveFile: DriveFile;
|
|
|
|
|
name: string;
|
|
|
|
|
category: string | null;
|
|
|
|
|
aliases: string[];
|
|
|
|
|
host: string | null;
|
2023-03-16 07:08:48 +01:00
|
|
|
|
license: string | null;
|
2022-09-17 20:27:08 +02:00
|
|
|
|
}): Promise<Emoji> {
|
|
|
|
|
const emoji = await this.emojisRepository.insert({
|
|
|
|
|
id: this.idService.genId(),
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
name: data.name,
|
|
|
|
|
category: data.category,
|
|
|
|
|
host: data.host,
|
|
|
|
|
aliases: data.aliases,
|
|
|
|
|
originalUrl: data.driveFile.url,
|
|
|
|
|
publicUrl: data.driveFile.webpublicUrl ?? data.driveFile.url,
|
|
|
|
|
type: data.driveFile.webpublicType ?? data.driveFile.type,
|
2023-03-16 07:08:48 +01:00
|
|
|
|
license: data.license,
|
2022-09-17 20:27:08 +02:00
|
|
|
|
}).then(x => this.emojisRepository.findOneByOrFail(x.identifiers[0]));
|
|
|
|
|
|
2023-01-26 06:29:28 +01:00
|
|
|
|
if (data.host == null) {
|
2023-04-06 04:14:43 +02:00
|
|
|
|
this.localEmojisCache.refresh();
|
2022-09-17 20:27:08 +02:00
|
|
|
|
|
2023-01-26 06:29:28 +01:00
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiAdded', {
|
2023-02-17 07:36:36 +01:00
|
|
|
|
emoji: await this.emojiEntityService.packDetailed(emoji.id),
|
2023-01-26 06:29:28 +01:00
|
|
|
|
});
|
|
|
|
|
}
|
2023-01-22 15:53:24 +01:00
|
|
|
|
|
2022-09-17 20:27:08 +02:00
|
|
|
|
return emoji;
|
|
|
|
|
}
|
2023-01-26 07:48:12 +01:00
|
|
|
|
|
2023-04-06 04:14:43 +02:00
|
|
|
|
@bindThis
|
|
|
|
|
public async update(id: Emoji['id'], data: {
|
|
|
|
|
name?: string;
|
|
|
|
|
category?: string | null;
|
|
|
|
|
aliases?: string[];
|
|
|
|
|
license?: string | null;
|
|
|
|
|
}): Promise<void> {
|
|
|
|
|
const emoji = await this.emojisRepository.findOneByOrFail({ id: id });
|
|
|
|
|
const sameNameEmoji = await this.emojisRepository.findOneBy({ name: data.name, host: IsNull() });
|
|
|
|
|
if (sameNameEmoji != null && sameNameEmoji.id !== id) throw new Error('name already exists');
|
|
|
|
|
|
|
|
|
|
await this.emojisRepository.update(emoji.id, {
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
name: data.name,
|
|
|
|
|
category: data.category,
|
|
|
|
|
aliases: data.aliases,
|
|
|
|
|
license: data.license,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
const updated = await this.emojiEntityService.packDetailed(emoji.id);
|
|
|
|
|
|
|
|
|
|
if (emoji.name === data.name) {
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiUpdated', {
|
|
|
|
|
emojis: [updated],
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiDeleted', {
|
|
|
|
|
emojis: [await this.emojiEntityService.packDetailed(emoji)],
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiAdded', {
|
|
|
|
|
emoji: updated,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async addAliasesBulk(ids: Emoji['id'][], aliases: string[]) {
|
|
|
|
|
const emojis = await this.emojisRepository.findBy({
|
|
|
|
|
id: In(ids),
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
for (const emoji of emojis) {
|
|
|
|
|
await this.emojisRepository.update(emoji.id, {
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
aliases: [...new Set(emoji.aliases.concat(aliases))],
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiUpdated', {
|
|
|
|
|
emojis: await this.emojiEntityService.packDetailedMany(ids),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async setAliasesBulk(ids: Emoji['id'][], aliases: string[]) {
|
|
|
|
|
await this.emojisRepository.update({
|
|
|
|
|
id: In(ids),
|
|
|
|
|
}, {
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
aliases: aliases,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiUpdated', {
|
|
|
|
|
emojis: await this.emojiEntityService.packDetailedMany(ids),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async removeAliasesBulk(ids: Emoji['id'][], aliases: string[]) {
|
|
|
|
|
const emojis = await this.emojisRepository.findBy({
|
|
|
|
|
id: In(ids),
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
for (const emoji of emojis) {
|
|
|
|
|
await this.emojisRepository.update(emoji.id, {
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
aliases: emoji.aliases.filter(x => !aliases.includes(x)),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiUpdated', {
|
|
|
|
|
emojis: await this.emojiEntityService.packDetailedMany(ids),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async setCategoryBulk(ids: Emoji['id'][], category: string | null) {
|
|
|
|
|
await this.emojisRepository.update({
|
|
|
|
|
id: In(ids),
|
|
|
|
|
}, {
|
|
|
|
|
updatedAt: new Date(),
|
|
|
|
|
category: category,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiUpdated', {
|
|
|
|
|
emojis: await this.emojiEntityService.packDetailedMany(ids),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async delete(id: Emoji['id']) {
|
|
|
|
|
const emoji = await this.emojisRepository.findOneByOrFail({ id: id });
|
|
|
|
|
|
|
|
|
|
await this.emojisRepository.delete(emoji.id);
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiDeleted', {
|
|
|
|
|
emojis: [await this.emojiEntityService.packDetailed(emoji)],
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
|
|
|
|
public async deleteBulk(ids: Emoji['id'][]) {
|
|
|
|
|
const emojis = await this.emojisRepository.findBy({
|
|
|
|
|
id: In(ids),
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
for (const emoji of emojis) {
|
|
|
|
|
await this.emojisRepository.delete(emoji.id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.localEmojisCache.refresh();
|
|
|
|
|
|
|
|
|
|
this.globalEventService.publishBroadcastStream('emojiDeleted', {
|
|
|
|
|
emojis: await this.emojiEntityService.packDetailedMany(emojis),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
2023-01-26 07:48:12 +01:00
|
|
|
|
@bindThis
|
|
|
|
|
private normalizeHost(src: string | undefined, noteUserHost: string | null): string | null {
|
|
|
|
|
// クエリに使うホスト
|
|
|
|
|
let host = src === '.' ? null // .はローカルホスト (ここがマッチするのはリアクションのみ)
|
|
|
|
|
: src === undefined ? noteUserHost // ノートなどでホスト省略表記の場合はローカルホスト (ここがリアクションにマッチすることはない)
|
|
|
|
|
: this.utilityService.isSelfHost(src) ? null // 自ホスト指定
|
|
|
|
|
: (src || noteUserHost); // 指定されたホスト || ノートなどの所有者のホスト (こっちがリアクションにマッチすることはない)
|
|
|
|
|
|
|
|
|
|
host = this.utilityService.toPunyNullable(host);
|
|
|
|
|
|
|
|
|
|
return host;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@bindThis
|
2023-04-06 04:14:43 +02:00
|
|
|
|
public parseEmojiStr(emojiName: string, noteUserHost: string | null) {
|
2023-01-26 07:48:12 +01:00
|
|
|
|
const match = emojiName.match(/^(\w+)(?:@([\w.-]+))?$/);
|
|
|
|
|
if (!match) return { name: null, host: null };
|
|
|
|
|
|
|
|
|
|
const name = match[1];
|
|
|
|
|
|
|
|
|
|
// ホスト正規化
|
|
|
|
|
const host = this.utilityService.toPunyNullable(this.normalizeHost(match[2], noteUserHost));
|
|
|
|
|
|
|
|
|
|
return { name, host };
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 添付用(リモート)カスタム絵文字URLを解決する
|
|
|
|
|
* @param emojiName ノートやユーザープロフィールに添付された、またはリアクションのカスタム絵文字名 (:は含めない, リアクションでローカルホストの場合は@.を付ける (これはdecodeReactionで可能))
|
|
|
|
|
* @param noteUserHost ノートやユーザープロフィールの所有者のホスト
|
|
|
|
|
* @returns URL, nullは未マッチを意味する
|
|
|
|
|
*/
|
|
|
|
|
@bindThis
|
|
|
|
|
public async populateEmoji(emojiName: string, noteUserHost: string | null): Promise<string | null> {
|
|
|
|
|
const { name, host } = this.parseEmojiStr(emojiName, noteUserHost);
|
|
|
|
|
if (name == null) return null;
|
|
|
|
|
if (host == null) return null;
|
|
|
|
|
|
|
|
|
|
const queryOrNull = async () => (await this.emojisRepository.findOneBy({
|
|
|
|
|
name,
|
|
|
|
|
host: host ?? IsNull(),
|
|
|
|
|
})) ?? null;
|
|
|
|
|
|
|
|
|
|
const emoji = await this.cache.fetch(`${name} ${host}`, queryOrNull);
|
|
|
|
|
|
|
|
|
|
if (emoji == null) return null;
|
|
|
|
|
|
|
|
|
|
const isLocal = emoji.host == null;
|
|
|
|
|
const emojiUrl = emoji.publicUrl || emoji.originalUrl; // || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
|
2023-01-26 10:44:43 +01:00
|
|
|
|
const url = isLocal
|
|
|
|
|
? emojiUrl
|
|
|
|
|
: this.config.proxyRemoteFiles
|
2023-02-04 05:38:51 +01:00
|
|
|
|
? `${this.config.mediaProxy}/emoji.webp?${query({ url: emojiUrl })}`
|
2023-01-26 10:44:43 +01:00
|
|
|
|
: emojiUrl;
|
2023-01-26 07:48:12 +01:00
|
|
|
|
|
|
|
|
|
return url;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 複数の添付用(リモート)カスタム絵文字URLを解決する (キャシュ付き, 存在しないものは結果から除外される)
|
|
|
|
|
*/
|
|
|
|
|
@bindThis
|
|
|
|
|
public async populateEmojis(emojiNames: string[], noteUserHost: string | null): Promise<Record<string, string>> {
|
|
|
|
|
const emojis = await Promise.all(emojiNames.map(x => this.populateEmoji(x, noteUserHost)));
|
|
|
|
|
const res = {} as any;
|
|
|
|
|
for (let i = 0; i < emojiNames.length; i++) {
|
|
|
|
|
if (emojis[i] != null) {
|
|
|
|
|
res[emojiNames[i]] = emojis[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return res;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 与えられた絵文字のリストをデータベースから取得し、キャッシュに追加します
|
|
|
|
|
*/
|
|
|
|
|
@bindThis
|
|
|
|
|
public async prefetchEmojis(emojis: { name: string; host: string | null; }[]): Promise<void> {
|
|
|
|
|
const notCachedEmojis = emojis.filter(emoji => this.cache.get(`${emoji.name} ${emoji.host}`) == null);
|
|
|
|
|
const emojisQuery: any[] = [];
|
|
|
|
|
const hosts = new Set(notCachedEmojis.map(e => e.host));
|
|
|
|
|
for (const host of hosts) {
|
|
|
|
|
if (host == null) continue;
|
|
|
|
|
emojisQuery.push({
|
|
|
|
|
name: In(notCachedEmojis.filter(e => e.host === host).map(e => e.name)),
|
|
|
|
|
host: host,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
const _emojis = emojisQuery.length > 0 ? await this.emojisRepository.find({
|
|
|
|
|
where: emojisQuery,
|
|
|
|
|
select: ['name', 'host', 'originalUrl', 'publicUrl'],
|
|
|
|
|
}) : [];
|
|
|
|
|
for (const emoji of _emojis) {
|
|
|
|
|
this.cache.set(`${emoji.name} ${emoji.host}`, emoji);
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-09-17 20:27:08 +02:00
|
|
|
|
}
|