2021-03-21 22:41:33 -05:00
|
|
|
|
import { In } from 'typeorm';
|
2022-02-26 20:07:39 -06:00
|
|
|
|
import { Emojis } from '@/models/index.js';
|
|
|
|
|
import { Emoji } from '@/models/entities/emoji.js';
|
|
|
|
|
import { Note } from '@/models/entities/note.js';
|
|
|
|
|
import { Cache } from './cache.js';
|
|
|
|
|
import { isSelfHost, toPunyNullable } from './convert-host.js';
|
|
|
|
|
import { decodeReaction } from './reaction-lib.js';
|
|
|
|
|
import config from '@/config/index.js';
|
|
|
|
|
import { query } from '@/prelude/url.js';
|
2021-03-21 10:44:38 -05:00
|
|
|
|
|
2021-03-21 22:46:46 -05:00
|
|
|
|
const cache = new Cache<Emoji | null>(1000 * 60 * 60 * 12);
|
2021-03-21 10:44:38 -05:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 添付用絵文字情報
|
|
|
|
|
*/
|
|
|
|
|
type PopulatedEmoji = {
|
|
|
|
|
name: string;
|
|
|
|
|
url: string;
|
|
|
|
|
};
|
|
|
|
|
|
2021-03-21 22:36:57 -05:00
|
|
|
|
function normalizeHost(src: string | undefined, noteUserHost: string | null): string | null {
|
|
|
|
|
// クエリに使うホスト
|
|
|
|
|
let host = src === '.' ? null // .はローカルホスト (ここがマッチするのはリアクションのみ)
|
|
|
|
|
: src === undefined ? noteUserHost // ノートなどでホスト省略表記の場合はローカルホスト (ここがリアクションにマッチすることはない)
|
|
|
|
|
: isSelfHost(src) ? null // 自ホスト指定
|
|
|
|
|
: (src || noteUserHost); // 指定されたホスト || ノートなどの所有者のホスト (こっちがリアクションにマッチすることはない)
|
|
|
|
|
|
|
|
|
|
host = toPunyNullable(host);
|
|
|
|
|
|
|
|
|
|
return host;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function parseEmojiStr(emojiName: string, noteUserHost: string | null) {
|
|
|
|
|
const match = emojiName.match(/^(\w+)(?:@([\w.-]+))?$/);
|
|
|
|
|
if (!match) return { name: null, host: null };
|
|
|
|
|
|
|
|
|
|
const name = match[1];
|
|
|
|
|
|
|
|
|
|
// ホスト正規化
|
|
|
|
|
const host = toPunyNullable(normalizeHost(match[2], noteUserHost));
|
|
|
|
|
|
|
|
|
|
return { name, host };
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-21 10:44:38 -05:00
|
|
|
|
/**
|
|
|
|
|
* 添付用絵文字情報を解決する
|
|
|
|
|
* @param emojiName ノートやユーザープロフィールに添付された、またはリアクションのカスタム絵文字名 (:は含めない, リアクションでローカルホストの場合は@.を付ける (これはdecodeReactionで可能))
|
2021-03-21 10:45:14 -05:00
|
|
|
|
* @param noteUserHost ノートやユーザープロフィールの所有者のホスト
|
2021-03-21 10:44:38 -05:00
|
|
|
|
* @returns 絵文字情報, nullは未マッチを意味する
|
|
|
|
|
*/
|
|
|
|
|
export async function populateEmoji(emojiName: string, noteUserHost: string | null): Promise<PopulatedEmoji | null> {
|
2021-03-21 22:36:57 -05:00
|
|
|
|
const { name, host } = parseEmojiStr(emojiName, noteUserHost);
|
|
|
|
|
if (name == null) return null;
|
2021-03-21 10:44:38 -05:00
|
|
|
|
|
|
|
|
|
const queryOrNull = async () => (await Emojis.findOne({
|
|
|
|
|
name,
|
2021-11-13 04:10:14 -06:00
|
|
|
|
host,
|
2021-03-21 10:44:38 -05:00
|
|
|
|
})) || null;
|
|
|
|
|
|
|
|
|
|
const emoji = await cache.fetch(`${name} ${host}`, queryOrNull);
|
|
|
|
|
|
|
|
|
|
if (emoji == null) return null;
|
|
|
|
|
|
2021-05-29 20:05:12 -05:00
|
|
|
|
const isLocal = emoji.host == null;
|
2022-01-21 03:47:02 -06:00
|
|
|
|
const emojiUrl = emoji.publicUrl || emoji.originalUrl; // || emoji.originalUrl してるのは後方互換性のため
|
|
|
|
|
const url = isLocal ? emojiUrl : `${config.url}/proxy/image.png?${query({ url: emojiUrl })}`;
|
2021-05-27 08:40:48 -05:00
|
|
|
|
|
2021-03-21 10:44:38 -05:00
|
|
|
|
return {
|
|
|
|
|
name: emojiName,
|
2021-05-27 08:40:48 -05:00
|
|
|
|
url,
|
2021-03-21 10:44:38 -05:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 複数の添付用絵文字情報を解決する (キャシュ付き, 存在しないものは結果から除外される)
|
|
|
|
|
*/
|
|
|
|
|
export async function populateEmojis(emojiNames: string[], noteUserHost: string | null): Promise<PopulatedEmoji[]> {
|
|
|
|
|
const emojis = await Promise.all(emojiNames.map(x => populateEmoji(x, noteUserHost)));
|
|
|
|
|
return emojis.filter((x): x is PopulatedEmoji => x != null);
|
|
|
|
|
}
|
|
|
|
|
|
2021-03-21 22:41:33 -05:00
|
|
|
|
export function aggregateNoteEmojis(notes: Note[]) {
|
|
|
|
|
let emojis: { name: string | null; host: string | null; }[] = [];
|
|
|
|
|
for (const note of notes) {
|
|
|
|
|
emojis = emojis.concat(note.emojis
|
|
|
|
|
.map(e => parseEmojiStr(e, note.userHost)));
|
|
|
|
|
if (note.renote) {
|
|
|
|
|
emojis = emojis.concat(note.renote.emojis
|
|
|
|
|
.map(e => parseEmojiStr(e, note.renote!.userHost)));
|
|
|
|
|
if (note.renote.user) {
|
|
|
|
|
emojis = emojis.concat(note.renote.user.emojis
|
|
|
|
|
.map(e => parseEmojiStr(e, note.renote!.userHost)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
const customReactions = Object.keys(note.reactions).map(x => decodeReaction(x)).filter(x => x.name != null) as typeof emojis;
|
|
|
|
|
emojis = emojis.concat(customReactions);
|
|
|
|
|
if (note.user) {
|
|
|
|
|
emojis = emojis.concat(note.user.emojis
|
|
|
|
|
.map(e => parseEmojiStr(e, note.userHost)));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return emojis.filter(x => x.name != null) as { name: string; host: string | null; }[];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 与えられた絵文字のリストをデータベースから取得し、キャッシュに追加します
|
|
|
|
|
*/
|
|
|
|
|
export async function prefetchEmojis(emojis: { name: string; host: string | null; }[]): Promise<void> {
|
|
|
|
|
const notCachedEmojis = emojis.filter(emoji => cache.get(`${emoji.name} ${emoji.host}`) == null);
|
|
|
|
|
const emojisQuery: any[] = [];
|
|
|
|
|
const hosts = new Set(notCachedEmojis.map(e => e.host));
|
|
|
|
|
for (const host of hosts) {
|
|
|
|
|
emojisQuery.push({
|
|
|
|
|
name: In(notCachedEmojis.filter(e => e.host === host).map(e => e.name)),
|
2021-11-13 04:10:14 -06:00
|
|
|
|
host: host,
|
2021-03-21 22:41:33 -05:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
const _emojis = emojisQuery.length > 0 ? await Emojis.find({
|
|
|
|
|
where: emojisQuery,
|
2022-01-21 03:47:02 -06:00
|
|
|
|
select: ['name', 'host', 'originalUrl', 'publicUrl'],
|
2021-03-21 22:41:33 -05:00
|
|
|
|
}) : [];
|
|
|
|
|
for (const emoji of _emojis) {
|
|
|
|
|
cache.set(`${emoji.name} ${emoji.host}`, emoji);
|
|
|
|
|
}
|
|
|
|
|
}
|