summaryrefslogtreecommitdiff
path: root/packages
diff options
context:
space:
mode:
authortamaina <tamaina@hotmail.co.jp>2022-12-31 06:18:32 +0000
committertamaina <tamaina@hotmail.co.jp>2022-12-31 06:18:32 +0000
commitaba06b4ef982d33e1adab17901c25cc5ebab9aac (patch)
tree00fbd44cf4be71f48fb1b4a276f950ceac637c92 /packages
parentFix #9438 (diff)
parent:art: (diff)
downloadmisskey-aba06b4ef982d33e1adab17901c25cc5ebab9aac.tar.gz
misskey-aba06b4ef982d33e1adab17901c25cc5ebab9aac.tar.bz2
misskey-aba06b4ef982d33e1adab17901c25cc5ebab9aac.zip
Merge branch 'develop' of https://github.com/misskey-dev/misskey into develop
Diffstat (limited to 'packages')
-rw-r--r--packages/backend/src/core/CustomEmojiService.ts132
-rw-r--r--packages/backend/src/core/entities/EmojiEntityService.ts6
-rw-r--r--packages/backend/src/core/entities/NoteEntityService.ts5
-rw-r--r--packages/backend/src/core/entities/NotificationEntityService.ts4
-rw-r--r--packages/backend/src/core/entities/UserEntityService.ts7
-rw-r--r--packages/backend/src/models/schema/note.ts18
-rw-r--r--packages/backend/src/models/schema/user.ts29
-rw-r--r--packages/backend/src/server/api/endpoints/meta.ts3
-rw-r--r--packages/backend/src/server/web/ClientServerService.ts14
-rw-r--r--packages/frontend/src/components/MkInstanceStats.vue4
-rw-r--r--packages/frontend/src/components/MkReactionsViewer.reaction.vue1
-rw-r--r--packages/frontend/src/components/MkRetentionHeatmap.vue2
-rw-r--r--packages/frontend/src/instance.ts1
-rw-r--r--packages/frontend/src/pages/about.vue2
14 files changed, 19 insertions, 209 deletions
diff --git a/packages/backend/src/core/CustomEmojiService.ts b/packages/backend/src/core/CustomEmojiService.ts
index f0be952612..18b4067f61 100644
--- a/packages/backend/src/core/CustomEmojiService.ts
+++ b/packages/backend/src/core/CustomEmojiService.ts
@@ -1,29 +1,14 @@
import { Inject, Injectable } from '@nestjs/common';
import { DataSource, In, IsNull } from 'typeorm';
-import { GlobalEventService } from '@/core/GlobalEventService.js';
import { DI } from '@/di-symbols.js';
import { IdService } from '@/core/IdService.js';
import type { DriveFile } from '@/models/entities/DriveFile.js';
import type { Emoji } from '@/models/entities/Emoji.js';
-import { Cache } from '@/misc/cache.js';
-import type { Note } from '@/models/entities/Note.js';
import type { EmojisRepository } from '@/models/index.js';
-import { UtilityService } from '@/core/UtilityService.js';
-import { ReactionService } from '@/core/ReactionService.js';
import { bindThis } from '@/decorators.js';
-/**
- * 添付用絵文字情報
- */
-type PopulatedEmoji = {
- name: string;
- url: string;
-};
-
@Injectable()
export class CustomEmojiService {
- private cache: Cache<Emoji | null>;
-
constructor(
@Inject(DI.db)
private db: DataSource,
@@ -32,11 +17,7 @@ export class CustomEmojiService {
private emojisRepository: EmojisRepository,
private idService: IdService,
- private globalEventServie: GlobalEventService,
- private utilityService: UtilityService,
- private reactionService: ReactionService,
) {
- this.cache = new Cache<Emoji | null>(1000 * 60 * 60 * 12);
}
@bindThis
@@ -63,117 +44,4 @@ export class CustomEmojiService {
return emoji;
}
-
- @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
- private 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 = this.utilityService.toPunyNullable(this.normalizeHost(match[2], noteUserHost));
-
- return { name, host };
- }
-
- /**
- * 添付用絵文字情報を解決する
- * @param emojiName ノートやユーザープロフィールに添付された、またはリアクションのカスタム絵文字名 (:は含めない, リアクションでローカルホストの場合は@.を付ける (これはdecodeReactionで可能))
- * @param noteUserHost ノートやユーザープロフィールの所有者のホスト
- * @returns 絵文字情報, nullは未マッチを意味する
- */
- @bindThis
- public async populateEmoji(emojiName: string, noteUserHost: string | null): Promise<PopulatedEmoji | null> {
- const { name, host } = this.parseEmojiStr(emojiName, noteUserHost);
- if (name == 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;
- // || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
- const emojiUrl = emoji.publicUrl || emoji.originalUrl;
- const url = emojiUrl;
-
- return {
- name: emojiName,
- url,
- };
- }
-
- /**
- * 複数の添付用絵文字情報を解決する (キャシュ付き, 存在しないものは結果から除外される)
- */
- @bindThis
- public async populateEmojis(emojiNames: string[], noteUserHost: string | null): Promise<PopulatedEmoji[]> {
- const emojis = await Promise.all(emojiNames.map(x => this.populateEmoji(x, noteUserHost)));
- return emojis.filter((x): x is PopulatedEmoji => x != null);
- }
-
- @bindThis
- public aggregateNoteEmojis(notes: Note[]) {
- let emojis: { name: string | null; host: string | null; }[] = [];
- for (const note of notes) {
- emojis = emojis.concat(note.emojis
- .map(e => this.parseEmojiStr(e, note.userHost)));
- if (note.renote) {
- emojis = emojis.concat(note.renote.emojis
- .map(e => this.parseEmojiStr(e, note.renote!.userHost)));
- if (note.renote.user) {
- emojis = emojis.concat(note.renote.user.emojis
- .map(e => this.parseEmojiStr(e, note.renote!.userHost)));
- }
- }
- const customReactions = Object.keys(note.reactions).map(x => this.reactionService.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 => this.parseEmojiStr(e, note.userHost)));
- }
- }
- return emojis.filter(x => x.name != null) as { name: string; host: string | null; }[];
- }
-
- /**
- * 与えられた絵文字のリストをデータベースから取得し、キャッシュに追加します
- */
- @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) {
- emojisQuery.push({
- name: In(notCachedEmojis.filter(e => e.host === host).map(e => e.name)),
- host: host ?? IsNull(),
- });
- }
- 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);
- }
- }
}
diff --git a/packages/backend/src/core/entities/EmojiEntityService.ts b/packages/backend/src/core/entities/EmojiEntityService.ts
index f1c7ee9035..8a2dc70eda 100644
--- a/packages/backend/src/core/entities/EmojiEntityService.ts
+++ b/packages/backend/src/core/entities/EmojiEntityService.ts
@@ -22,7 +22,6 @@ export class EmojiEntityService {
@bindThis
public async pack(
src: Emoji['id'] | Emoji,
- opts: { omitUrl?: boolean } = {},
): Promise<Packed<'Emoji'>> {
const emoji = typeof src === 'object' ? src : await this.emojisRepository.findOneByOrFail({ id: src });
@@ -32,17 +31,14 @@ export class EmojiEntityService {
name: emoji.name,
category: emoji.category,
host: emoji.host,
- // || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
- url: opts.omitUrl ? undefined : (emoji.publicUrl || emoji.originalUrl),
};
}
@bindThis
public packMany(
emojis: any[],
- opts: { omitUrl?: boolean } = {},
) {
- return Promise.all(emojis.map(x => this.pack(x, opts)));
+ return Promise.all(emojis.map(x => this.pack(x)));
}
}
diff --git a/packages/backend/src/core/entities/NoteEntityService.ts b/packages/backend/src/core/entities/NoteEntityService.ts
index 73d3184957..2b179643f3 100644
--- a/packages/backend/src/core/entities/NoteEntityService.ts
+++ b/packages/backend/src/core/entities/NoteEntityService.ts
@@ -11,12 +11,12 @@ import type { User } from '@/models/entities/User.js';
import type { Note } from '@/models/entities/Note.js';
import type { NoteReaction } from '@/models/entities/NoteReaction.js';
import type { UsersRepository, NotesRepository, FollowingsRepository, PollsRepository, PollVotesRepository, NoteReactionsRepository, ChannelsRepository, DriveFilesRepository } from '@/models/index.js';
+import { bindThis } from '@/decorators.js';
import type { OnModuleInit } from '@nestjs/common';
import type { CustomEmojiService } from '../CustomEmojiService.js';
import type { ReactionService } from '../ReactionService.js';
import type { UserEntityService } from './UserEntityService.js';
import type { DriveFileEntityService } from './DriveFileEntityService.js';
-import { bindThis } from '@/decorators.js';
@Injectable()
export class NoteEntityService implements OnModuleInit {
@@ -300,7 +300,6 @@ export class NoteEntityService implements OnModuleInit {
repliesCount: note.repliesCount,
reactions: this.reactionService.convertLegacyReactions(note.reactions),
tags: note.tags.length > 0 ? note.tags : undefined,
- emojis: this.customEmojiService.populateEmojis(note.emojis.concat(reactionEmojiNames), host),
fileIds: note.fileIds,
files: this.driveFileEntityService.packMany(note.fileIds),
replyId: note.replyId,
@@ -385,8 +384,6 @@ export class NoteEntityService implements OnModuleInit {
}
}
- await this.customEmojiService.prefetchEmojis(this.customEmojiService.aggregateNoteEmojis(notes));
-
return await Promise.all(notes.map(n => this.pack(n, me, {
...options,
_hint_: {
diff --git a/packages/backend/src/core/entities/NotificationEntityService.ts b/packages/backend/src/core/entities/NotificationEntityService.ts
index 346faae6b0..208d653877 100644
--- a/packages/backend/src/core/entities/NotificationEntityService.ts
+++ b/packages/backend/src/core/entities/NotificationEntityService.ts
@@ -8,12 +8,12 @@ import type { Notification } from '@/models/entities/Notification.js';
import type { NoteReaction } from '@/models/entities/NoteReaction.js';
import type { Note } from '@/models/entities/Note.js';
import type { Packed } from '@/misc/schema.js';
+import { bindThis } from '@/decorators.js';
import type { OnModuleInit } from '@nestjs/common';
import type { CustomEmojiService } from '../CustomEmojiService.js';
import type { UserEntityService } from './UserEntityService.js';
import type { NoteEntityService } from './NoteEntityService.js';
import type { UserGroupInvitationEntityService } from './UserGroupInvitationEntityService.js';
-import { bindThis } from '@/decorators.js';
@Injectable()
export class NotificationEntityService implements OnModuleInit {
@@ -143,8 +143,6 @@ export class NotificationEntityService implements OnModuleInit {
myReactionsMap.set(target, myReactions.find(reaction => reaction.noteId === target) ?? null);
}
- await this.customEmojiService.prefetchEmojis(this.customEmojiService.aggregateNoteEmojis(notes));
-
return await Promise.all(notifications.map(x => this.pack(x, {
_hintForEachNotes_: {
myReactions: myReactionsMap,
diff --git a/packages/backend/src/core/entities/UserEntityService.ts b/packages/backend/src/core/entities/UserEntityService.ts
index 4a027d1de6..a123746220 100644
--- a/packages/backend/src/core/entities/UserEntityService.ts
+++ b/packages/backend/src/core/entities/UserEntityService.ts
@@ -392,7 +392,6 @@ export class UserEntityService implements OnModuleInit {
host: user.host,
avatarUrl: this.getAvatarUrlSync(user),
avatarBlurhash: user.avatar?.blurhash ?? null,
- avatarColor: null, // 後方互換性のため
isAdmin: user.isAdmin ?? falsy,
isModerator: user.isModerator ?? falsy,
isBot: user.isBot ?? falsy,
@@ -408,9 +407,7 @@ export class UserEntityService implements OnModuleInit {
faviconUrl: instance.faviconUrl,
themeColor: instance.themeColor,
} : undefined) : undefined,
- emojis: this.customEmojiService.populateEmojis(user.emojis, user.host),
onlineStatus: this.getOnlineStatus(user),
- driveCapacityOverrideMb: user.driveCapacityOverrideMb,
...(opts.detail ? {
url: profile!.url,
@@ -420,7 +417,6 @@ export class UserEntityService implements OnModuleInit {
lastFetchedAt: user.lastFetchedAt ? user.lastFetchedAt.toISOString() : null,
bannerUrl: user.banner ? this.driveFileEntityService.getPublicUrl(user.banner, false) : null,
bannerBlurhash: user.banner?.blurhash ?? null,
- bannerColor: null, // 後方互換性のため
isLocked: user.isLocked,
isSilenced: user.isSilenced ?? falsy,
isSuspended: user.isSuspended ?? falsy,
@@ -447,6 +443,9 @@ export class UserEntityService implements OnModuleInit {
userId: user.id,
}).then(result => result >= 1)
: false,
+ ...(isMe || opts.includeSecrets ? {
+ driveCapacityOverrideMb: user.driveCapacityOverrideMb,
+ } : {}),
} : {}),
...(opts.detail && isMe ? {
diff --git a/packages/backend/src/models/schema/note.ts b/packages/backend/src/models/schema/note.ts
index 7cc70cdeaf..72c0c62285 100644
--- a/packages/backend/src/models/schema/note.ts
+++ b/packages/backend/src/models/schema/note.ts
@@ -141,24 +141,6 @@ export const packedNoteSchema = {
type: 'boolean',
optional: true, nullable: false,
},
- emojis: {
- type: 'array',
- optional: false, nullable: false,
- items: {
- type: 'object',
- optional: false, nullable: false,
- properties: {
- name: {
- type: 'string',
- optional: false, nullable: false,
- },
- url: {
- type: 'string',
- optional: false, nullable: true,
- },
- },
- },
- },
reactions: {
type: 'object',
optional: false, nullable: false,
diff --git a/packages/backend/src/models/schema/user.ts b/packages/backend/src/models/schema/user.ts
index 1c8fe97858..aac5e9332c 100644
--- a/packages/backend/src/models/schema/user.ts
+++ b/packages/backend/src/models/schema/user.ts
@@ -32,11 +32,6 @@ export const packedUserLiteSchema = {
type: 'any',
nullable: true, optional: false,
},
- avatarColor: {
- type: 'any',
- nullable: true, optional: false,
- default: null,
- },
isAdmin: {
type: 'boolean',
nullable: false, optional: true,
@@ -55,25 +50,6 @@ export const packedUserLiteSchema = {
type: 'boolean',
nullable: false, optional: true,
},
- emojis: {
- type: 'array',
- nullable: false, optional: false,
- items: {
- type: 'object',
- nullable: false, optional: false,
- properties: {
- name: {
- type: 'string',
- nullable: false, optional: false,
- },
- url: {
- type: 'string',
- nullable: false, optional: false,
- format: 'url',
- },
- },
- },
- },
onlineStatus: {
type: 'string',
format: 'url',
@@ -120,11 +96,6 @@ export const packedUserDetailedNotMeOnlySchema = {
type: 'any',
nullable: true, optional: false,
},
- bannerColor: {
- type: 'any',
- nullable: true, optional: false,
- default: null,
- },
isLocked: {
type: 'boolean',
nullable: false, optional: false,
diff --git a/packages/backend/src/server/api/endpoints/meta.ts b/packages/backend/src/server/api/endpoints/meta.ts
index 66c9f0620a..05da011979 100644
--- a/packages/backend/src/server/api/endpoints/meta.ts
+++ b/packages/backend/src/server/api/endpoints/meta.ts
@@ -309,7 +309,6 @@ export const paramDef = {
type: 'object',
properties: {
detail: { type: 'boolean', default: true },
- omitEmojiUrl: { type: 'boolean', default: false },
},
required: [],
} as const;
@@ -391,7 +390,7 @@ export default class extends Endpoint<typeof meta, typeof paramDef> {
backgroundImageUrl: instance.backgroundImageUrl,
logoImageUrl: instance.logoImageUrl,
maxNoteTextLength: MAX_NOTE_TEXT_LENGTH, // 後方互換性のため
- emojis: await this.emojiEntityService.packMany(emojis, { omitUrl: ps.omitEmojiUrl }),
+ emojis: await this.emojiEntityService.packMany(emojis),
defaultLightTheme: instance.defaultLightTheme,
defaultDarkTheme: instance.defaultDarkTheme,
ads: ads.map(ad => ({
diff --git a/packages/backend/src/server/web/ClientServerService.ts b/packages/backend/src/server/web/ClientServerService.ts
index 02b779d741..7ef178403b 100644
--- a/packages/backend/src/server/web/ClientServerService.ts
+++ b/packages/backend/src/server/web/ClientServerService.ts
@@ -228,6 +228,8 @@ export class ClientServerService {
return;
}
+ reply.header('Cache-Control', 'public, max-age=86400');
+
const name = path.split('@')[0].replace('.webp', '');
const host = path.split('@')[1]?.replace('.webp', '');
@@ -244,7 +246,7 @@ export class ClientServerService {
reply.header('Content-Security-Policy', 'default-src \'none\'; style-src \'unsafe-inline\'');
- const url = new URL("/proxy/emoji.webp", this.config.url);
+ const url = new URL('/proxy/emoji.webp', this.config.url);
// || emoji.originalUrl してるのは後方互換性のため(publicUrlはstringなので??はだめ)
url.searchParams.set('url', emoji.publicUrl || emoji.originalUrl);
url.searchParams.set('emoji', '1');
@@ -347,15 +349,15 @@ export class ClientServerService {
fastify.get('/opensearch.xml', async (request, reply) => {
const meta = await this.metaService.fetch();
- const name = meta.name || "Misskey";
- let content = "";
- content += `<OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/" xmlns:moz="http://www.mozilla.org/2006/browser/search/">`;
+ const name = meta.name || 'Misskey';
+ let content = '';
+ content += '<OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/" xmlns:moz="http://www.mozilla.org/2006/browser/search/">';
content += `<ShortName>${name} Search</ShortName>`;
content += `<Description>${name} Search</Description>`;
- content += `<InputEncoding>UTF-8</InputEncoding>`;
+ content += '<InputEncoding>UTF-8</InputEncoding>';
content += `<Image width="16" height="16" type="image/x-icon">${this.config.url}/favicon.ico</Image>`;
content += `<Url type="text/html" template="${this.config.url}/search?q={searchTerms}"/>`;
- content += `</OpenSearchDescription>`;
+ content += '</OpenSearchDescription>';
reply.header('Content-Type', 'application/opensearchdescription+xml');
return await reply.send(content);
diff --git a/packages/frontend/src/components/MkInstanceStats.vue b/packages/frontend/src/components/MkInstanceStats.vue
index 3500a340a8..382aaf16ef 100644
--- a/packages/frontend/src/components/MkInstanceStats.vue
+++ b/packages/frontend/src/components/MkInstanceStats.vue
@@ -31,7 +31,7 @@
</MkSelect>
</div>
<div class="chart _panel">
- <MkChart :src="chartSrc" :span="chartSpan" :limit="chartLimit" :detailed="detailed"></MkChart>
+ <MkChart :src="chartSrc" :span="chartSpan" :limit="chartLimit" :detailed="true"></MkChart>
</div>
</div>
</MkFolder>
@@ -122,7 +122,7 @@ Chart.register(
Filler,
);
-const chartLimit = 90;
+const chartLimit = 500;
let chartSpan = $ref<'hour' | 'day'>('hour');
let chartSrc = $ref('active-users');
let heatmapSrc = $ref('active-users');
diff --git a/packages/frontend/src/components/MkReactionsViewer.reaction.vue b/packages/frontend/src/components/MkReactionsViewer.reaction.vue
index 8f2f48dcd7..f3c77231db 100644
--- a/packages/frontend/src/components/MkReactionsViewer.reaction.vue
+++ b/packages/frontend/src/components/MkReactionsViewer.reaction.vue
@@ -81,7 +81,6 @@ useTooltip(buttonRef, async (showing) => {
os.popup(XDetails, {
showing,
reaction: props.reaction,
- emojis: props.note.emojis,
users,
count: props.count,
targetElement: buttonRef.value,
diff --git a/packages/frontend/src/components/MkRetentionHeatmap.vue b/packages/frontend/src/components/MkRetentionHeatmap.vue
index f8ae401faa..547fe70a8c 100644
--- a/packages/frontend/src/components/MkRetentionHeatmap.vue
+++ b/packages/frontend/src/components/MkRetentionHeatmap.vue
@@ -72,7 +72,7 @@ async function renderChart() {
const wide = rootEl.offsetWidth > 600;
const narrow = rootEl.offsetWidth < 400;
- const maxDays = wide ? 20 : narrow ? 10 : 15;
+ const maxDays = wide ? 20 : narrow ? 7 : 14;
const raw = await os.api('retention', { });
diff --git a/packages/frontend/src/instance.ts b/packages/frontend/src/instance.ts
index c6fd1756a5..51464f32fb 100644
--- a/packages/frontend/src/instance.ts
+++ b/packages/frontend/src/instance.ts
@@ -15,7 +15,6 @@ export const instance: Misskey.entities.InstanceMetadata = reactive(instanceData
export async function fetchInstance() {
const meta = await api('meta', {
detail: false,
- omitEmojiUrl: true,
});
for (const [k, v] of Object.entries(meta)) {
diff --git a/packages/frontend/src/pages/about.vue b/packages/frontend/src/pages/about.vue
index 0ed692c5c5..5091114c21 100644
--- a/packages/frontend/src/pages/about.vue
+++ b/packages/frontend/src/pages/about.vue
@@ -76,7 +76,7 @@
<XFederation/>
</MkSpacer>
<MkSpacer v-else-if="tab === 'charts'" :content-max="1000" :margin-min="20">
- <MkInstanceStats :chart-limit="500" :detailed="true"/>
+ <MkInstanceStats/>
</MkSpacer>
</MkStickyContainer>
</template>